Deploying Angular SSR (Universal) to App Service Windows

23 minute read | By Anthony Salemo

This post will cover deploying a Angular SSR (Universal) application to App Service Windows.

Overview

Angular SSR (previously Universal) is a server-side rendered implemenation of Angular that uses Express.js.

As opposed to the “original” implementation which is a single page application - Angular SSR is ran from a .mjs file when built for production (previously a .js file in older Universal projects) like a typical node-based server side project.

Where as with a SPA you can serve static files directly, typically by serving the index.html directly with it’s production built client-side .js files, Angular SSR deployments will be slightly different comapred to SPA’s.

This post will cover deploying these kinds of projects.

Prerequisites

App Service Logs

Ensure App Service Logs are enabled for your App Service (Windows) application.

“Application logging (Filesystem)” turns off after 24 hours on Windows applications, by design. Ensure this is kept on while going through this post.

Without these logs enabled, post-deployment/runtime troubleshooting will make it much harder or almost impossible to diagnose problems.

You can view/retrieve these logs directly from the Kudu site or FTP. Below is an example of what it would look like when these logs are enabled.

App Service Logs

Angular CLI

If you don’t have the Angular CLI (ng) run npm install -g @angular/cli. This is required for build commands within the Angular project iself.

Quickstart

This will follow Angulars SSR quickstart for a net-new project.

  1. Create a Angular SSR project with the Angular CLI by running ng new --ssr. This command will prompt for the name of the project, which can be arbitrary. This will then create a new directory with the name project

    This will also ask what kind of style formatting to use. The below project used here is named angularssr with scss for styling.

    Angular quickstart

    After this, the project will be scaffolded.

  2. cd into the new project folder and run npm run start:

    Local Angular SSR application

  3. You should now be able to browse the application on localhost:4200

    Local Angular SSR application

Deployment

Requirements for deployment

