Не удалось запустить NPM при публикации в IIS

Я публикую свое веб-приложение asp.net core 2.2 Angular 7 в IIS и получаю следующие ошибки:

  1. Убедитесь, что 'npm' установлен и находится в одном из каталогов PATH.
  2. InvalidOperationException: не удалось запустить npm.
  3. AggregateException: произошла одна или несколько ошибок. (Произошла одна или несколько ошибок. (Не удалось запустить 'npm'. Чтобы решить эту проблему :. [1] Убедитесь, что 'npm' установлен и находится в одном из каталогов PATH).

Это для основного веб-приложения Asp.Net на сервере Windows 2012 R2 под управлением IIS 8. На моем локальном компьютере с IIS 8 такие же ошибки.


Мой профиль launchsettings.json:

   "Test.IIS": {
        "commandName": "IIS",
        "launchBrowser": true,
        "environmentVariables": {
        "ASPNETCORE_ENVIRONMENT": "Development"
   },
   "ancmHostingModel": "InProcess",
   "applicationUrl": "ttps://localhost:5002"
   }

Ниже мой файл startup.cs:

    public void ConfigureServices(IServiceCollection services)
    {
        services.AddMvc().SetCompatibilityVersion(CompatibilityVersion.Version_2_2);

        // In production, the Angular files will be served from this directory
        services.AddSpaStaticFiles(configuration =>
        {
            configuration.RootPath = "ClientApp/dist";
        });
    }


    public void Configure(IApplicationBuilder app, IHostingEnvironment env)
    {
        if (env.IsDevelopment())
        {
            app.UseDeveloperExceptionPage();
        }
        else
        {
            app.UseExceptionHandler("/Error");
            // The default HSTS value is 30 days. You may want to change this for production scenarios, see https://aka.ms/aspnetcore-hsts.
            app.UseHsts();
        }

        //To check which process is runing the application
        //app.Run(async (contenxt) => {
        //    await contenxt.Response.WriteAsync(System.Diagnostics.Process.GetCurrentProcess().ProcessName);
        //});

        app.UseHttpsRedirection();
        app.UseStaticFiles();
        app.UseSpaStaticFiles();

        app.UseMvc(routes =>
        {
            routes.MapRoute(
                name: "default",
                template: "{controller}/{action=Index}/{id?}");
        });

        app.UseSpa(spa =>
        {
            // To learn more about options for serving an Angular SPA from ASP.NET Core,
            // see https://go.microsoft.com/fwlink/?linkid=864501

            spa.Options.SourcePath = "ClientApp";

            if (env.IsDevelopment())
            {
                spa.UseAngularCliServer(npmScript: "start");
            }
        });
    }

Код успешно компилируется, но выдает ошибки во время выполнения.


person Shankar Naru    schedule 23.01.2019    source источник


Ответы (1)


Это похоже на известную проблему в asp.net core 2.2, если вы приняли модель внутрипроцессного хостинга IIS.

Проблема задокументирована здесь: https://github.com/aspnet/AspNetCore/issues/5263, который связан с другой проблемой github здесь: https://github.com/aspnet/AspNetCore/issues/4206

Пока разрабатывается патч, обходной путь заключается в добавлении следующего помощника (при необходимости измените пространство имен):

using System;

namespace SampleApp
{
    internal class CurrentDirectoryHelpers
    {
        internal const string AspNetCoreModuleDll = "aspnetcorev2_inprocess.dll";

        [System.Runtime.InteropServices.DllImport("kernel32.dll")]
        private static extern IntPtr GetModuleHandle(string lpModuleName);

        [System.Runtime.InteropServices.DllImport(AspNetCoreModuleDll)]
        private static extern int http_get_application_properties(ref IISConfigurationData iiConfigData);

        [System.Runtime.InteropServices.StructLayout(System.Runtime.InteropServices.LayoutKind.Sequential)]
        private struct IISConfigurationData
        {
            public IntPtr pNativeApplication;
            [System.Runtime.InteropServices.MarshalAs(System.Runtime.InteropServices.UnmanagedType.BStr)]
            public string pwzFullApplicationPath;
            [System.Runtime.InteropServices.MarshalAs(System.Runtime.InteropServices.UnmanagedType.BStr)]
            public string pwzVirtualApplicationPath;
            public bool fWindowsAuthEnabled;
            public bool fBasicAuthEnabled;
            public bool fAnonymousAuthEnable;
        }

        public static void SetCurrentDirectory()
        {
            try
            {
                // Check if physical path was provided by ANCM
                var sitePhysicalPath = Environment.GetEnvironmentVariable("ASPNETCORE_IIS_PHYSICAL_PATH");
                if (string.IsNullOrEmpty(sitePhysicalPath))
                {
                    // Skip if not running ANCM InProcess
                    if (GetModuleHandle(AspNetCoreModuleDll) == IntPtr.Zero)
                    {
                        return;
                    }

                    IISConfigurationData configurationData = default(IISConfigurationData);
                    if (http_get_application_properties(ref configurationData) != 0)
                    {
                        return;
                    }

                    sitePhysicalPath = configurationData.pwzFullApplicationPath;
                }

                Environment.CurrentDirectory = sitePhysicalPath;
            }
            catch
            {
                // ignore
            }
        }
    }
}

а затем следующий код в первую строку вашего файла Program.cs:

CurrentDirectoryHelpers.SetCurrentDirectory();

Я столкнулся с той же проблемой, и эта работа решила ее для меня.

person Kenspeckle    schedule 06.02.2019
comment
Спасибо Kenspeckle! Попробую это решение и обновлю вас. - person Shankar Naru; 08.02.2019