You can find the previous article here:
data:image/s3,"s3://crabby-images/66972/66972b43e92522a9047aff0c13358bfd83d33bf8" alt=""
Power Platform ALM: Building Basic Solution Deployment Pipelines - Step by Step Guide
I had an opportunity to use "Pipelines," a new management tool in Power Platform, so I'll document how to use it.What We...
GitHub Commit Integration
We’ll extend the Power Platform deployment pipeline we created previously to commit solutions to GitHub. While the process generally follows the official documentation, some modifications are necessary.Official Documentation:
data:image/s3,"s3://crabby-images/0bd4e/0bd4ef1a62d7dadf756a3aa2d439c8db5a3877d9" alt=""
Extend pipelines using GitHub Actions - Power Platform
Download, unpack, and commit a solution using a GitHub workflow called from a Power Automate Flow.
Implementation Steps
Follow these steps to set up the integration:
- Register an application in Entra ID (formerly Azure AD)
- Add an application user to your Power Platform environment
- GitHub: Create a repository and set up Actions
- GitHub: Generate an access token
- Build Power Automate flow: Configure GitHub API calls
Entra ID Registration
First, let’s start with Entra ID registration. During this process, make note of these three values:
- Client Secret
- Client ID (Application ID)
- Tenant ID
Begin by opening Entra ID in the Azure Portal
data:image/s3,"s3://crabby-images/38012/3801288ea6d3f6eee1e5fe483898936e3f18538d" alt=""
data:image/s3,"s3://crabby-images/38012/3801288ea6d3f6eee1e5fe483898936e3f18538d" alt=""
Select [App registrations] and click [New registration] data:image/s3,"s3://crabby-images/50258/502589caa8a68c99ce7116e359972987a564c3e2" alt=""
data:image/s3,"s3://crabby-images/50258/502589caa8a68c99ce7116e359972987a564c3e2" alt=""
Enter an appropriate name and register it as a single-tenant application. data:image/s3,"s3://crabby-images/7122e/7122e4cc2b8738117d3a0a780889f26295b49dd2" alt=""
data:image/s3,"s3://crabby-images/7122e/7122e4cc2b8738117d3a0a780889f26295b49dd2" alt=""
Next, go to [API permissions] of the registered app and select [Dynamics CRM] data:image/s3,"s3://crabby-images/32d55/32d5573bebe10c71183f12a8065af43d6e3575bb" alt=""
data:image/s3,"s3://crabby-images/32d55/32d5573bebe10c71183f12a8065af43d6e3575bb" alt=""
Check “user_impersonation” and click [Add permissions] data:image/s3,"s3://crabby-images/e4da3/e4da3f6a8c2a8a60a9918995d5bbe1a0dcccf109" alt=""
data:image/s3,"s3://crabby-images/e4da3/e4da3f6a8c2a8a60a9918995d5bbe1a0dcccf109" alt=""
Go to [Certificates & secrets] and select [New client secret] data:image/s3,"s3://crabby-images/a6a91/a6a914f0c1b579e6e1a7f33296dab2ad1149260a" alt=""
data:image/s3,"s3://crabby-images/a6a91/a6a914f0c1b579e6e1a7f33296dab2ad1149260a" alt=""
Make note of the generated value data:image/s3,"s3://crabby-images/f3ecf/f3ecf1193d9ec71a3484cd0da660dfb6e8926c90" alt=""
data:image/s3,"s3://crabby-images/f3ecf/f3ecf1193d9ec71a3484cd0da660dfb6e8926c90" alt=""
Finally, go to [Overview] and note down both the [Application ID] and [Tenant ID]. This completes this section. data:image/s3,"s3://crabby-images/13f67/13f67e8f303e8a13969833c303cc0cce73a16644" alt=""
data:image/s3,"s3://crabby-images/13f67/13f67e8f303e8a13969833c303cc0cce73a16644" alt=""
Registering an Application User in Power Platform Environment
Next, we’ll create an application user in the Power Platform environment and assign Dataverse security roles for GitHub integration.In the Power Platform admin center, select the environment where you built the pipeline and click [Settings]. data:image/s3,"s3://crabby-images/7e7c9/7e7c90266d25f81fee18ebc0fb1500b736107761" alt=""
data:image/s3,"s3://crabby-images/7e7c9/7e7c90266d25f81fee18ebc0fb1500b736107761" alt=""
Select [Application users] data:image/s3,"s3://crabby-images/1a9ce/1a9ceadc975cec89672aa92b8fc64b6cd85b2db5" alt=""
data:image/s3,"s3://crabby-images/1a9ce/1a9ceadc975cec89672aa92b8fc64b6cd85b2db5" alt=""
Click [New app user], then [Add an app] data:image/s3,"s3://crabby-images/52f2f/52f2f1b13bc868480d75f8716ab3ec32400dd938" alt=""
data:image/s3,"s3://crabby-images/52f2f/52f2f1b13bc868480d75f8716ab3ec32400dd938" alt=""
Select the application you registered in Entra ID from the previous steps. data:image/s3,"s3://crabby-images/5b365/5b365aafcb0a6bbcfb9d8cb51d51266b17418986" alt=""
data:image/s3,"s3://crabby-images/5b365/5b365aafcb0a6bbcfb9d8cb51d51266b17418986" alt=""
Set the business unit (in this example, we’re using the root business unit) and security role, then click [Create].
Note: While the System Customizer role will definitely work, it’s recommended to create a custom role if you want to minimize permissions.data:image/s3,"s3://crabby-images/da043/da0433d64ad239dc6a1f902e8d85a6e456eb80e6" alt=""
This completes the application user registration in your Power Platform environment.
Note: While the System Customizer role will definitely work, it’s recommended to create a custom role if you want to minimize permissions.
data:image/s3,"s3://crabby-images/da043/da0433d64ad239dc6a1f902e8d85a6e456eb80e6" alt=""
GitHub: Creating Repository and Setting Up Actions
Repository Creation
Navigate to GitHub and create a new repository. data:image/s3,"s3://crabby-images/39f7b/39f7ba624ca8ed07fbd3ddeb27334145a1a82a66" alt=""
data:image/s3,"s3://crabby-images/39f7b/39f7ba624ca8ed07fbd3ddeb27334145a1a82a66" alt=""
Actions Configuration
After creating the repository, go to Actions and select [set up a workflow yourself]. data:image/s3,"s3://crabby-images/b13ca/b13ca2d5c96864277a6dd96d1ce991e18fbf3508" alt=""
data:image/s3,"s3://crabby-images/b13ca/b13ca2d5c96864277a6dd96d1ce991e18fbf3508" alt=""
Then, paste the following modified version of the official code and click [Commit changes]. This code includes our specific modifications to the original. data:image/s3,"s3://crabby-images/dce6b/dce6b325e7e5053095a944f3445e8a5c877faf6d" alt=""
The code modifications are as follows:
data:image/s3,"s3://crabby-images/dce6b/dce6b325e7e5053095a944f3445e8a5c877faf6d" alt=""
# Before # Unpack the solution - name: unpack solution uses: microsoft/powerplatform-actions/unpack-solution@v0 with: solution-file: "${{ github.event.inputs.solution_name }}.zip" solution-folder: "${{ github.event.repository.name }}" solution-type: 'Both' process-canvas-apps: false overwrite-files: true # After the change (v1 was used because the unpack-solution did not work with v0) # Install PAC - name: Install Power Platform Tools uses: microsoft/powerplatform-actions/actions-install@v1 # Unpack the solution - name: unpack solution uses: microsoft/powerplatform-actions/unpack-solution@v1 with: solution-file: "${{ github.event.inputs.solution_name }}.zip" solution-folder: "${{ github.event.repository.name }}" solution-type: 'Both' process-canvas-apps: false overwrite-files: trueBelow is the full source code.
name: Download, unpack and commit the solution to git run-name: Getting ${{ github.event.inputs.solution_name }} from pipelines host environment and committing on: workflow_dispatch: inputs: artifact_url: description: "The url of the Dataverse record ID for the artifact created by the pipelines (Example: https://[your-env].crm.dynamics.com/api/data/v9.0/deploymentartifacts([your-artifact-id])/artifactfile/$value)." required: true solution_name: description: "Name of the Solution in Dataverse environment" required: true user_name: description: "User name for the commit" required: true source_branch: description: "Branch for the solution commit" required: true target_branch: description: "Branch to create for the solution commit" required: false commit_message: description: "Message to provide for the commit" required: true permissions: contents: write jobs: export-unpack-commit: runs-on: ubuntu-latest steps: - uses: actions/checkout@v3 with: ref: ${{ github.event.inputs.source_branch }} # Commit changes to the existing or new branch - name: create new branch if specified shell: pwsh run: | if('${{ github.event.inputs.target_branch }}' -ne '') { git checkout -b ${{ github.event.inputs.target_branch }} ${{ github.event.inputs.source_branch }} } # Export the solution from the artifact created by pipelines - name: download solution from artifact env: CLIENT_ID: ${{secrets.CLIENT_ID}} TENANT_ID: ${{secrets.TENANT_ID}} CLIENT_SECRET: ${{secrets.CLIENT_SECRET}} shell: pwsh run: | $aadHost = "login.microsoftonline.com" $url = "${{ github.event.inputs.artifact_url }}" $options = [System.StringSplitOptions]::RemoveEmptyEntries $dataverseHost = $url.Split("://", $options)[1].Split("/")[0] $body = @{client_id = $env:CLIENT_ID; client_secret = $env:CLIENT_SECRET; grant_type = "client_credentials"; scope = "https://$dataverseHost/.default"; } $OAuthReq = Invoke-RestMethod -Method Post -Uri "https://$aadHost/$env:TENANT_ID/oauth2/v2.0/token" -Body $body $spnToken = $OAuthReq.access_token $headers = New-Object "System.Collections.Generic.Dictionary[[String],[String]]" $headers.Add("Authorization", "Bearer $spnToken") $headers.Add("Content-Type", "application/json") # Download the managed solution $response = Invoke-RestMethod "${{ github.event.inputs.artifact_url }}" -Method 'GET' -Headers $headers $bytes = [Convert]::FromBase64String($response.value) [IO.File]::WriteAllBytes("${{ github.event.inputs.solution_name }}_managed.zip", $bytes) # Download the unmanaged solution (for now we will need to use string manipulation to get the unmanaged solution URL, until the API provides this value) $unmanaged_artifact_url = "${{ github.event.inputs.artifact_url }}".Replace("artifactfile", "artifactfileunmanaged") $response = Invoke-RestMethod "$unmanaged_artifact_url" -Method 'GET' -Headers $headers $bytes = [Convert]::FromBase64String($response.value) [IO.File]::WriteAllBytes("${{ github.event.inputs.solution_name }}.zip", $bytes) # Install PAC - name: Install Power Platform Tools uses: microsoft/powerplatform-actions/actions-install@v1 # Unpack the solution - name: unpack solution uses: microsoft/powerplatform-actions/unpack-solution@v1 with: solution-file: "${{ github.event.inputs.solution_name }}.zip" solution-folder: "${{ github.event.repository.name }}" solution-type: 'Both' process-canvas-apps: false overwrite-files: true # Commit changes to the existing or new branch - name: commit changes shell: pwsh run: | rm -rf ${{ github.event.inputs.solution_name }}.zip rm -rf ${{ github.event.inputs.solution_name }}_managed.zip git config user.name ${{ github.event.inputs.user_name }} git pull git add --all git commit -am "${{ github.event.inputs.commit_message }}" --allow-empty # Push the committed changes to the source branch - name: push to branch shell: pwsh run: | if('${{ github.event.inputs.target_branch }}' -ne '') { git push origin ${{ github.event.inputs.target_branch }} } else { git push origin ${{ github.event.inputs.source_branch }} }
Registering Repository Secrets
Open [Settings] in your repository, navigate to [Secrets and variables] > [Actions], and select [New repository secret]. data:image/s3,"s3://crabby-images/ce90a/ce90a401c7f6b02b43958e35c21df8a0253b2138" alt=""
data:image/s3,"s3://crabby-images/ce90a/ce90a401c7f6b02b43958e35c21df8a0253b2138" alt=""
Configure the three values you noted during Entra ID setup:
data:image/s3,"s3://crabby-images/cdd45/cdd4513977995ed6466f40fb2ea6727bedc91aef" alt=""
This completes the repository and Actions configuration setup.
- CLIENT_ID: Your Client ID
- CLIENT_SECRET: Your Secret value
- TENANT_ID: Your Tenant ID
data:image/s3,"s3://crabby-images/ac53b/ac53b9eadcb63617e0e9c18f008af22e163b5f35" alt=""
data:image/s3,"s3://crabby-images/bd204/bd2045a3a2a0ae5d8b328549100cea5e157d9b26" alt=""
data:image/s3,"s3://crabby-images/cdd45/cdd4513977995ed6466f40fb2ea6727bedc91aef" alt=""
GitHub: Creating Access Token
Open your GitHub tenant (user) settings, navigate to [Developer Settings] > [Personal access tokens] > [Fine-grained tokens], and click [Generate new token]. data:image/s3,"s3://crabby-images/f22dc/f22dce7232bbb56196201a2cb922d2e02efb9ac1" alt=""
data:image/s3,"s3://crabby-images/f22dc/f22dce7232bbb56196201a2cb922d2e02efb9ac1" alt=""
Set an appropriate name and token expiration date data:image/s3,"s3://crabby-images/ee488/ee4889747f577bde6e195b980bd6225a510b096f" alt=""
data:image/s3,"s3://crabby-images/ee488/ee4889747f577bde6e195b980bd6225a510b096f" alt=""
Limit the Repository access to the repository you just created data:image/s3,"s3://crabby-images/b2474/b247482bbb134d54e880dfc26f7de96bf36fc12f" alt=""
data:image/s3,"s3://crabby-images/b2474/b247482bbb134d54e880dfc26f7de96bf36fc12f" alt=""
Under Permissions, set both [Actions] and [Contents] to “Read and Write” data:image/s3,"s3://crabby-images/5c918/5c918d2ef4b3a8b61180a012c8ccf6ccc07be52a" alt=""
data:image/s3,"s3://crabby-images/5c918/5c918d2ef4b3a8b61180a012c8ccf6ccc07be52a" alt=""
Click [Generate] and make note of the GitHub access token that is generated data:image/s3,"s3://crabby-images/32142/3214288836ae4d6f0c3db0cd7ddd165347230146" alt=""
This completes the access token creation process.
data:image/s3,"s3://crabby-images/32142/3214288836ae4d6f0c3db0cd7ddd165347230146" alt=""
Power Automate Setup: GitHub API Call
Finally, let’s build the Power Automate flow that calls the GitHub Actions we created. Here’s the overall flow diagram: data:image/s3,"s3://crabby-images/e63a3/e63a305f0af317ea276373bf4e02f5e3f87977b7" alt=""
data:image/s3,"s3://crabby-images/e63a3/e63a305f0af317ea276373bf4e02f5e3f87977b7" alt=""
First, configure the trigger as shown below:
Note: While the official reference mentions “OnDeploymentRequested”, this might cause issues in some cases. For now, we’ll use this configuration.
data:image/s3,"s3://crabby-images/36152/361527de688c22a55f79f108f758aaa368a043e0" alt=""
Note: While the official reference mentions “OnDeploymentRequested”, this might cause issues in some cases. For now, we’ll use this configuration.
data:image/s3,"s3://crabby-images/36152/361527de688c22a55f79f108f758aaa368a043e0" alt=""
Next, use the “Get row by ID” action to retrieve a row from the “Deployment Stage Execution” table. data:image/s3,"s3://crabby-images/39e31/39e31187c483b1288bbca058766661d874a9bf19" alt=""
data:image/s3,"s3://crabby-images/39e31/39e31187c483b1288bbca058766661d874a9bf19" alt=""
Finally, use the HTTP action to POST to the following URL:
URL: ~~~/repos/[Tenant(Username)]/[RepositoryName]/actions/workflows/[WorkflowFileName]/dispatches
Header: Authorization : Bearer [GitHub Access Token]
Configure the Body as shown below.
Note: Adjust the source_branch and target_branch settings according to your needs.
Although it was a bit lengthy, this completes the entire setup process.
URL: ~~~/repos/[Tenant(Username)]/[RepositoryName]/actions/workflows/[WorkflowFileName]/dispatches
Header: Authorization : Bearer [GitHub Access Token]
data:image/s3,"s3://crabby-images/7704a/7704aba3242b9b2bfa1f51c6a0ec31dd5163c2e9" alt=""
Configure the Body as shown below.
Note: Adjust the source_branch and target_branch settings according to your needs.
{ "ref": "main", "inputs": { "artifact_url": "@{triggerOutputs()?['body/OutputParameters/ArtifactFileDownloadLink']}", "solution_name": "@{triggerOutputs()?['body/OutputParameters/ArtifactName']}", "user_name": "@{triggerOutputs()?['body/OutputParameters/DeployAsUser']}", "source_branch": "main", "commit_message": "@{outputs('Get row by ID')?['body/deploymentnotes']}" } }
data:image/s3,"s3://crabby-images/35c60/35c60f9b3b2c99d8f51ae49fc041d3a86bc61e5d" alt=""
data:image/s3,"s3://crabby-images/a8684/a8684aa32e6bedb5b268ad96fce9b812310fc3c5" alt=""
data:image/s3,"s3://crabby-images/4bb72/4bb72f92e3b9dfe886c57b340d6f597eb57f8565" alt=""
data:image/s3,"s3://crabby-images/76312/76312b923abfef4c702d30775e95c7816aa96b12" alt=""
Operation Verification
Change the label content of the canvas app in the solution created in the previous article to v1.0.0.4, then save and publish. data:image/s3,"s3://crabby-images/e44d0/e44d0034f79c1328b9693c0bfead86890e870960" alt=""
data:image/s3,"s3://crabby-images/e44d0/e44d0034f79c1328b9693c0bfead86890e870960" alt=""
After publishing, when you trigger the configured pipeline,
data:image/s3,"s3://crabby-images/a4f8d/a4f8dc5df6f0082666d4e41202076ba479908899" alt=""
data:image/s3,"s3://crabby-images/33227/33227f5e88a2c6ddf122f31d3e1ba5d56fa6a4ef" alt=""
data:image/s3,"s3://crabby-images/a4f8d/a4f8dc5df6f0082666d4e41202076ba479908899" alt=""
Power Automate executes and calls the GitHub Actions. data:image/s3,"s3://crabby-images/cb428/cb428b5050f4641ea52397355d4147308ce7e00e" alt=""
data:image/s3,"s3://crabby-images/cb428/cb428b5050f4641ea52397355d4147308ce7e00e" alt=""
After a short wait, the GitHub action completes successfully, data:image/s3,"s3://crabby-images/42f9e/42f9ecfec333f842363885ed73d65a3986277e23" alt=""
data:image/s3,"s3://crabby-images/42f9e/42f9ecfec333f842363885ed73d65a3986277e23" alt=""
And the solution contents are committed to GitHub. data:image/s3,"s3://crabby-images/fb744/fb744359bdac3bf2ef679e32693400020156aabf" alt=""
data:image/s3,"s3://crabby-images/fb744/fb744359bdac3bf2ef679e32693400020156aabf" alt=""
Since we modified the canvas app content, we can confirm that the .msapp file has been successfully updated. data:image/s3,"s3://crabby-images/512a8/512a8d09e78f481f58cdb298636a57dccbe1a82c" alt=""
data:image/s3,"s3://crabby-images/512a8/512a8d09e78f481f58cdb298636a57dccbe1a82c" alt=""
Actions May Sometimes Fail
When running the pipeline using exactly the same steps, data:image/s3,"s3://crabby-images/2ec09/2ec09e178b682fdb35ef82aad31b0f040fa275dd" alt=""
data:image/s3,"s3://crabby-images/2ec09/2ec09e178b682fdb35ef82aad31b0f040fa275dd" alt=""
While Power Automate executes successfully, data:image/s3,"s3://crabby-images/bbcf9/bbcf96e0c314e0f0452712ce31b2dde7a2b7ac16" alt=""
data:image/s3,"s3://crabby-images/bbcf9/bbcf96e0c314e0f0452712ce31b2dde7a2b7ac16" alt=""
GitHub Actions may sometimes fail. data:image/s3,"s3://crabby-images/e4024/e4024e8442fc7913505816a0233a345f830ebbd6" alt=""
As indicated by the error message “No file attachment found for attribute”, this likely occurs when the GitHub action attempts to retrieve the managed solution zip file before the pipeline has completed the solution packaging process.
data:image/s3,"s3://crabby-images/e4024/e4024e8442fc7913505816a0233a345f830ebbd6" alt=""
コメント