There are 3 very important concepts/requisites for any deployment of Angular SSR/Universal onto Windows App Service.

  1. You need to generate a production build. You can do this locally prior to deployment, through a custom deployment script, or with CI/CD. With Angular, in general, this will create a folder named /dist at the top level of your project root. This folder contains our entrypoint to the application, which will be ran on App Service later on.

    If /dist doesn’t already exist, you can generate this by running npm run build. Prior to each deployment, regardless where it’s occurring, you want to ensure that npm run build is being ran to reflect any changed application content during development.

    Local Angular SSR application

    The entrypoint to the application would then be dist/[projectname]/server/server.mjs

  2. web.config - This is required. This is only relevant on Windows App Service and Node since it uses IIS.

    Note, by default, a web.config will not be automatically generated with the way the package.json is configured since it looks for an existing start script pointing to an existing entrypoint file.

     "scripts": {
         "ng": "ng",
         "start": "ng serve",
         "build": "ng build",
         "watch": "ng build --watch --configuration development",
         "test": "ng test",
         "serve:ssr:angularssr": "node dist/angularssr/server/server.mjs"
     },
    

    ng serve is the value of start and is invalid (in Kudu deployment engines case). But this is okay, as we can manually create one.

    Use the below web.config - create this file in the root of your project, note the server.js reference, keep this and read the next point 3)

    <?xml version="1.0" encoding="utf-8"?>
     <!--
         This configuration file is required if iisnode is used to run node processes behind
         IIS or IIS Express.  For more information, visit:
    
         https://github.com/tjanczuk/iisnode/blob/master/src/samples/configuration/web.config
     -->
    
     <configuration>
     <system.webServer>
         <!-- Visit http://blogs.msdn.com/b/windowsazure/archive/2013/11/14/introduction-to-websockets-on-windows-azure-web-sites.aspx for more information on WebSocket support -->
         <webSocket enabled="false" />
         <handlers>
         <!-- Indicates that the server.js file is a node.js site to be handled by the iisnode module -->
         <add name="iisnode" path="server.js" verb="*" modules="iisnode"/>
         </handlers>
         <rewrite>
         <rules>
             <!-- Do not interfere with requests for node-inspector debugging -->
             <rule name="NodeInspector" patternSyntax="ECMAScript" stopProcessing="true">
             <match url="^server.js\/debug[\/]?" />
             </rule>
    
             <!-- First we consider whether the incoming URL matches a physical file in the /public folder -->
             <rule name="StaticContent">
             <action type="Rewrite" url="public{REQUEST_URI}"/>
             </rule>
    
             <!-- All other URLs are mapped to the node.js site entry point -->
             <rule name="DynamicContent">
             <conditions>
                 <add input="{REQUEST_FILENAME}" matchType="IsFile" negate="True"/>
             </conditions>
             <action type="Rewrite" url="server.js"/>
             </rule>
         </rules>
         </rewrite>
            
         <!-- 'bin' directory has no special meaning in node.js and apps can be placed in it -->
         <security>
         <requestFiltering>
             <hiddenSegments>
             <remove segment="bin"/>
             </hiddenSegments>
         </requestFiltering>
         </security>
    
         <!-- Make sure error responses are left untouched -->
         <httpErrors existingResponse="PassThrough" />
    
         <!--
         You can control how Node is hosted within IIS using the following options:
             * watchedFiles: semi-colon separated list of files that will be watched for changes to restart the server
             * node_env: will be propagated to node as NODE_ENV environment variable
             * debuggingEnabled - controls whether the built-in debugger is enabled
    
         See https://github.com/tjanczuk/iisnode/blob/master/src/samples/configuration/web.config for a full list of options
         -->
         <!--<iisnode watchedFiles="web.config;*.js"/>-->
     </system.webServer>
     </configuration>
    
  3. iisNode is the default handler for Node applications on App Service Windows. However, iisNode doesn’t support es6 modules. Because of, this needs to be followed to get our application to work - Supporting ES6 import on Windows App Service (Node.js/IISNode)

    In our Angular SSR project, create a file named server.js in the project root.

    Add the following contents, where [project_name] is the project name of the application you created. You can navigate down into /dist to find your full path name to the server.mjs file.

    (server.js)

     import("./dist/[project_name]/server/server.mjs");
    

    In this blog example, we’d set our server.js contents to import("./dist/angularssr/server/server.mjs"); to match our structure below:

    Angular SSR project structure

  4. For sites that require some degree of build automation (npm install) ensure the App Setting SCM_DO_BUILD_DURING_DEPLOYMENT is set to true. More on this is explained here

  5. This section only applies to Angular v19 - if you’re using a ssr template generated from a version prior to this, you do notneed to do this

    Starting with Angular v19, a change in the ng --ssr template was introduced which added a logic check prior to starting the server. This is now located in src/server.ts (instead of top level server.ts). It now includes this following logic:

     /**
      * Start the server if this module is the main entry point.
     * The server listens on the port defined by the `PORT` environment variable, or defaults to 4000.
     */
     if (isMainModule(import.meta.url)) {
       const port = process.env['PORT'] || 4000;
       app.listen(port, () => {
         console.log(`Node Express server listening on http://localhost:${port}`);
       });
     }
    

    This can and will introduce an issue when deployed to App Service on Windows, since we need to specify a different entrypoint to get around iisnode’s es6 limitations, this causes isMainModule(), which returns a boolean, to return false. This function checks process.argv[1], and with that being the case, would cause the following if left as is:

     import.meta.url = file:///C:/home/site/wwwroot/dist/angularssr/server/server.mjs
     process.argv[1] = C:\home\site\wwwroot\server.js
    

    This is where the mismatch comes in. To get around this, you can remove the if statement, or if desiring to keep this, alter this as need be. See GitHub issues - 28993 as an example. For simplicity and the sake of this post, we’ll just use the following, which simply just removes the if/else check.

       const port = process.env['PORT'] || 4000;
       app.listen(port, () => {
         console.log(`Node Express server listening on http://localhost:${port}`);
       });
    

Recap:

  1. Generate a production build. Either manually or automatically depending on how you’re deploying.
  2. Create a web.config in your project root. This is a requirement. Use the above one.
  3. Create a new .js file to act as your entrypoint since es6 is not supported with iisNode
  4. Enable build automation, if needed. This is only going to apply for npm install, not npm run build. Be mindful of this.

Local Git / VSCode

For setting up local git credentials - follow Local Git deployment to Azure App Service first.

  1. Navigate to your App Service and select Deployment Center and then click on Local Git and then click on Save.

    Deployment Center

  2. Copy the remote git repository from Azure Portal.

    Local Git

  3. IMPORTANT: npm run build is not going to automatically be ran against our project with build automation, however npm install would. Obviously, this would not create a production build. So we have two options:

    a. Manually run npm run build first to generate /dist. On each deployment, if there is a code change, this needs to be manually done before deployment.Also, Remove /dist from .gitignore or else this directory won’t be pushed.

    b. Or, use the custom deployment script here as an example to regenerate the production build on each deployment. This method would let you retain /dist within .gitignore. If you go with this option, you need to ensure deploy.cmd and .deployment is in the root of your project.

    • You can also utilize the kuduScript NPM package which would automatically create these two files for you.
    • Note, .deployment needs to contain:
     [config]
     command = deploy.cmd
    
  4. In your local terminal run the following commands in order:
     git add .
     git commit -m "Initial Commit"
     git remote add azure https://<sitename>.scm.azurewebsites.net:443/<sitename>.git
     git push azure master
    
  5. Assuming that the section Requirements for deployment was followed above, you should see the below site contents with an accessible site.

    Local Git

    Local Git


