Angularjs2 - предварительно загрузить конфигурацию сервера перед запуском приложения

Я работаю с Angular2 RC5. Я хотел бы иметь возможность загружать IP-адрес своего сервера и некоторые другие параметры конфигурации до загрузки моего приложения. Как я могу сделать это в последней версии RC5?

Я видел другие статьи, но они не помогают:

  1. Как предварительно загрузить файл конфигурации в angular2: Ответ, приведенный здесь, относится к веб-пакету. Я использую systemjs.
  2. https://medium.com/@hasan.hameed/reading-configuration-files-in-angular-2-9d18b7a6aa4: это для более старой версии Angular2. Некоторые из используемых классов устарели.

Любая помощь будет принята с благодарностью.

редактировать

Я попытался использовать APP_INITIALIZER следующим образом в моем app.module.ts:

    import { NgModule, provide, APP_INITIALIZER }       from "@angular/core";
    import { ConfigService } from "./shared/services/config.service";
    @NgModule({
        declarations: [AppComponent],
        imports: [BrowserModule,
            routes,
            FormsModule,
            HttpModule],
        providers: [AuthService,
            Title,
            appRoutingProviders,
            ConfigService],
        bootstrap: [AppComponent
            , provide(APP_INITIALIZER,
                {
                    useFactory: (config: ConfigService) => () => config.load(),
                    deps: [ConfigService], multi: true
                })]
    })
    export class AppModule { }

Это мой файл config.service.ts:

    import { Config } from "../model/config";

    export class ConfigService {
        constructor() {
        }

        load(): Promise<Config> {
            let config: Config = new Config("localhost", "5050");
            return Promise.resolve(config);
        }
    }

Обратите внимание, что в конечном итоге я перезаписываю load, чтобы фактически прочитать свойство или какой-то такой файл для загрузки конфигурации.

Вот как выглядит config.ts:

export class Config {

    constructor(
        private machine: string,
        private port: string
    ) {
    }

    private base_url: string = "http://" + this.machine +
    this.port + "/";

    public loginURL: string = this.base_url + "login";
}

Я получаю ошибку компиляции

public/app/app.module.ts(27,11): error TS2345: Argument of type '{ declarations: (typeof AppComponent ...' is not assignable to parameter of type 'NgModuleMetadataType'.
  Types of property 'bootstrap' are incompatible.
    Type '(typeof AppComponent | Provider)[]' is not assignable to type '(Type | any[])[]'.
      Type 'typeof AppComponent | Provider' is not assignable to type 'Type | any[]'.
        Type 'Provider' is not assignable to type 'Type | any[]'.
          Type 'Provider' is not assignable to type 'any[]'.
            Property '[Symbol.iterator]' is missing in type 'Provider'.

Редактировать 2

Я перехожу к следующему шагу. Обновите ConfigService для чтения файла json с помощью http.get. Теперь я получаю ошибку. Вот обновленные файлы:

export class ConfigService {
    private config: Config;
        constructor(private http: Http) {
    }

    load(): Promise<Config> {
            this.http.get("/blah/config.json")
                 .map(res => res.json())
                 .subscribe((env_data) => {
                     console.log(env_data);
                  });
        this.config = new Config("localhost", "5050");
        return Promise.resolve(this.config);
    }

    getConfig(): Config {
        return this.config;
    }
}

Вот раздел провайдеров в классе NgModule

    providers: [AuthService,
    Title,
    appRoutingProviders,
    ConfigService,
    {
        provide: APP_INITIALIZER,
        useFactory: (config: ConfigService) => () => config.load(),
        deps: [ConfigService, Http],
        multi: true
    }],

Обратите внимание, что в RC6 я не могу импортировать HTTP_PROVIDERS, так как я получаю сообщение об ошибке, что они больше не определены в модуле http. Вот почему я попробовал Http.

Во время выполнения я получаю следующую ошибку

(index):27 Error: Error: Can't resolve all parameters for ConfigService: (?).(…)(anonymous function) @ (index):27
ZoneDelegate.invoke @ zone.js:332
Zone.run @ zone.js:225(anonymous function) @ zone.js:591
ZoneDelegate.invokeTask @ zone.js:365
Zone.runTask @ zone.js:265
drainMicroTaskQueue @ zone.js:497
ZoneTask.invoke @ zone.js:437

1 ответ

Решение

Предоставьте это в вашем корневом модуле

{provide: APP_INITIALIZER, useValue: () => promise, multi: true}]}

где () => promise это некоторый вызов, который возвращает Обещание, которое разрешается до желаемого значения.

Смотрите также Как передать параметры, отрисованные из бэкэнда, в метод начальной загрузки angular2

Вы можете передать сервис как зависимость, где вы можете сохранить результат.

Обновить

export function loadConfig(config: ConfigService) => () => config.load()

@NgModule({
        declarations: [AppComponent],
        imports: [BrowserModule,
            routes,
            FormsModule,
            HttpModule],
        providers: [AuthService,
            Title,
            appRoutingProviders,
            ConfigService,
            { provide: APP_INITIALIZER,
              useFactory: loadConfig,
              deps: [ConfigService], 
              multi: true }
        ],
        bootstrap: [AppComponent]
    })
    export class AppModule { }

обновление 2

Пример плунжера

Другие вопросы по тегам