Skip to content
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

Do Not Deprecate baseHref in i18n Configuration #29396

Closed
jusfeel opened this issue Jan 18, 2025 · 7 comments · Fixed by #29429
Closed

Do Not Deprecate baseHref in i18n Configuration #29396

jusfeel opened this issue Jan 18, 2025 · 7 comments · Fixed by #29429

Comments

@jusfeel
Copy link

jusfeel commented Jan 18, 2025

Which @angular/* package(s) are relevant/related to the feature request?

localize

Description:

The recent deprecation of the baseHref field in the i18n.sourceLocale configuration causes significant issues for projects that rely on a shared base URL for multiple localized builds.

While the subPath feature is a welcome addition for structuring locale builds, it is limiting for developers who:

  1. Want locale-specific builds in subdirectories (subPath) without altering the <base> tag in index.html.
  2. Depend on a static baseHref for all builds due to server or application constraints.

Current Behavior:

With subPath enabled:

Angular modifies the <base href> tag in index.html to include the locale-specific subpath (e.g., /zh-hans/), and there is no built-in way to override this behavior.
Without baseHref:

Developers are forced to implement error-prone and maintenance-heavy post-build scripts to reset the <base> tag to their desired value.

Expected Behavior:

Projects with subPath configured should still have the option to:

Use baseHref as a static value across all locale builds.
Separate the functionality of subPath (output directory) from the <base> tag in index.html.

Proposed solution

Instead of fully deprecating baseHref, it could coexist with subPath to give developers more flexibility. Suggested behavior:

  1. baseHref takes precedence if defined. This ensures that:

    • The <base> tag in index.html uses baseHref for all builds.
    • subPath is only used for structuring the output directory, not modifying index.html.
  2. If baseHref is not specified, the current subPath behavior remains unchanged.

Alternatives considered

Developers are forced to implement error-prone and maintenance-heavy post-build scripts to reset the tag to their desired value.

@JeanMeche JeanMeche transferred this issue from angular/angular Jan 18, 2025
@alan-agius4
Copy link
Collaborator

Can you please share your i18n configure?

@jusfeel
Copy link
Author

jusfeel commented Jan 19, 2025

      "i18n": {
        "sourceLocale": {
            "code": "en",
            "baseHref": ""
        },
        "locales": {
          "zh-hans": {
            "translation": "src/locale/messages.zh-hans.xlf",
            "baseHref": ""
          }
        }
      }
<base href="/">

@zip-fa
Copy link

zip-fa commented Jan 19, 2025

He is probably talking about two different domains, eg:

https://my-cool-app.com
https://zh-audience-app.cn

Where single app is served and / url is different locale.
But this only possible when:

  1. Developer moves en/zh locale from directory to / with custom script
  2. I don't understand how SSR will work in that case; probably manifest will be generated with two '' entries

So i am pretty unsure on this issue. Probably moving i18n to configuration will be more suitable

@anisabboud
Copy link

anisabboud commented Jan 19, 2025

I'm facing a similar issue/dilemma trying to transition to subPath.

Consider the following angular.json configuration:

      "i18n": {  // Define i18n locales - https://angular.dev/guide/i18n/merge.
        "sourceLocale": { "code": "en", "baseHref": "" },  // English primary language (default /).
        "locales": {
          "de": "src/i18n/messages.de.xlf",  // German secondary language (<base href="/de/">).
          "es-419": "src/i18n/messages.es-419.xlf",  // Spanish secondary language (<base href="/es/">).
          "fr": "src/i18n/messages.fr.xlf",  // French secondary language (<base href="/fr/">).
          "it": "src/i18n/messages.it.xlf",  // Italian secondary language (<base href="/it/">).

          "de-base": { "translation": "src/i18n/messages.de.xlf", "baseHref": "" },  // German customer (default /).
          "fr-base": { "translation": "src/i18n/messages.fr.xlf", "baseHref": "" },  // French customer (default /).
          "it-base": { "translation": "src/i18n/messages.it.xlf", "baseHref": "" },  // Italian customer (default /).

          "en-seco": { "translation": "src/i18n/messages.xlf", "baseHref": "en" }  // Non-english primary with English secondary (<base href="/en/">).
        }
      },

This configuration provides the flexibility to create different multi-language builds for different customers. E.g.,

          "configurations": {
            "us": { "localize": ["en", "es"] },  // American customer - English by default, with ability to switch language to Spanish.
            "ch": { "localize": ["de-base", "fr", "it", "en-seco"] },  // Swiss customer - German by default, with ability to switch language to French, Italian, or English as secondary languages.
            "it": { "localize": ["it"] },  // Italian customer - Italian only.
          }

I know this might be pushing the limits of what's possible in the current configuration, but since your goal is to improve the configuration, it would be great if the case above could be handled/supported (or at least not be rendered impossible).
With the new update (deprecating baseHref in favor of subPath), ng serve no longer works:
Error: Invalid i18n configuration: Locales 'en' and 'de-base' cannot have the same subPath: ''.


Suggestion

Perhaps subPath should be configured/overridden under the "localize" array, and not under "i18n" > "locales"?

"localize" would take an array of locales (just like now), and the first locale in the array would be considered the default (baseHref "/") while the rest of the locales in the array would have their subPath set to the locale code by default, unless specifically overridden.

In such case, the configuration above could be simplified as follows:

      "i18n": {
        "sourceLocale": "en",  // English.
        "locales": {
          "de": "src/i18n/messages.de.xlf",  // German.
          "es-419": "src/i18n/messages.es-419.xlf",  // Spanish.
          "fr": "src/i18n/messages.fr.xlf",  // French.
          "it": "src/i18n/messages.it.xlf",  // Italian.
        }
      },
          "configurations": {
            "us": { "localize": ["en", {"locale": "es-419", "subPath": "es"}] },  // "en" would be at baseHref "/". Example of overriding subPath for Spanish to "/es/".
            "ch": { "localize": ["de", "fr", "it", "en"] },  // "de" would be at baseHref "/", while the other languages would be at "/fr/", "/it/", "/en/".
            "it": { "localize": ["it"] },  // "it" would be at baseHref "/" in this build.
          }

This would also solve a minor issue we currently have with the previous setup, where we define another locale (e.g., it-base) in addition to the standard locale (e.g., it) to support both cases of Italian as the primary language (baseHref "/") as well as Italian as a secondary language (baseHref "/it/"): When we inject(LOCALE_ID), it injects "it-base" instead of "it", and similarly in the HTML: <html lang="it-base">. This is behaving "OK", but it's not ideal.
The suggested configuration (moving "subPath" in the schema from "locales" to "localize") would solve this as well.

@alan-agius4
Copy link
Collaborator

@jusfeel, just to confirm—you're not building both locales that share the same baseHref/subPath simultaneously, correct?

alan-agius4 added a commit to alan-agius4/angular-cli that referenced this issue Jan 20, 2025
…te `subPaths` with inlined locales

The i18n configuration validation was incorrectly flagging errors for identical `subPaths` when both locales were not inlined within the same build. This was due to the validation not properly accounting for the inlining of locales.

This commit fixes this issue by ensuring that the validation only checks for duplicate `subPaths` when the locales are inlined.

Closes angular#29396 and closes angular#29398
alan-agius4 added a commit to alan-agius4/angular-cli that referenced this issue Jan 20, 2025
…te `subPaths` with inlined locales

The i18n configuration validation was incorrectly flagging errors for identical `subPaths` when both locales were not inlined within the same build. This was due to the validation not properly accounting for the inlining of locales.

This commit fixes this issue by ensuring that the validation only checks for duplicate `subPaths` when the locales are inlined.

Closes angular#29396 and closes angular#29398
@jusfeel
Copy link
Author

jusfeel commented Jan 21, 2025

@jusfeel, just to confirm—you're not building both locales that share the same baseHref/subPath simultaneously, correct?

I build them into 2 subfolders with the same base in the index.html

This is the config:

     "architect": {
          "build": {
               "options": {              
                  "baseHref": "/da/",
               }
          }
    }
     ...
      "i18n": {
        "sourceLocale": {
            "code": "en",
            "baseHref": ""
        },
        "locales": {
          "zh-hans": {
            "translation": "src/locale/messages.zh-hans.xlf",
            "baseHref": ""
          }
        }
      }

The final build under dist/browser/ is:

  • 2 folders: en, zh-hans.
  • base in the index.html under both folder are the same: build.options.baseHref. <base href="/da/">

alan-agius4 added a commit to alan-agius4/angular-cli that referenced this issue Jan 21, 2025
In certain scenarios, users build applications with the same `baseHref` when using i18n, primarily for deploying localized applications across multiple domains. To address this, we are removing the deprecation of `i18n.baseHref` and will revisit potential options as part of angular#29111

Instead of deprecating `i18n.baseHref`, we now issue a warning when it is used with SSR, as this may lead to undefined behavior.

Closes angular#29396
alan-agius4 added a commit to alan-agius4/angular-cli that referenced this issue Jan 21, 2025
…rning

In certain scenarios, users build applications with the same `baseHref` when using i18n, primarily for deploying localized applications across multiple domains. To address this, we are removing the deprecation of `i18n.baseHref` and will revisit potential options as part of angular#29111

Instead of deprecating `i18n.baseHref`, we now issue a warning when it is used with SSR, as this may lead to undefined behavior.

Closes angular#29396
alan-agius4 added a commit that referenced this issue Jan 21, 2025
…rning

In certain scenarios, users build applications with the same `baseHref` when using i18n, primarily for deploying localized applications across multiple domains. To address this, we are removing the deprecation of `i18n.baseHref` and will revisit potential options as part of #29111

Instead of deprecating `i18n.baseHref`, we now issue a warning when it is used with SSR, as this may lead to undefined behavior.

Closes #29396

(cherry picked from commit 8535c11)
@angular-automatic-lock-bot
Copy link

This issue has been automatically locked due to inactivity.
Please file a new issue if you are encountering a similar or related problem.

Read more about our automatic conversation locking policy.

This action has been performed automatically by a bot.

@angular-automatic-lock-bot angular-automatic-lock-bot bot locked and limited conversation to collaborators Feb 21, 2025
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.