次の方法で共有


Azure Blob Storage を使用した Azure Communication Services Chat で UI ライブラリを使ったファイル共有を有効にする

重要

Azure Communication Services のこの機能は、現在プレビュー段階にあります。

プレビューの API と SDK は、サービス レベル アグリーメントなしに提供されます。 運用環境のワークロードには使用しないことをお勧めします。 一部の機能はサポート対象ではなく、機能が制限されることがあります。

詳細については、「Microsoft Azure プレビューの追加利用規約」を確認してください。

Azure Communication Services Chat では、通信ユーザー間のファイル共有を有効にすることができます。 Azure Communication Services チャットは、Teams 相互運用性チャット ("相互運用チャット") とは異なることにご注意ください。 相互運用チャットでファイル共有を有効にする場合は、Teams の相互運用性チャットでの UI ライブラリを使用したファイル共有の追加に関するページを参照してください。

このチュートリアルでは、ファイル共有を有効にするために、Azure Communication Services UI ライブラリ チャット コンポジットを構成します。 UI ライブラリ チャット コンポジットには、ファイル共有を有効にするために使用できる一連の高機能コンポーネントと UI コントロールが用意されています。 Azure Blob Storage を使用して、チャット スレッドで共有されるファイルを格納できるようにします。

重要

Azure Communication Services では、ファイル ストレージ サービスは提供されません。 ファイルを共有するために独自のファイル ストレージ サービスを使用する必要があります。 このチュートリアルでは、Azure Blob Storage を使用します。**

コードをダウンロードする

このチュートリアルの完全なコードには GitHub でアクセスしてください。 UI コンポーネントを使用してファイル共有を使用する場合は、このサンプルを参照してください。

前提条件

このチュートリアルでは、チャット コンポジットを設定して実行する方法を既に理解していることを前提としています。 チャット コンポジットを設定する方法については、チャット コンポジットのチュートリアルを参照してください。

概要

UI ライブラリ チャット コンポジットは、ホストされるファイルの URL を開発者が渡す (Azure Communication Services チャット サービスを介して送信される) ことを可能にして、ファイル共有をサポートします。 UI ライブラリは、添付されたファイルをレンダリングします。また、送信されたファイルの外観を構成するために複数の拡張機能がサポートされています。 具体的には、次の機能がサポートされています。

  1. OS ファイル ピッカーを使用してファイルを選択するための [ファイルの添付] ボタン
  2. 許可されるファイル拡張子を構成します。
  3. 複数のアップロードを有効または無効にします。
  4. ファイルの種類ごとのさまざまなファイル アイコン。
  5. 進行状況インジケーターを含む [ファイルのアップロード]/[ファイルのダウンロード] カード。
  6. ファイルのアップロードそれぞれを動的に検証して UI にエラーを表示する機能。
  7. アップロードを取り消し、アップロードしファイルを送信前に削除する機能。
  8. アップロードしたファイルを MessageThread に表示し、ダウンロードします。 非同期ダウンロードを許可します。

以下の図は、ファイル共有シナリオのアップロードとダウンロード両方の典型的なフローを示しています。 Client Managed としてマークされたセクションは、開発者が実装する必要がある構成要素を示しています。

ファイル共有の一般的なフロー

Azure BLOB を使用してファイル ストレージを設定する

チュートリアル「Azure 関数を使用してファイルを Azure Blob Storage にアップロードする」に従って、ファイル共有に必要なバックエンド コードを作成できます。

実装が完了したら、この Azure 関数を handleAttachmentSelection 関数内で呼び出して、ファイルを Azure Blob Storage にアップロードできます。 このチュートリアルの残りの部分では、以前にリンクの Azure Blob Storage チュートリアルを使用して関数を生成したことを前提としています。

Azure Blob ストレージ コンテナーのセキュリティ保護

このチュートリアルでは、お使いの Azure Blob ストレージ コンテナーでアップロードするファイルへのパブリック アクセスが許可されていることを前提としています。 実際の運用アプリケーションでは、Azure Storage コンテナーをパブリックにすることはお勧めしません。

Azure Blob ストレージにアップロードするファイルをダウンロードするには、Shared Access Signature (SAS) を使用できます。 Shared Access Signature (SAS) を使用すると、ストレージ アカウント内のリソースへのセキュリティで保護された委任アクセスが可能になります。 SAS を使用すると、クライアントがデータにアクセスする方法をきめ細かく制御できます。

ダウンロード可能な GitHub サンプルは、Azure Storage コンテンツに SAS URL を作成するための SAS の使用方法を示しています。 SAS についてさらに詳しい情報は、こちらをご覧ください。