VSCode You can deploy to applications on App Service by following Deploy a Node.js web app in Azure. More importantly, you’ll need to install the Azure App Service for Visual Studio Code extension. Sign into your Azure account afterwards.

This method follows the same concept as Local Git. You can either prebuild for /dist or use a custom deployment script as well. A web.config is required in all scenarios, regardless of deployment method.

AZ CLI

You can deploy your application with the Azure CLI. This example will use the az webapp deploy command.

We’ll need to zip our content. On Windows, you can highlight all project files and right click to Compress to ZIP file in File Explorer. On WSL2/Linux - you can install the zip package and zip the project with sudo zip app.zip ./*.

Using this method, you have an option for a much simpler deployment - as well as a few general options:

  • Recommended: Only include server.js, /dist, and web.config in your zip
  • Alternatively, you can still include everything (including node_modules) and deploy that zip as a whole. However, this is a bit redundant. If doing this, set SCM_DO_BUILD_DURING_DEPLOYMENT = false
  • You can also include your source code, except node_modules - but if doing this, you need to set SCM_DO_BUILD_DURING_DEPLOYMENT = true

npm run build will not automatically be ran - so we need to run this ourselves prior to deployment, even when SCM_DO_BUILD_DURING_DEPLOYMENT = true.

  1. Run the command npm run build to regenerate our production build /dist
  2. Zip up dist, server.js and web.config

    Zip artifact

  3. Deploy the site with the following command:

     az webapp deploy -g somerg -n someapp --src-path ./app.zip --type zip --async true
    

A successful deployment output would show the following in your terminal. If you go to your Kudu site - you should also see that node_modules has been recreated since npm install was implicitly ran:

Azure CLI deployment

If you look at the file system, you’ll see just dist, server.js and web.config. You should still have a functional app in this case since dist contains the production .js files and other assets.

Deployed filesystem

GitHub Actions

You can use CI/CD like GitHub Actions to deploy your Angular SSR/Universal application.

Ensure your web.config already exists in your project root. Or create it on the site’s filesystem after the fact. This pipeline will not auto-generate one, neither will App Service Build Server (explained earlier)

Note, using this method, the typical approach is to keep the installation/production building of your application within the pipeline. Avoid setting SCM_DO_BUILD_DURING_DEPLOYMENT = true. Assuming you’re using the workflow generated from the portal - this would end up running npm install on the pipeline - and then again on Kudu, which is redundant and cause build descrenpencies in some cases.

  1. Go to Deployment Center and select GitHub
  2. Fill out the required fields:
    • “Signed in as”
    • Organization
    • Repository
    • Branch
    • Authentication type
    • Subscription
    • Identity

This example uses User Assigned Identity as the authentication type for the deployment task in the pipeline

You can select “Preview file” to see what the file looks like before saving. Once saved, it’ll commit and create a .github/workflows/[branchname_apname].yml file to the repository and kick off an Actions run

Github Actions

The below is the generated file as of this blog post:

# Docs for the Azure Web Apps Deploy action: https://github.com/Azure/webapps-deploy
# More GitHub Actions for Azure: https://github.com/Azure/actions

name: Build and deploy Node.js app to Azure Web App - someapp

on:
  push:
    branches:
      - main
  workflow_dispatch:

jobs:
  build:
    runs-on: windows-latest

    steps:
      - uses: actions/checkout@v4

      - name: Set up Node.js version
        uses: actions/setup-node@v3
        with:
          node-version: '18.x'

      - name: npm install, build, and test
        run: |
          npm install
          npm run build --if-present
          npm run test --if-present
      
      - name: Upload artifact for deployment job
        uses: actions/upload-artifact@v4
        with:
          name: node-app
          path: .

  deploy:
    runs-on: ubuntu-latest
    needs: build
    environment:
      name: 'Production'
      url: $
    permissions:
      id-token: write #This is required for requesting the JWT

    steps:
      - name: Download artifact from build job
        uses: actions/download-artifact@v4
        with:
          name: node-app
      
      - name: Login to Azure
        uses: azure/login@v2
        with:
          client-id: $
          tenant-id: $
          subscription-id: $

      - name: 'Deploy to Azure Web App'
        uses: azure/webapps-deploy@v3
        id: deploy-to-webapp
        with:
          app-name: 'someapp'
          slot-name: 'Production'
          package: .
      name: node-app Note, if you have tests that take a while to run - these can be removed by deleting the line `npm run test --if-present`. This should be done in some cases where default Angular tests with Karma try to open a chromium browser - this will cause the workflow run to "hang" until it times out.

Also, given the size of node_modules historically, copying files from the “build” stage to the “deploy” stage can take tens of minutes, even with a “starter” application, since the . under actions/upload-artifact@v4 is copying file-by-file.

To make this faster, you can zip the contents in the build stage and move it to the deployment stage, which can significantly reduce overall deployment time. Below is a full example:

# Docs for the Azure Web Apps Deploy action: https://github.com/Azure/webapps-deploy
# More GitHub Actions for Azure: https://github.com/Azure/actions

name: Build and deploy Node.js app to Azure Web App - someapp

on:
  push:
    branches:
      - main
  workflow_dispatch:

jobs:
  build:
    runs-on: windows-latest

    steps:
      - uses: actions/checkout@v4

      - name: Set up Node.js version
        uses: actions/setup-node@v3
        with:
          node-version: '18.x'

      - name: npm install, build, and test
        run: |
          npm install
          npm run build --if-present
          npm run test --if-present

      - name: Zip all files for upload between jobs
        run: Compress-Archive -Path .\* -DestinationPath angularssr.zip
    
      - name: Upload artifact for deployment job
        uses: actions/upload-artifact@v4
        with:
          name: node-app
          path: angularssr.zip

  deploy:
    runs-on: windows-latest
    needs: build
    environment:
      name: 'Production'
      url: $
    permissions:
      id-token: write #This is required for requesting the JWT

    steps:
      - name: Download artifact from build job
        uses: actions/download-artifact@v4
        with:
          name: node-app
      
      - name: Login to Azure
        uses: azure/login@v2
        with:
          client-id: $
          tenant-id: $
          subscription-id: $

      - name: 'Deploy to Azure Web App'
        uses: azure/webapps-deploy@v3
        id: deploy-to-webapp
        with:
          app-name: 'someapp'
          slot-name: 'Production'
          package: angularssr.zip

      - name: Delete zip file
        run: rm next.zip
          

You can even go one step further and only zip up the files you exactly need, which in this case, just like with the AZ CLI example above, is dist, web.config and server.js by changing your Zip all files for upload between jobs script to the below:

      - name: Zip all files for upload between jobs
        run: Compress-Archive -Path .\dist, web.config, server.js -DestinationPath angularssr.zip
    
      - name: Upload artifact for deployment job
        uses: actions/upload-artifact@v4
        with:
          name: node-app
          path: angularssr.zip

In summary, we are:

  • Installing node/application packages and building for production to generate /dist
  • Zipping the contents from build and pass it to stage
  • Deploy the zip from build, which will be extracted on the Kudu site to wwwroot.
  • Delete the zip in our pipeline after we’re done

Again, ensure SCM_DO_BUILD_DURING_DEPLOYMENT is false if you have npm install in your pipeline.

Azure Pipelines (ADO)

Much like how GitHub Actions is structured, you can use Azure Pipelines to deploy your Angular SSR/Universal application.

The same as above still applies:

  • Ensure your web.config already exists in your project root. Or create it on the site’s filesystem after the fact. This pipeline will not auto-generate one, neither will App Service Build Server (explained earlier)
  • Since we want to have package installation and building done in the pipeline - SCM_DO_BUILD_DURING_DEPLOYMENT should be false or unset.
  1. Go to your DevOps organization to Create a project. Then go to Pipelines and create New pipeline.
  2. Select where your repository is stored (GitHub, Azure Repos, etc.)
  3. Select your code repository.
  4. Create an empty template template or use the “Starter Pipepine” and override it with the below. To make it easier to authenticate for deployment, when adding the below tasks, use the “Tasks Assistant” on the right-hand side. This will give an option in the UI for the deployment task to Authorize the Service Connection needed for deployment.
trigger:
- main

variables:
  # Agent VM image name
  vmImageName: 'windows-latest'
  environmentName: 'appname'

stages:
- stage: Build
  displayName: Build stage
  jobs:
  - job: Build
    displayName: Build
    pool:
      vmImage: $(vmImageName)

    steps:
    - task: NodeTool@0
      inputs:
        versionSpec: '18.x'
      displayName: 'Install Node.js'

    - script: |
        npm install
      displayName: 'npm install'
      
    - script: |
        npm run build
      displayName: 'npm run build'
      
    - task: ArchiveFiles@2
      inputs:
        rootFolderOrFile: '$(System.DefaultWorkingDirectory)'
        includeRootFolder: false
        archiveType: 'zip'
        archiveFile: '$(Build.ArtifactStagingDirectory)/$(Build.BuildId).zip'
        replaceExistingArchive: true
        
    - task: PublishBuildArtifacts@1
      inputs:
        PathtoPublish: '$(Build.ArtifactStagingDirectory)/$(Build.BuildId).zip'
        ArtifactName: 'drop'

- stage: Deploy
  displayName: Deploy stage
  dependsOn: Build
  condition: succeeded()
  jobs:
  - deployment: Deploy
    displayName: Deploy
    environment: $(environmentName)
    pool:
      vmImage: $(vmImageName)
    strategy:
      runOnce:
        deploy:
          steps:
          - task: AzureWebApp@1
            inputs:
              azureSubscription: 'subscriptionName(00000000-0000-0000-0000-000000000000)'
              appType: 'webApp'
              appName: 'appname'
              package: '$(Pipeline.Workspace)/drop/$(Build.BuildId).zip'

This approach, like with GitHub Actions, will build for production in the pipeline and deploy the dist folder required. Under the hood, ZipDeploy is used to deploy the zip artifact to Kudu.

Additionally, like our updated Github Actions .yml seen earlier - this also zips our content between build and deployment stages in the pipeline to avoid spending a large amount of time on file copying between stages.

NOTE: Just like in the GitHub Actions and AZ CLI examples - you can replace ArchiveFiles@2 and instead zip just only the files you need. Just ensure these are output to a folder under $(Build.ArtifactStagingDirectory)/$(Build.BuildId).zip

Troubleshooting

HTTP 500.1xxx errors after deployments

Review this post for common iisnode-based issues - Troubleshooting Common iisnode Issues

A table of iisnode substatus codes can be found here - this can be used to track down what may be occurring. This usual indicates that node.exe is crashing. A logging-errors.txt file will be created (assuming that App Service Logs are enabled). If an uncaught exception is occurring, it will be logged into this file.

Common scenarios for this directly after a deployment may be:

  • Forgetting to add a package into your package.json but referencing the missing package as an import in code
  • Forgetting needed environment variables that may otherwise cause the application to crash
  • Targeting a version of Node that may not support the Angular version being used or underlying libraries use (ex., outdated Node version - if the App Setting WEBSITE_NODE_DEFAULT_VERSION is not set, this falls back to a v0.x version.)

You do not have permission to view this directory or page.

You may see You do not have permission to view this directory or page. when browsing the application. This is normally due to:

  • Missing web.config
  • Inconfigured web.config, eg:
    • Pointing to a non-existent .js entrypoint
    • Pointing to an incorrect location for a .js entrypoint
    • web.config is not at the root path (relative to package.json)

“parking page” is still being displayed

If the “parking page” / default page is being displayed after a deployment - it likely means a following scenario:

  • You deployed a nested zip - which means the outer zip was extracted but the inner .zip was not. Therefor no relevant site contents can be discovered
  • You deployed a nested directory. For example, from outside of your project root. This means there is two levels of directories - so site contents can not be ran or discovered.

Hardcoding ports

The quickstart Angular SSR project does not hardcode ports in the sense the port variable is set as the below:

const port = process.env['PORT'] || 4000;

However, if you remove process.env.PORT - the application will fail to start on App Service. Ensure this is always set. For easier switching between local/dev environments and deployed ones - ensure the ternary operator (above) is kept

Timing out and failing to start with HTTP 500.1001’s

See the Requirements for deployment section, specifically step 5, if you’re using Angular v19.

If this is a version prior to this - ensure your server.ts does not have any logic blocking the invocation of express.js startup (or any other engine you may be using). In both of these cases, if express.js (or any other engine) is never successfully starting (or never gets invoked), this will cause no logs to be written unless you add a console.log line earlier on in server.ts, prior to before that logic with server startup/invocation is hit.

If the issue is neither of these, consult Troubleshooting Common iisnode Issues