Data Platform for Log Analytics - Teams AbnormalMeetings

Deployment demo here. (Note: Please login with your microsoft account)

Table of Contents

Architecture

  • Services we used
  • Work Flow
    • The RenewSubscription azure function is our entry point
    • It will subscribe for the resources that we needed, such as callRecord and user event. Then the information that we need will be sent to the endpoint azure functions.
    • Then the endpoint azure function will get the target resource using specific API and store them to our storage account.

General Information

Advantage

  • This is a project about Security.
  • If confidential disclosure happened, we can utilize these logs to find out who are the suspicious insiders.

Problem Definition and Scenario

  • Problem: Whether the insider exist in teams?
  • Definition of an insider:
    • They may hold or join some abnormal meetings.
  • Definition of an abnormal meetings:
    • 1 on 1 meeting. (especially a meeting who’s email address ends with @gmail, @hotmail, etc.
    • Screen sharing in meeting would be suspicious.

Steps To Demo

Prerequisite

  1. Having a teams admin account.
  2. Register an Application having following permissions.
    1. User.Read.All (For "List Users“)
    2. Calendars.Read (For "List Events“)
    3. CallRecords.Read.All (For "CallRecords“)
    • How to add permissions

Deployment

  1. Publish the project.

    • Open the project in Visual Studio 2022. (The project can be clone from Github.)
    • Publish it to Azure Functions (You can also create a new one here).
      • Right click "Publish..." in Visual Studio 2022 vsPublish_withSteps
      • Choose to publish in "Azure" -> "Azure Function App (Windows) -> "AbnormalMeetings"

        Or just Create new here. (recommanded)

      • Click "finish" botton vsPublish_profile
      • Then click "publish" button.
  2. Fill in needed configuration

    You can fill one after one by yourself. Or use my tool below to generate config by copy-and-paste

    • The explination of the config here
    • Fill needed parameters in src\AbnormalMeetings\local.settings.json & AssistTools\originalConfig.json
    • Run the python script AssistTools\settingsToConfig.py (this is a tool that I write to save time to fill the config one after one)
    • Paste config to azure function configuration.
  3. Initialize the service

    • Use postman to trigger "renewsubscription" to subscribe all needed resource
      • POST to https://<Endpoint_Name>.azurewebsites.net/admin/functions/RenewSubscription
        • Headers:
          • x-functions-key: <your_master_key>
          • Content-Type:application/json
        • Body: { "input": "test" }
  4. Bind with ADX

Demo

UserEvents

  • Step 1: [Create/Modify] a event in [outlook/teams].
  • Step 2: Wait 30 sec.
  • Step 3: Go to your blob storage or ADX to check your log!

CallRecords

  • Step 1: [Join/Create] a [call/meeting] in teams.
  • Step 2: Wait others to join.
  • Step 3: [Share/not share] your screen.
  • Step 4: After all tests, end the [call/meeting].
  • Step 5: Wait 30 minutes.
  • Step 6: Go to your blob storage or ADX to check your log!

Setup

Setting Environment Variables (Configuration)

  • Belows are all needed configuration variables.
  • If you are deploying them on Azure, please go to "Function App" -> "AbnormalMeetings" -> "Configuration" (in settings) and add/modify the configuration. azPortal_functionApp_configuration

Variables Needed Modified

  • local.setting.json
    {
        "IsEncrypted": false,
        "Values": {
            "AzureWebJobsStorage": "UseDevelopmentStorage=true",
            "FUNCTIONS_WORKER_RUNTIME": "dotnet",
            "AzureWebJobsSecretStorageType": "files",
            "IsChatApi": "true",
    
            "BlobConnectionString": "",
    
            "Tenant": "",
            "ClientId": "",
            "ClientSecret": "",
    
            "FunctionAppName": "",
            "FunctionDefaultKey": ""
        }
    }

Using Chat Api

  • "IsChatApi"
    • This is a feature flag that is used to decide whether the application can get the chatMessage

Blob storage related

  • "BlobConnectionString"
    • BlobConnectionString is the blob connectionstring, where you want to store your data. In this case, it is the connection string of abnormalmeetingstorage

Azure Active Directory

  • "Tenant",
  • "ClientId",
  • "ClientSecret",
How to get Tenant, ClientId, ClientSecret
  • You have already get the ClientSecret here
  • Go to "Azure Active Dirrectory" -> "App Registrations" -> "Overview" and record several infornmation
    • Tenant: Specified in "Directory (tenant) ID"
    • ClientId: Specified in "Application (client) ID"

Function URLs (for webhook)

  • "FunctionAppName"
    • The name of your azure function app (endpoint)
  • "FunctionDefaultKey"
    • The master key of azure function app

Features

Here we describe what each .cs file does.

RenewSubscription.cs

  • All subscription-related tasks will be done here.
  • The program will read the json file specifiled in BlobFileName, under the container specified in BlobContainerName_SubscriptionList.
  • The json file's format is, for example,
    {
        "value":[
            {
                "UserId":"7f5b5e74-ba40-4aed-95be-d55a57e684fa",
                "SubscriptionId":"b32157a5-3dfc-4743-b84b-5dcb6c90c0ed"
            },
            ...,
            {
                "UserId":"callRecordId","SubscriptionId":"b6978c22-d086-47de-8f0b-31e68a5302f8"
            }
        ]
    }        

Renew CallRecords Subscription

  • If the CallRecords have been subscribed before and have been recorded in BlobFileName ("UserId":"callRecordId"), just simply renew the subscription; otherwise, create a subscription and record it into BlobFileName.

Renew UserEvent Subscriptions

  • I will get all users under the target tenant, if the user have been subscribed before and have been recorded in BlobFileName ("UserId":"{TheUserId}}"), just simply renew the subscription; otherwise, create a subscription and record it into BlobFileName.

GetCallRecords.cs

  • When the program recieves http request from the subscription resource, the program will be triggered.
  • Use the call Id to get the callrecord and save it as a json file to container specified in BlobContainerName_CallRecords, using graph api SDK
    CallRecord callrecord = await graphServiceClient.
    // Here we used "Expand" to get full infornmation
    // in callrecord
    Communications.CallRecords[call_Id]
                    .Request()
                    .Expand("sessions($expand=segments)")
                    .GetAsync();
  • If the environment variable IsChatApi is true, we are going to get the chatMessage from chatMessage API
    // Get the chat id from regex
    string chatid = matches[0].Value; 
    string resource = $"chats/{chatid}/messages";
    string webApiUrl = $"{config.ApiUrl}v1.0/{resource}"; // combine the resource to endpoint api
    HttpResponseMessage response = await httpClient.GetAsync(webApiUrl);

GetUserEvents.cs

  • When the program recieve http request from the subscription resource, the program will be triggered.
  • Use the resource to get the event object and save it as a json file to container specified in BlobContainerName_UserEvents, using Http request.
    // The resource are the target
    // of user event subscription resource
    string resource = subscriptionData.value[0].resource;
    string webApiUrl = $"{config.ApiUrl}v1.0/{resource}";
    HttpResponseMessage response = await httpClient.GetAsync(webApiUrl);

Technologies Used

Azure Functions Core Tools

  • Core Tools Version: 4.0.4736 Commit hash: N/A (64-bit)
  • Function Runtime Version: 4.8.1.18957

PackageReference

  • "Microsoft.NET.Sdk.Functions" - Version="4.0.1"
  • "Microsoft.Graph" - Version="4.35.0"
  • "Microsoft.Identity.Client" - Version="4.45.0"
  • "Microsoft.Identity.Web" - Version="1.25.1"
  • "Microsoft.Azure.WebJobs.Extensions.Storage.Blobs" - Version="5.0.1"

Project Status

Project is: complete

Room for Improvement

  • Error handling functions

Acknowledgements

Many thanks to Goerge Liang, my mentor in MS.

File / Folder Architecture

Folder

  • AssistTools: Can use this to generate configuration to fill in azure
  • img: all image files for README.md
  • src: source code of project

Contact

Created by @Eric - feel free to contact me!