UI ライブラリでは、React 環境を設定する必要があります。 これは、次に行う予定です。 既に React アプリをお持ちの場合は、このセクションをスキップしてかまいません。

React アプリを設定する

このクイックスタートでは、create-react-app テンプレートを使用します。 詳細については、次を参照してください。React の概要に関するページ


npx create-react-app ui-library-quickstart-composites --template typescript

cd ui-library-quickstart-composites

このプロセスが終了すると、ui-library-quickstart-composites フォルダー内に完全なアプリケーションが作成されます。 このクイックスタートでは、src フォルダー内のファイルを変更します。

パッケージをインストールする

npm install コマンドを使用して、JavaScript 用のベータ版 Azure Communication Services UI ライブラリをインストールします。


npm install @azure/communication-react@1.16.0-beta.1

@azure/communication-react はコア Azure Communication Services を peerDependencies として指定するため、アプリケーションでコア ライブラリの API を一貫して使用できます。 これらのライブラリもインストールする必要があります。


npm install @azure/communication-calling@1.24.1-beta.2
npm install @azure/communication-chat@1.6.0-beta.1

React アプリを作成する

次を実行して、create-react-app のインストールをテストしましょう。


npm run start

ファイル共有を有効にするチャット コンポジットの構成

チャット コンポジットを初期化するために必要な両方の共通変数の変数値を置き換える必要があります。

App.tsx

import { initializeFileTypeIcons } from '@fluentui/react-file-type-icons';
import {
  ChatComposite,
  AttachmentUploadTask,
  AttachmentUploadOptions,
  AttachmentSelectionHandler,
  fromFlatCommunicationIdentifier,
  useAzureCommunicationChatAdapter
} from '@azure/communication-react';
import React, { useMemo } from 'react';

initializeFileTypeIcons();

function App(): JSX.Element {
  // Common variables
  const endpointUrl = 'INSERT_ENDPOINT_URL';
  const userId = ' INSERT_USER_ID';
  const displayName = 'INSERT_DISPLAY_NAME';
  const token = 'INSERT_ACCESS_TOKEN';
  const threadId = 'INSERT_THREAD_ID';

  // We can't even initialize the Chat and Call adapters without a well-formed token.
  const credential = useMemo(() => {
    try {
      return new AzureCommunicationTokenCredential(token);
    } catch {
      console.error('Failed to construct token credential');
      return undefined;
    }
  }, [token]);

  // Memoize arguments to `useAzureCommunicationChatAdapter` so that
  // a new adapter is only created when an argument changes.
  const chatAdapterArgs = useMemo(
    () => ({
      endpoint: endpointUrl,
      userId: fromFlatCommunicationIdentifier(userId) as CommunicationUserIdentifier,
      displayName,
      credential,
      threadId
    }),
    [userId, displayName, credential, threadId]
  );
  const chatAdapter = useAzureCommunicationChatAdapter(chatAdapterArgs);

  if (!!chatAdapter) {
    return (
      <>
        <div style={containerStyle}>
          <ChatComposite
            adapter={chatAdapter}
            options={{
              attachmentOptions: {
                uploadOptions: uploadOptions,
                downloadOptions: downloadOptions,
              }
            }} />
        </div>
      </>
    );
  }
  if (credential === undefined) {
    return <h3>Failed to construct credential. Provided token is malformed.</h3>;
  }
  return <h3>Initializing...</h3>;
}

const uploadOptions: AttachmentUploadOptions = {
  // default is false
  disableMultipleUploads: false,
  // define mime types
  supportedMediaTypes: ["image/jpg", "image/jpeg"]
  handleAttachmentSelection: attachmentSelectionHandler,
}

const attachmentSelectionHandler: AttachmentSelectionHandler = async (uploadTasks) => {
  for (const task of uploadTasks) {
    try {
      const uniqueFileName = `${v4()}-${task.file?.name}`;
      const url = await uploadFileToAzureBlob(task);
      task.notifyUploadCompleted(uniqueFileName, url);
    } catch (error) {
      if (error instanceof Error) {
        task.notifyUploadFailed(error.message);
      }
    }
  }
}

