page_type | description | products | languages | extensions | urlFragment | ||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|
sample |
In-Meeting Document Signing |
|
|
|
officedev-microsoft-teams-samples-meetings-share-to-stage-signing-csharp |
Meeting Signing is a Teams sample app that demonstrates the scenario of enabling users to sign documents in real time during meetings making possible to review and sign documents in a single session. This sample lets users electronically sign documents with their current tenant identity. Some example documents that can be signed during meetings are Purchase Agreements and Purchase Orders. This template is also applicable, and could be adapted for other scenarios like reviewing and approving code reviews or collaborative reviewing of documents.
The sample app could be customized to incorporate other kind of signatures like handwritten signatures or certificate based signatures.
This sample has 3 main personas:
- Document creator: this role can add their own documents to be signed using the app
- Signer: this role will be able to sign documents reviewed during the meeting
- Reader: this role will only be able to read the documents reviewed during the meeting
- Add documents to be reviewed during a meeting
- Share documents to be reviewed to main stage
- Sign documents using the signer’s identity
- Teams SSO
- Programmatic sharing of content to stage
- Stage app view depends on the authentication of the user
- Adaptive Cards
- People Picker
- Anonymouse User support
- MSAL.js 2.0 support
sequenceDiagram
Teams User->>+Teams Client: Schedules a Teams Meeting with members
Teams Client->>+MeetingSigning App: Installs the App
Teams User->>+Teams Client: Pre meeting, organises the documents
Teams Client->>+MeetingSigning App: Opens the pre-meeting tab
MeetingSigning App->>+Document Creation : From pre-meeting tab, creates a document and assigns signers & readers
Document Creation-->>-MeetingSigning App: Document created
Teams User->>+Teams Client: Starts the meeting
Teams User->>+MeetingSigning App: Opens the MeetingSigning side panel
MeetingSigning App->>+Side Panel: Loads documents
Side Panel-->>-MeetingSigning App: Loads meeting organizer's documents
MeetingSigning App->>+Side Panel: Select document to share to stage
Side Panel-->>-Teams Client: Tells the team client to open a document on the stage
Teams Client->>+Document Stage: Tells the app which document to open
Document Stage-->>-MeetingSigning App: Shares the document to share to stage in the meeting
-
.NET Core SDK version 6.0
# determine dotnet version dotnet --version
-
Publicly addressable https url or tunnel such as ngrok or Tunnel Relay
NOTE: The free ngrok plan will generate a new URL every time you run it, which requires you to update your Azure AD registration, the Teams app manifest, and the project configuration. A paid account with a permanent ngrok URL is recommended.
-
Ensure that you've enabled the Teams Channel
NOTE: When you create your bot you will create an App ID and App password - make sure you keep these for later.
-
Make sure to copy and save the
https
url (it should look likehttps://<randomsubdomain>.ngrok-free.app
) -
Register an App in AzureAD that can be used for Teams SSO](https://docs.microsoft.com/en-us/microsoftteams/platform/tabs/how-to/authentication/tab-sso-register-aad)
-
Once the app is registered update Redirect Uris under the Authentication section with the ngrok url, followed by /auth-end (https://.ngrok-free.app/auth-end)
-
Ensure the following API permissions are granted to the app for Microsoft Graph access
- offline_access,
- openid,
- profile,
- User.Read,
- User.ReadBasic.All
-
After you create an AzureAD app, under Certificates & secrets create a new Client secret. Copy the secret value and set it in
appSettings.json
-
Note: if you restart Ngrok you may have to update any fully qualified domain name you have set in your AzureAD App
- For anonymous users we authorize who the user is by having them log in to a Microsoft Personal account. You can use the same AzureAD App as above, but we have separated the apps so things are clearer.
- Register an App in AzureAD that can be used for Microsoft Personal account users
- Register the app
- We tested with an app that supports "Accounts in any organizational directory (Any Azure AD directory - Multi-tenant) and personal Microsoft accounts"
- Expose the API
- Configure the API scopes
- You do not need "To configure authorized client application"
- Ensure the following API permissions are granted to the app for Microsoft Graph access - email, openid
- Note: if you restart Ngrok you may have to update any fully qualified domain name you have set in your AzureAD App
- After you create an AzureAD app, under Certificates & secrets create a new Client secret. Copy the secret value and set it in
appSettings.json
- Register the app
# ngrok http https://localhost:44326 --host-header=localhost:44326
- The sample contains 3 projects
* `Web` - Exposes REST APIs for documents and signing scenarios supported in this POC.
* `Web\ClientApp` contains the Front End code to support document sharing in a meeting via share to stage.
* `Domain` - Contains the business logic to support the REST APIs.
* `Infrastructure` - Fulfils `Domain`'s dependencies like data repositories, graph support needed.
-
Clone the repository
git clone https://github.com/OfficeDev/Microsoft-Teams-Samples.git
-
Install modules navigate to
samples/meetings-share-to-stage-signing/csharp/Source/MeetingSigning/ClientApp
npm install
-
In
appSettings.json
,manifest.json
and.env
replace:<<deployment-url>>
with your ngrok url, minus the https://.<<aad-id>>
with your AzureAD Application (Client) Id.<<client secret>>
with the client secret you created above.<<msa-only-aad-client-id>>
with the Application (Client) Id from the AzureAD App for personal users<<msa-only-aad-client-secret>>
with the client secret from the AzureAD App for personal users that you created above<<msa-only-scope>>
api://<<deployment-url>>/<<msa-only-aad-client-id>>/access_as_user email openid
-
Run the bot from a terminal, Visual Studio or Docker choose one of the following options:
- Point Ngrok to port 5001:
ngrok http -host-header=rewrite 5001
- In a terminal, navigate to
samples/meetings-share-to-stage-signing/csharp/Source/MeetingSigning
- Run
dotnet run
- Point Ngrok to port 44326:
ngrok http 44326 --host-header="localhost:44326"
- Launch Visual Studio
- File -> Open -> Project/Solution
- Navigate to
samples/meetings-share-to-stage-signing/csharp/Source
folder - Select
MeetingSigning.sln
file - Ensure the start-up project is set to
Microsoft.Teams.Samples.MeetingSigning.Web
- Press
F5
to start Debugging using IIS Express
Note the below instructions are using Podman, but Docker's commands are similar. There are instructions for setting up Podman on WSL2 here
- From this directory build the Docker image
podman build -f Deployment/Dockerfile --ignorefile Deployment/.dockerignore ./Source --build-arg REACT_APP_AAD_CLIENT_ID --build-arg REACT_APP_MSA_ONLY_CLIENT_ID --build-arg REACT_APP_MSA_ONLY_SCOPE
- Wait for the container to build
- Run
podman images
to view available images, copy the Image ID - Point Ngrok to port 8080:
ngrok http -host-header=rewrite 8080
- Run
podman run -d -p 8080:80 --name MeetingSigning <IMAGE_ID>
to start the container - Open http://localhost:8080/ to view the service running
- Point Ngrok to port 5001:
-
This step is specific to Teams.
- Edit the
manifest.json
contained in theSource\MeetingSigning.Web\Manifest
folder to replace your Microsoft App Id (that was created when you registered your bot earlier) everywhere you see the place holder string<<Your Microsoft App Id>>
(depending on the scenario the Microsoft App Id may occur multiple times in themanifest.json
) - Edit the
manifest.json
forconfigurationUrl
insideconfigurableTabs
. Replace<yourNgrok.ngrok-free.app>
with base Url domain. E.g. if you are using ngrok it would behttps://1234.ngrok-free.app
then your domain-name will be1234.ngrok-free.app
. - Edit the
manifest.json
forvalidDomains
with base Url domain. E.g. if you are using ngrok it would behttps://1234.ngrok-free.app
then your domain-name will be1234.ngrok-free.app
. - Zip up the contents of the
Source\MeetingSigning.Web\Manifest
folder to create amanifest.zip
(Make sure that zip file does not contains any subfolder otherwise you will get error while uploading your .zip package) - Upload the
manifest.zip
to Teams (In Teams Apps/Manage your apps click "Upload an app". Browse to and Open the .zip file. At the next dialog, click the Add button.)
- Edit the
Install app:
Add Meeting Signing:
Create documents:
Join call and add app:
App meeting side panel:
Click Share Meeting:
Click to sign:
Click done:
Create multiple document:
Create multiple signing and document:
Multiple document UI:
Multiple share meeting UI:
- Sideload the app to a meeting.
- In the meeting group chat, add the Meeting Signing app as a new Tab.
- In the pre-meeting tab, create a test document so you can test the meeting share to stage APIs.
- Provide one-time consent to the app for Graph permissions while creating the document.
- Start the meeting.
- In the bar at the top, select Meeting Signing to open the sidepanel.
- You will see any documents you created here
- Select Share to Meeting on a document you want to share to stage.
- All participants of the meeting will see the app being shared to the stage. Participants who are either signers, viewers or document creators will be able to see the document. Participants not a signer, viewer or creator will see an error stating they do not have permissions to view the document.
- Signers are able to sign a document, and all viewers will have their view of their document updated to include that signature.
- Anonymous Users
- When you join the meeting, if a document is shared to stage, you will be prompted to log in to your personal Microsoft account.
- Once you log in, if your email matches those defined in the document details you will be able to view the document.
- A user/meeting attendee can be either a viewer or a signer. If neither, an error message "you aren't allowed to see this document" will be displayed.
- A viewer can only view the document.
- All signers can view and sign a document.
We have added Live Share to this demo to showcase some features of the SDK. The examples we are showing are scroll syncing, taking control and shared cursors.
- Scroll Offset Syncing, when the person in control scrolls their view of the document, everyone following will see their document scroll too. We the controller scrolls we send an LiveEvent that viewers listen for, and if the viewer is following the controller their view will update. If the viewer has scrolled to a different part of the document, their following of the controller will be suspended until they follow the controller again.
- Take Control allows for other viewers to become the scroll controller. Only those that have an approved role (in our example Organiser and Presenter) will be able to claim control.
- Shared Cursors shares presenters cursors across every attendees screen. The approach we are using is primitive and uses DOM manipulation to position the cursors. If you plan to implement cursor support in your app using the live-share-canvas SDK is recommended.
We have discussed some more thoughts on Live Share here
Currently, this app is not fully supported in the following scenarios:
- Mobile
- We are tracking two separate issues that cause issues with opening of the task module and the submitting of the adaptive card used for signing to not work on iOS and Android.
- People Picker is used in Document Creation. As of March 2022, people picker on mobile is in public developer preview.
- The current generally available Teams for Web does not support meeting apps.
- Tenant - If assigned by the document creator, the User will be able to see and sign the document.
- Federated/Guest Users:
- The people picker does not allow users outside of the tenant to be selected. Similarly, if a federated user creates the document, they are only able to select people in their tenant as signers/viewers, and nobody from outside their tenant can view the document.
- Anonymous Users
- Coming soon! When creating a document you can declare the emails of the anonymous users who should view or sign the document. The app prompts anonymous meeting participants to sign in with their Microsoft account, then uses the login information to validate that their email address is allowed to access the document. You can extend this approach to support other identity providers.
Below is a diagram on how the auth flow works for anonymous and non-anonymous users (AzureAD in-tenant, federated and guest)
graph TD
Join[User joins meeting] -->|Document is shared to stage| IsUserAnonymous{Is User Anonymous?}
IsUserAnonymous -->|No| GetAuthToken[GetAuthToken via Teams SSO]
IsUserAnonymous -->|Yes| AnonLogin1[Log in using 3P identity provider]
AnonLogin1 --> AnonLogin2[[Login with personal MSA]]
AnonLogin2 --> AnonLogin3[[Get auth token]]
GetAuthToken & AnonLogin3--> GetDocument[Call GetDocument Api with auth token]
GetDocument --> UserAuthorized{Is user authorized?}
UserAuthorized --> |Yes| DocumentReturned[Document returned and rendered]
UserAuthorized --> |No| NotAuthorised[Unauthorised error]
NotAuthorised --> IsAnonymousAndNotAuth{Is User Anonymous?}
IsAnonymousAndNotAuth --> |No| AadNotAuth[Show Not Authorised error]
IsAnonymousAndNotAuth --> |Yes| MsaNotAuth[Show option to login with 3P Identity provider]
- When the solution is run on a local web browser (anywhere outside of Teams), it will load an expected error message stating that " Unable to get information about the App. This happens if you are running the application in a normal browser, and not inside Teams. Install the app inside teams to test this application. To upload the app to Teams follow the instructions on https://docs.microsoft.com/en-us/microsoftteams/platform/concepts/deploy-and-publish/apps-upload"
- "Something went wrong" when opening the app in the meeting. If you have yet to authorise the application, and you open the app in the meeting, Teams doesn't seem to prompt for you to authorise the application. To solve this, if you create a document in the pre-meeting flow, you should be prompted to auth there and the app will be able to get a token when in a meeting next time.
- Anonymous users in private windows are asked to login multiple times.
- We use MSAL.js with localStorage to log anonymous users in and to save the login details between contexts (e.g. SidePanel and Stage). However, in private windows third-party cookies and localStorage can not be set, and as the Teams app model is to iframe the tab, the application is considered a third-party.
This repository uses VSCode Code Tours to explain how the code works.
The tour files can be found in the .tours
directory.