Module 2 - Communication Between Microservices in ACA¶
Module Duration
60 minutes
Objective¶
In this module, we will accomplish three objectives:
- Create a web app named
ACA Web - Frontend
, which is the UI to interact withACA API - Backend
. - Deploy the
ACA Web - Frontend
container app to Azure. - Shield
ACA API - Backend
from external access.
Module Sections¶
- From the VS Code Terminal tab, open developer command prompt or PowerShell terminal in the project folder
TasksTracker.ContainerApps
(root of your project):
- Restore the previously-stored variables by executing the local script. The output informs you how many variables have been set.
1. Create the Frontend Web App project¶
-
Initialize the web project. This will create and ASP.NET Razor Pages web app project.
-
We need to containerize this application, so we can push it to the Azure Container Registry before we deploy it to Azure Container Apps:
- Open the VS Code Command Palette (Ctrl+Shift+P) and select Docker: Add Docker Files to Workspace...
- Use
.NET: ASP.NET Core
when prompted for the application platform. - Choose the newly-created project, if prompted.
- Choose
Linux
when prompted to choose the operating system. - Set the application port to
8080
, which is the default non-privileged port since .NET 8. - You will be asked if you want to add Docker Compose files. Select
No
. Dockerfile
and.dockerignore
files are added to the project workspace.-
Open
Dockerfile
and remove--platform=$BUILDPLATFORM
from theFROM
instruction.Dockerfile Build Platform
Azure Container Registry does not set
$BUILDPLATFORM
presently when building containers. This consequently causes the build to fail. See this issue for details. Therefore, we remove it from the file for the time being. We expect this to be corrected in the future.
-
From inside the Pages folder, add a new folder named Tasks. Within that folder, add a new folder named Models, then create file as shown below.
-
Now, in the Tasks folder, we will add 3 Razor pages for CRUD operations which will be responsible for listing tasks, creating a new task, and updating existing tasks. By looking at the cshtml content notice that the page is expecting a query string named
createdBy
which will be used to group tasks for application users.Note
We are following this approach here to keep the workshop simple, but for production applications, authentication should be applied and the user email should be retrieved from the claims identity of the authenticated users.
What does this code do?
In the code above we've injected named HttpClientFactory which is responsible to call the Backend API service as HTTP request. The index page supports deleting and marking tasks as completed along with listing tasks for certain users based on the
createdBy
property stored in a cookie namedTasksCreatedByCookie
. More about populating this property later in the workshop.What does this code do?
The code is self-explanatory here. We just injected the type HttpClientFactory in order to issue a POST request and create a new task.
What does this code do?
The code added is similar to the create operation. The Edit page accepts the TaskId as a Guid, loads the task, and then updates the task by sending an HTTP PUT operation.
-
Now we will inject an HTTP client factory and define environment variables. To do so, we will register the HttpClientFactory named
BackEndApiExternal
to make it available for injection in controllers. Open theProgram.cs
file and update it with highlighted code below. Your file may be flattened rather than indented and not contain some of the below elements. Don't worry. Just place the highlighted lines in the right spot: -
Next, we will add a new environment variable named
BackendApiConfig:BaseUrlExternalHttp
intoappsettings.json
file. This variable will contain the Base URL for the backend API deployed in the previous module to ACA. Later on in the workshop, we will see how we can set the environment variable once we deploy it to ACA. Use the output from this script as theBaseUrlExternalHttp
value.
-
Lastly, we will update the web app landing page
Index.html
andIndex.cshtml.cs
inside Pages folder to capture the email of the tasks owner user and assign this email to a cookie namedTasksCreatedByCookie
. -
From VS Code Terminal tab, open developer command prompt or PowerShell terminal and navigate to the frontend directory which hosts the
.csproj
project folder and build the project.
Note
Make sure that the build is successful and that there are no build errors. Usually you should see a Build succeeded message in the terminal upon a successful build.
2. Deploy Razor Pages Web App Frontend Project to ACA¶
-
We need to add the below shell variables:
-
Now we will build and push the Web App project docker image to ACR. Use the below command to initiate the image build and push process using ACR. The
.
at the end of the command represents the docker build context. In our case, we need to be on the parent directory which hosts the .csproject. -
Once this step is completed you can verify the results by going to the Azure portal and checking that a new repository named
tasksmanager/tasksmanager-frontend-webapp
has been created and that a new docker image with alatest
tag has been created. -
Next, we will create and deploy the Web App to ACA using the following command:
Tip
Notice how we used the property env-vars
to set the value of the environment variable named BackendApiConfig:BaseUrlExternalHttp
which we added in the AppSettings.json file, using the double underscore delimiter for the indented property. You can set multiple environment variables at the same time by using a space between each variable.
The ingress
property is set to external
as the Web frontend App will be exposed to the public internet for users.
After you run the command, copy the FQDN (Application URL) of the Azure container app named tasksmanager-frontend-webapp
and open it in your browser, and you should be able to browse the frontend web app and manage your tasks.
3. Update Backend Web API Container App Ingress property¶
So far the Frontend App is sending HTTP requests to the publicly exposed Web API. This means that any REST client can invoke the Web API. We need to change the Web API ingress settings and make it accessible only by applications deployed within our Azure Container Environment. Any application outside the Azure Container Environment should not be able to access the Web API.
-
To change the settings of the Backend API, execute the following command:
Want to know more about the command?
When you do this change, the FQDN (Application URL) will change, and it will be similar to the one shown below. Notice how there is an Internal
part of the URL. https://tasksmanager-backend-api.internal.[Environment unique identifier].eastus.azurecontainerapps.io/api/tasks/
If you try to invoke the URL directly it will return 403 - Forbidden as this internal Url can only be accessed successfully from container apps within the container environment. This means that while the API is not accessible, it still provides a clue that something exists at that URL. Ideally, we would want to see a 404 - Not Found. However, recall from module 1 that we did not set internal-only
for simplicity's sake of the workshop. In a production scenario, this should be done with completely private networking to not reveal anything.
The FQDN consists of multiple parts. For example, all our Container Apps will be under a specific Environment unique identifier (e.g. agreeablestone-8c14c04c
) and the Container App will vary based on the name provided, check the image below for a better explanation.
-
Now we will need to update the Frontend Web App environment variable to point to the internal backend Web API FQDN. The last thing we need to do here is to update the Frontend WebApp environment variable named
BackendApiConfig:BaseUrlExternalHttp
with the new value of the internal Backend Web API base URL, to do so we need to update the Web App container app and it will create a new revision implicitly (more about revisions in the upcoming modules). We need to use the double underscore format to set the variable. The following command will update the container app with the changes:
Success
Browse the web app again, and you should be able to see the same results and access the backend API endpoints from the Web App. You can obtain the frontend URL by displaying this variable.
-
Navigate to the root and persist the module to Git.
-
From the root folder of your project, execute the script you created in the Prerequisites section to persist a list of all current variables.
Execute the Set-Variables.ps1
in the root to update the variables.ps1
file with all current variables. The output of the script will inform you how many variables are written out.
Execute the set_variables.sh
in the root to update the variables.sh
file with all current variables. The output of the script will inform you how many variables are written out.
- From the root folder of your project, persist a list of all current variables.
Review¶
In this module, we have accomplished three objectives:
- Created a web app named
ACA Web - Frontend
, which is the UI to interact withACA API - Backend
. - Deployed the
ACA Web - Frontend
container app to Azure. - Shielded
ACA API - Backend
from external access.
In the next module, we will start integrating Dapr and use the service to service Building block for services discovery and invocation.