const uploadFileToAzureBlob = async (uploadTask: AttachmentUploadTask) => {
  // You need to handle the file upload here and upload it to Azure Blob Storage.
  // This is how you can configure the upload
  // Optionally, you can also update the file upload progress.
  uploadTask.notifyUploadProgressChanged(0.2);
  return {
    url: 'https://sample.com/sample.jpg', // Download URL of the file.
  };

Azure Blob ストレージを使用するようにアップロード方法を構成する

Azure Blob Storage のアップロードを有効にするには、以前に宣言した uploadFileToAzureBlob メソッドを次のコードで変更します。 ファイルをアップロードするには、Azure 関数情報を置き換える必要があります。

App.tsx

const uploadFileToAzureBlob = async (uploadTask: AttachmentUploadTask) => {
  const file = uploadTask.file;
  if (!file) {
    throw new Error("uploadTask.file is undefined");
  }

  const filename = file.name;
  const fileExtension = file.name.split(".").pop();

  // Following is an example of calling an Azure Function to handle file upload
  // The https://zcusa.951200.xyz/azure/developer/javascript/how-to/with-web-app/azure-function-file-upload
  // tutorial uses 'username' parameter to specify the storage container name.
  // the container in the tutorial is private by default. To get default downloads working in
  // this sample, you need to change the container's access level to Public via Azure Portal.
  const username = "ui-library";

  // You can get function url from the Azure Portal:
  const azFunctionBaseUri = "<YOUR_AZURE_FUNCTION_URL>";
  const uri = `${azFunctionBaseUri}&username=${username}&filename=${filename}`;

  const formData = new FormData();
  formData.append(file.name, file);

  const response = await axios.request({
    method: "post",
    url: uri,
    data: formData,
    onUploadProgress: (p) => {
      // Optionally, you can update the file upload progress.
      uploadTask.notifyUploadProgressChanged(p.loaded / p.total);
    },
  });

  const storageBaseUrl = "https://<YOUR_STORAGE_ACCOUNT>.blob.core.windows.net";

  return {
    url: `${storageBaseUrl}/${username}/${filename}`,
  };
};

エラー処理

アップロードが失敗すると、UI ライブラリ チャット コンポジットにエラー メッセージが表示されます。

ファイルのアップロード エラー バー

次のサンプル コードでは、サイズ検証エラーのためにアップロードがどのように失敗するかを示します:

App.tsx

import { AttachmentSelectionHandler } from from '@azure/communication-react';

const attachmentSelectionHandler: AttachmentSelectionHandler = async (uploadTasks) => {
  for (const task of uploadTasks) {
    if (task.file && task.file.size > 99 * 1024 * 1024) {
      // Notify ChatComposite about upload failure.
      // Allows you to provide a custom error message.
      task.notifyUploadFailed('File too big. Select a file under 99 MB.');
    }
  }
}

export const attachmentUploadOptions: AttachmentUploadOptions = {
  handleAttachmentSelection: attachmentSelectionHandler
};

ファイルのダウンロード - 高度な使用法

既定では、UI ライブラリは、notifyUploadCompleted したときに設定した URL を指す新しいタブを開きます。 または、actionsForAttachment 経由で添付ファイルのダウンロードを処理するカスタム ロジックを用意することもできます。 1 つ例を見てみましょう。

App.tsx

import { AttachmentDownloadOptions } from "communication-react";

const downloadOptions: AttachmentDownloadOptions = {
  actionsForAttachment: handler
}

const handler = async (attachment: AttachmentMetadata, message?: ChatMessage) => {
   // here we are returning a static action for all attachments and all messages
   // alternately, you can provide custom menu actions based on properties in `attachment` or `message` 
   return [defaultAttachmentMenuAction];
};

const customHandler = = async (attachment: AttachmentMetadata, message?: ChatMessage) => {
   if (attachment.extension === "pdf") {
    return [
      {
        title: "Custom button",
        icon: (<i className="custom-icon"></i>),
        onClick: () => {
          return new Promise((resolve, reject) => {
              // custom logic here
              window.alert("custom button clicked");
              resolve();
              // or to reject("xxxxx") with a custom message
          })
        }
      },
      defaultAttachmentMenuAction
    ];
  } else if (message?.senderId === "user1") {
    return [
      {
        title: "Custom button 2",
        icon: (<i className="custom-icon-2"></i>),
        onClick: () => {
          return new Promise((resolve, reject) => {
            window.alert("custom button 2 clicked");
            resolve();
          })
        }
      },
      // you can also override the default action partially
      {
        ...defaultAttachmentMenuAction,
        onClick: () => {
          return new Promise((resolve, reject) => {
            window.alert("default button clicked");
            resolve();
          })
        }
      }
    ];
  }
}

ダウンロード中に問題が発生し、ユーザーに通知する必要がある場合は、onClick 関数のメッセージでエラーを throw するだけで、Chat Composite の上のエラー バーにメッセージが表示されます。

ファイル ダウンロード エラー

リソースをクリーンアップする

Communication Services サブスクリプションをクリーンアップして解除する場合は、リソースまたはリソース グループを削除できます。 リソース グループを削除すると、それに関連付けられている他のリソースも削除されます。 「Azure Communication Services のリソースのクリーンアップ」と「Azure Functions のリソースのクリーンアップ」で詳細を確認できます。

次のステップ

次のことも実行できます。