-
Notifications
You must be signed in to change notification settings - Fork 12k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
localized components containing i18n tags will produce identical chunk hashes when building with --localize #17416
Comments
Duplicate of #16526 |
Seems like it was supposed to be fixed with #16817 but I am having the same issue with |
I'm also having this issue. My repro: https://github.com/pjm0616/pastebin/tree/202004-angular-localize-outputhash
|
#16526 was never addressed and was closed without resolution. #16817 fixes a separate issue, where the hashes would remain the same when rebuilding after changing translations and not changing code. |
I agree with @bridzius that #16817 fix only a symptom of the more deep issue of #16526. I also have the same hash for multiple (Side note, I think that, even if I'm not 100% sure, it mess with Angular service worker itself when I switch between translations of my app). @clydin in #16526 (comment) you say I quote:
It seems like the right things to do to fix this issue. (@nas90 thanks for (re-)opening this issue!) |
@alan-agius4 any updates on this one? |
Recently migrated to Angular 9 and faced this issue. Was really disappointing to see this. Even the different configurations for different languages (for example |
Just tried on latest Angular (10.0.5) and I'm still facing this issue 😢 |
This issue is still open hence it hasn’t been addressed yet. While this is on our radar, there are issues which have a higher priority. PRs to implement this feature are welcome. |
I am having the same issue with angular 11.1.0. |
I confirm this is still an issue in v13, the workaround to build the different locales separately can become faster using |
As a different workaround that doesn't involve running a separate build for each language, I wrote a shell script that adds language codes to the file hashes and their occurrences after the build finishes. This ensures uniqueness between different languages. There is probably a more efficient way of doing this and I haven't tested it on different setups, but here it is in case it helps anyone: Edit May 6, 2022: Improve performance and make compatible with macOS locale-utils/fix-hashes.sh #!/bin/bash -eu
# Appends language code to file hashes for js files
# (and references to those files) in dist/
# Motivation: https://github.com/angular/angular-cli/issues/17416
shopt -s extglob nullglob
# https://stackoverflow.com/questions/19242275/re-error-illegal-byte-sequence-on-mac-os-x
LC_CTYPE=C
# https://stackoverflow.com/a/40777793
suffixArg=()
sed --version 2>/dev/null | grep -q GNU || suffixArg=( '' )
cd ./dist/
LOCALES=($(ls))
cd "./${LOCALES[0]}"
# Find js files | extract hash | filter duplicates
# Hashes are the same for every language (this is the problem we are trying to solve)
# So we only need to do this once.
HASHES=($(find . -maxdepth 1 -name '*.js' | sed -E 's|.*\.([a-z0-9]*)\.js|\1|' | sort -u))
cd ..
for LOCALE in ${LOCALES[@]}
do
echo "Updating file hashes for $LOCALE"
cd "./${LOCALE}"
# Replace occurrences of each hash.
for HASH in ${HASHES[@]}
do
HASH_SUFFIXED=${HASH}_${LOCALE}
# echo "Replacing occurrences of hash: $HASH with $HASH_SUFFIXED"
sed -i "${suffixArg[@]}" -E "s|$HASH|$HASH_SUFFIXED|g" *.@(js|html)
done
# Rename files to update their hashes too.
for FILE in *.js*
do
# echo "Renaming $FILE"
mv "$FILE" "${FILE/.js/_${LOCALE}.js}"
done
cd ..
done I added it as a postbuild script in package.json so it runs automatically after each build:
|
I want to point out that this won't fix the problem of source maps not working properly |
Have the same problem with Angular 13 and didn't find a workaround. No matter what I do I always end up with different main-bundles that have the same hash.. |
Is there is an update on this issue? Having the same hash for different file contents is counter-intuituve, especially as this breaks cache busting, and make it impossible to do a location.reload() when the language is set by a cookie, which is the case for Firebase. This issue makes it impossible to deploy Angular localized apps on Firebase |
The way I solved this was to use a post build step to change the file names and the references. I serve with nginx and use cookies to select language. I'm considering a write-up of my solution if there is some interest. |
That would be great. Could the locale be appended to the filename, as a quick fix, basically producing |
I have updated the workaround script in my previous comment to improve performance and provide support for macOS. |
Looking over the comments quickly, it sounds like most of the motivation here comes down to wanting to put multiple locale outputs into a single flat directory, is that the ask here? Why exactly do folks want to do this, does it make some server configuration or URL pathing easier? I think a better understanding of the motivation here would help us identify the right solution. |
For our projects we do path based routing in our ingress, so we opted to do localization with other methods (ie cookies). Without different hashing on the localized files, the browser cache will return the same locale even though the user switched locales, and a different index file is being served. We dont need to flatten anything. Our server-side picks up the cookie and serves the right files. |
That sounds like the I can see that marking those URLs as uncacheable could reduce performance for users who don't change languages, but I think that's a consequence of not putting the locale in the URL. Using a different hash is just another form of making the URL locale-dependent so the locale is taken into account in the browser cache key. Whether that's done via a locale-dependent hash, or an explicit locale path / query parameter is an implementation detail to the browser cache. |
I don't think this is related to caching at all. You can reproduce it like explained above with just a regular "ng build --translate" - you end with 2 folders, one for each locale - so far so good! But if you deep-dive it you notice that your main.js bundles (you have one of those for each language) is messed up. You end up with different bundles because there are differences due to language. But the HASH of all these main.ts bundles is the same. Now depending on your solution you might end up with rewrites that (for the users browser) appear to be from the same path - that would be no problem, but since your ENGLISH-main.ts and your GERMAN-main.ts have the same hash you end up with issues when you switch the language. Given, there may be ways to host it differently - but still: how could that NOT be a bug? The Hash of my german main.ts file is obviously based on the content of my english main.ts file. That's just wrong, it should be based on the content of my german main.ts, right? |
As I understand it, I believe we use Webpack's Given that generated resources shouldn't be loaded into the same page (you shouldn't be loading a French chunk into a German application), then I don't think it's necessarily wrong to see duplicate hashes across different locales. |
Thanks for explaining and the SO link! I think I understand, but it still feels wrong and there might be some bugs with that implementation - if I build my project while only changing the german translation file the hash would still be unchanged, right? That would lead to people not getting the newest version of my german main.ts file! So basically only changing translations without touching a source file is not supported with the current build? The question is: is there a good reason NOT to use [contenthash] or maybe add an optional flag for that? This puts a strong opinion/restriction on how files are served. For me it also seems a bit also unintuitive (because of what I believe a file-hash should do) Examplewill try to explain my approach while hosting my mini-homepage https://budisoft.at that lead to this problem:
This was a naive - but I think reasonable - approach and would work perfectly with [contenthash], but fails with [hash] (wrong language bundle get's loaded from cache) SolutionThanks to your response I think I finally understand what I'd have to do instead of dirty workaround:
I will implement it that way, but still hope that there can be a solution that enables some flexibility while serving the bundles sometime. |
Thanks for following up! I'll try to elaborate the use case in our application, please let me know if my explanation is not clear. Our application is typically gated by a user login. To determine the language of the application, it will look at a user's language preferences, saved in their user account. This ensures that they are served the correct language wherever they log in. If the user is not yet logged in, or the user's language is not yet set, there is logic to determine the most preferable fallback language based on the environment in which the application is deployed, as well as the user's Accept-Language list. Once the correct language is served, we store it in the browser's cookies, ensuring that these checks and redirects do not happen again until a different user - with a different preferred language - signs in. Because of this process, we do not use the browser URL to determine which language should be served. This does not mean that we can't have the language in the URL - we could determine the user's preferred language and then redirect to the correct URL once we know, this is true. However, there are other features of the application that complicate this:
While none of these individual problems might be insurmountable, it makes much more sense for our use case to simply not use the browser URL for the language. It is there that we run into this issue - switching languages is unreliable when the file hashes for each language are the same, when these files are served with the same URL. We currently use the workaround I mentioned in an earlier comment of adding language codes to file hashes, but this is somewhat slow and likely to break in the future. |
Hey Bart, I think you are a bit in the same "mental trap" as I was. Yes, URLs to a specific content-page should not contain a language url, to ensure deep-linkability across languages. But this is only important for your own pages/routes! This doesn't matter for JavaScript bundles though. Let me explain: If you give the link above to a german user your reverse-proxy will probably use your cookie (+ fallback) to determine the correct index.html and retrieve it from a file system like this: In each index.html you have <script> tags, linking to your bundles. And these links could absolutely contain the language, because once the user has the correct index.html does it really matter how he loads the bundle? Sadly it seems as of now you NEED to use the second variant though. You end up with two main.03b15c0004fb5dfc.js files (same hash, but different content), but since you load them from different URLs you don't have caching issues. Hope this is helpful |
This issue causes problems when we upload source maps to Sentry. We have to create a release for each Language in Sentry. This then breaks resolved in the next release working correctly. |
Hey, I just rewoved the script-workaround (thank you for that @bartcorremans !) from my homepage build https://budisoft.at and now everything is working fine. What I did was completely removing
I changed my rewrite-rules to only rewrite for my index.html - all other urls/files can be loaded as-is Now my bundles (e.g. main.1234567.js) are loaded from different URLs and it doesn't matter any longer that they have the same filename. One small issue I had with that, was that routing did include the base-href at first. (e.g. routing to /articles would bring me to https://budisoft.at/en_US/articles). After consulting the documentation I added a provider in my app-module to bring the context back to /: With that everything seems to work as expected and I am happy |
The hash probably should be dependent on the translation files, as I think it's reasonable to expect these hashes to update on any text content changes. If that's not the case, then that sounds like a separate issue we can look at. Since all the hashes are currently tied together across locales, that would mean that your French build is actually dependent on all the other translations, which is a bit weird and bad for build caching. It would make more sense that the French build only depends on French translations. @alan-agius4 would there be any build improvements from better isolating languages this way? In theory I think it would be better, but in practice everything is done via a Webpack compilation which probably preprocesses everything for this hash, so I'm not sure this would actually have a meaningful effect?
Yes, I think that is the right approach. If you want "language-agnostic" URLs, those should be limited to the URLs actually visible to users (such as HTML pages). JS bundles aren't typically user-visible or stable (hashes change on each deployment), so putting them in a locale-specific URL doesn't negatively impact the user. This feels like it also addresses @bartcorremans' issue.
I'm not very familiar with Sentry, but it sounds like they're keying sourcemaps by file name? If so, that makes it a subset of the flattening locale resources use case. Given that Sentry has direct Angular support I wonder if that might be better addressed on their end? Seems to me like a sourcemapping utility should be able to handle multiple files with the same name in different directories, but I'm not an expert in that area and I'm sure there's more nuance to it than that.
I agree that this can be unintuitive. Devs have a strong bias to interpret a hash in a file name as a hash of that file's contents. That's not always true (as in this case), but it is often the initial expectation. One other advantage to consider about a I think the main reason not to change the current behavior is that Webpack chunking happens before localization as I understand it. So just changing to |
Hi matrium0, Thank you for the thoughtful reply! After some tinkering I was able to get our setup to work this way. The main struggle came from angular-router redirecting us to the baseHref, as you mentioned in your later comment - which I wrongly thought was being caused by our server configuration. After figuring that out it was pretty straightforward! Thank you again! This sees our use case covered without the need for unique file hashes per locale. |
Following up with this conversation, really appreciate everyone's feedback here. It looks like we've mostly figured out how to meet these needs with the CLI's current behavior. I think the recommendation for anyone finding this issue in the future:
I agree this behavior can be confusing, and in a perfect world asset hashes would probably be locale dependent. However, there is a lot of complexity and nuance to make that work and maintain it over time. Given that this can be worked around, I don't think changing this is the best use of our limited time and energy. So I'll close this issue for now, and we can reevaluate if it becomes a bigger challenge in the future. |
This issue has been automatically locked due to inactivity. Read more about our automatic conversation locking policy. This action has been performed automatically by a bot. |
🐞 Bug report
Command (mark with an
x
)Description
I have an application that is being translated in several (21 at the moment) languages using the Angular i18n mechanism. I'm compiling the app AOT, therefore I need a separate build for each locale. To achieve this I setup my Azure pipeline to build the app using multiple agents in parallel, one for each locale, and it works like a charm. However, this is very resource-consuming, so I'm updating to Angular 9 in order to use the
--localize
option, which perfectly fits my needs.After the update I can build the app once and translate it to all languages in a few seconds, which is amazing.
I am now facing this issue: when I build my application for all locales, the resulting chunk hashes are identical. As a result, browser cache is not invalidated and files are not re-requested, so the app is not translated.
🔬 Minimal Reproduction
npm install
ng build --localize
main.js
is different for each locale (template text is translated), while the hashes are the same🌍 Your Environment
Anything else relevant?
In my scenario, the user language preference is stored in the cookies, and a shell app serves the corresponding folder. I can't add the locale to the URL.
Therefore in my
angular.json
I setbaseHref="/"
for all locales.I am aware of this issue and the PR mentioned there, however the issue is still not resolved.
The text was updated successfully, but these errors were encountered: