外部添付ファイルと提出

これは、Classroom アドオンのチュートリアル シリーズの 7 番目のチュートリアルです。

このチュートリアルでは、ウェブ アプリケーションに動作を追加して、Google Classroom の外部からアドオンの添付ファイルを作成します。この動作を使用すると、ユーザーは既存のプロダクトやウェブサイトからアドオンの添付ファイルを作成できます。また、CourseWork との統合にも最適です。ユーザーのフローを変更することなく、既存のトラフィックをアドオンが提供する改善されたユーザー エクスペリエンスに誘導できるためです。推奨されるプロセスについては、Classroom 以外で添付ファイルを作成するガイドページをご覧ください。

また、アドオンに動作を追加して、アドオンの添付ファイルを使用して課題をプログラムで変更します。課題の作成者に関係なく、アドオンの添付ファイルが追加されている課題はすべて変更できます。これは、生徒がアクティビティを完了した後に課題を提出し、割り当てられたタスクが完了し、生徒の課題がレビューの準備ができたことを教師に知らせる場合に特に便利です。

コンテンツ タイプまたはアクティビティ タイプ添付ファイルをサポートするアドオンの最終バージョンを拡張します。このガイドでは、content-type 添付ファイルを使用します。

課題管理の OAuth スコープを追加する

アプリケーションが次のスコープをリクエストしていることを確認します。

  • https://www.googleapis.com/auth/classroom.addons.teacher
  • https://www.googleapis.com/auth/classroom.addons.student
  • https://www.googleapis.com/auth/classroom.coursework.students

classroom.coursework.students スコープはこれまで必要ありませんでしたが、CourseWork 割り当ての作成または変更に使用されます。このスコープを、Cloud プロジェクトの Google Workspace Marketplace SDKOAuth 同意画面、サーバーコードのスコープのリストに追加します。

Python

  SCOPES = [
    "https://www.googleapis.com/auth/classroom.addons.teacher",
    "https://www.googleapis.com/auth/classroom.addons.student",
    "https://www.googleapis.com/auth/classroom.coursework.students",
  ]

Classroom で課題を作成する

iframe 以外のウェブページにボタンを追加する

このチュートリアルで説明するフローでは、Google 以外のプロダクトから Google Classroom の課題と添付ファイルを作成できます。実際には、これは既存のウェブサイトまたはアプリケーションである可能性があります。この例では、外部サイトとして機能するモック ウェブページを作成する必要があります。クリックすると、推奨の CourseWork フローを実行して新しい割り当てを作成する新しいルートを開くボタンまたはリンクが必要です。

まだボタンやリンクがない場合は、ユーザーがログインできるようにボタンやリンクを追加する必要があります。後続の API リクエストを行うにはユーザー認証情報が必要になるため、OAuth 2.0 ハンドシェイクを完了する必要があります。具体的な手順については、ログインの手順をご覧ください。

Python

提供されている Python の例では、最初のチュートリアル ステップで導入された /index ルートを変更します。

<!-- /webapp/templates/index.html -->
<a href="clear-credentials.html">Logout</a>
<a href="start-auth-flow.html">Login</a>

<br>

<a href="create-coursework-assignment.html">Create a CourseWork Assignment</a>

ウェブサイトの目的地を表す HTML テンプレートを追加します。このページは、CourseWork 割り当てに添付されるコンテンツを表します。

<!-- /webapp/templates/example-coursework-assignment.html -->
<h1>CourseWork assignment loaded!</h1>
<p>You've loaded a CourseWork assignment! It was created from an external web page.</p>

CourseWork 関連のルートを処理する新しい Python モジュール ファイルを作成します。提供されている例では、これは coursework_routes.py です。次の 3 つのルートを追加します。一部の内容は後で入力します。

# /webapp/coursework_routes.py
@app.route("/create-coursework-assignment")
def create_coursework_assignment():
  """
  Completes the assignment creation flow.
  """

  # Check that the user is signed in. If not, perform the OAuth 2.0
  # authorization flow.
  credentials = get_credentials()

  if not credentials:
    return start_auth_flow("coursework_assignment_callback")

  # Construct the Google Classroom service.
  classroom_service = get_classroom_service()

  pass  # To be completed later.

@app.route("/example-coursework-assignment/<assignment_type>")
def example_coursework_assignment(assignment_type):
  """
  Renders the "example-coursework-assignment.html" template.
  """
  return flask.render_template(
      "example-coursework-assignment.html", assignment_type=assignment_type
  )

@app.route("/coursework-assignment-callback")
def coursework_assignment_callback():
  """
  Completes the OAuth 2.0 handshake and stores credentials in the session.
  This is identical to the callback introduced in the sign-in walkthrough,
  but redirects the user to the index page instead of the attachment
  discovery page.
  """
  flow = google_auth_oauthlib.flow.Flow.from_client_secrets_file(
      CLIENT_SECRETS_FILE,
      scopes=SCOPES,
      state=flask.session["state"],
      redirect_uri=flask.url_for("coursework_assignment_callback", _external=True),
  )

  flow.fetch_token(authorization_response=flask.request.url)

  credentials = flow.credentials
  flask.session["credentials"] = session_credentials_to_dict(
      credentials
  )

  # Close the current window and redirect the user to the index page.
  return flask.render_template("close-me.html", redirect_destination="index")

ユーザーが添付ファイルを作成できるかどうかを確認する

ユーザーに代わってアドオンの添付ファイルを作成するには、ユーザーが満たす必要のある前提条件がいくつかあります。利便性を高めるため、Google は、ユーザーがこれらの前提条件を満たしているかどうかを判断する userProfiles.checkUserCapability メソッドを提供しています。前提条件を満たしているユーザーは、対象ユーザーと呼ばれます。

CourseWork 作成ルートの実装に適格性チェックを追加します。次に、レスポンスの allowed フィールドをテストします。対象となるユーザーについては、ロジックに沿ってアドオンの添付ファイル付きの割り当てを作成します。それ以外の場合は、リンク マテリアルを作成します。ユーザーが課題を作成するコースの ID を知っておく必要があります。通常は、使用するコースを指定するようユーザーに求めます。わかりやすくするため、この例ではハードコードされた値を使用します。

Python

# /webapp/coursework_routes.py
@app.route("/create-coursework-assignment")
def create_coursework_assignment():
  """
  Completes the assignment creation flow.
  """
  # ... Check that the user is signed in and get the Classroom service ...

  # Check whether the user can create add-on attachments.
  eligibility_response = (
      classroom_service.userProfiles()
      .checkUserCapability(
        userId="me",
        capability="CREATE_ADD_ON_ATTACHMENT",
        # The previewVersion is necessary while the method is available in the
        # Workspace Developer Preview Program.
        previewVersion="V1_20240930_PREVIEW",
      ).execute()
  )
  is_create_attachment_eligible = eligibility_response.get("allowed")

  if is_create_attachment_eligible:
    # See the "Create an assignment with add-on attachment for eligible users" section for implementation.
  if not is_create_attachment_eligible:
    # See the "Create a Link Material" section for implementation.

対象ユーザー向けにアドオンの添付ファイル付きの課題を作成する

ユーザーがアドオンの添付ファイルを作成できる場合は、次の操作を行います。

  1. API リクエストを送信して、添付ファイルなしcourseWork 課題を Google Classroom で作成します。
  2. 新しく作成された課題の id を抽出します。
  3. 新しい CourseWork AddOnAttachment を作成します。
  4. Google Classroom で新しく作成された課題にアドオンの添付ファイルを作成するリクエストを送信します。

Python

# The ID of the course to which the assignment will be added.
course_id = 1234567890  # TODO(developer) Replace with an actual course ID.

# /webapp/coursework_routes.py
if is_create_attachment_eligible:
  # Create an assignment.
  coursework = {
      "title": "My CourseWork Assignment with Add-on Attachment",
      "description": "Created using the Classroom CourseWork API.",
      "workType": "ASSIGNMENT",
      "state": "DRAFT",  # Set to 'PUBLISHED' to assign to students.
  }

  # Issue a request to create the assignment.
  create_assignment_response = (
      classroom_service.courses()
      .courseWork()
      .create(courseId=course_id, body=coursework)
      .execute()
  )

  # Create an add-on attachment that links to the selected content and
  # associate it with the new assignment.
  content_url = flask.url_for(
      "example_coursework_assignment",
      assignment_type="add-on-attachment",
      _scheme="https",
      _external=True,
  )

  # Construct an AddOnAttachment instance.
  attachment = {
      "teacherViewUri": {"uri": content_url},
      "studentViewUri": {"uri": content_url},
      "title": f'Test Attachment for Assignment {create_assignment_response.get("id")}',
  }

  # Issue a request to create the attachment.
  add_on_attachment_response = (
      classroom_service.courses()
      .courseWork()
      .addOnAttachments()
      .create(
          courseId=course_id,
          itemId=create_assignment_response.get("id"),  # ID of the new assignment.
          body=attachment,
      )
      .execute()
  )

ユーザーがアドオンの添付ファイルを作成する資格がない場合は、次の手順で代わりにリンク マテリアルを作成します。

Python

# The ID of the course to which the assignment will be added.
course_id = 1234567890  # TODO(developer) Replace with an actual course ID.

if not is_create_attachment_eligible:
    coursework = {
        "title": "My CourseWork Assignment with Link Material",
        "description": "Created using the Classroom CourseWork API.",
        "workType": "ASSIGNMENT",
        "state": "DRAFT",  # Set to 'PUBLISHED' to assign to students.
        # Specify the URL for your content as a Link Material.
        "materials": [
            {
                "link": {
                    "url": flask.url_for(
                        "example_coursework_assignment",
                        assignment_type="link-material",
                        _scheme="https",
                        _external=True,
                    )
                }
            }
        ],
    }

    # Issue a request to create the assignment.
    assignment_response = (
        classroom_service.courses()
        .courseWork()
        .create(courseId=course_id, body=coursework)
        .execute()
    )

作成済みの課題を変更する

ストリーム アイテムを作成したユーザーに関係なく、アドオンの添付ファイルが 1 つ以上含まれている Google Classroom のストリーム アイテムには、アクセス、変更、提出、再提出、返品を行うことができます。ストリーム アイテムは、AnnouncementCourseWork 課題、CourseWorkMaterial のいずれかです。

このことを示すために、指定されたストリーム アイテムを変更するルートを追加します。このメソッドを使用して、API を使用して作成したストリーム アイテムと、Google Classroom UI を通じて教師が作成したストリーム アイテムにアクセスして変更できることを確認します。

このチュートリアルで最初に編集したウェブページに、もう 1 つリンクまたはボタンを追加します。CourseWork 割り当てを変更する新しいルートが開きます。

Python

提供されている Python の例では、このチュートリアルの前の方で変更した /index ルートを変更します。

<!-- /webapp/templates/index.html -->
<a href="modify-coursework-assignment.html">Create a CourseWork Assignment</a>

CourseWork 関連のルートを処理する新しいルートを作成します。これは、提供されている例の coursework_routes.py ファイルにあります。

# Check that the user is signed in.
credentials = get_credentials()

if not credentials:
  return start_auth_flow("coursework_assignment_callback")

# Get the Google Classroom service.
classroom_service = get_classroom_service()

# The ID of the course to which the assignment will be added.
# Ordinarily, you'll prompt the user to specify which course to use. For
# simplicity, we use a hard-coded value in this example.
course_id = 1234567890  # TODO(developer) Replace with an actual course ID.
assignment_id = 1234567890  # TODO(developer) Replace with an actual assignment ID.

# Retrieve details about the CourseWork assignment.
get_coursework_response = (
    classroom_service.courses()
    .courseWork()
    .get(courseId=course_id, id=assignment_id)
    .execute()
)

# Alter the current title.
assignment_title = f"{get_coursework_response.get('title')} (Modified by API request)"

# Issue a request to modify the assignment.
modify_coursework_response = (
    classroom_service.courses()
    .courseWork()
    .patch(
        courseId=course_id,
        id=assignment_id,
        updateMask="title",
        body={"title": assignment_title},
    )
    .execute()
)

アドオンをテストする

わかりやすくするため、提供されている例ではコースと課題の識別子がハードコードされています。これらの識別子は、教師の認証情報を使用して courses リソースと courseWork リソースの get メソッドと list メソッドにリクエストを送信することで取得できます。また、courseWork 割り当ての作成時にレスポンスで返されます。

サーバーを実行し、インデックス ページに移動して、Google Workspace for Education Teaching & Learning または Plus のライセンスがない教師ユーザーとしてログインします。テスト ドメインの管理コンソールでユーザーのライセンス ステータスを切り替えることができます。[Create a CourseWork Assignment] ボタンをクリックし、Google Classroom の UI を開いて、リンク マテリアルが添付された課題が作成されていることを確認します。添付ファイルには、リンク先のウェブページのタイトルと URL が表示されます。

アドオン添付ファイルの作成をテストする

インデックス ページに戻り、Google Workspace for Education の Teaching & Learning または Plus ライセンスを持つ教師ユーザーとしてログインします。[Create a CourseWork Assignment] ボタンをクリックし、Google Classroom の UI を開いて、アドオンの添付ファイルを含む課題が作成されたことを確認します。添付ファイルには、アドオン アプリケーションの名前とコードで指定したタイトルが表示されます。

テストの割り当ての変更

インデックス ページに戻り、Teaching and Learning または Plus ライセンスを持つ教師ユーザーとしてログインしていることを確認します。[Modify a CourseWork Assignment] ボタンをクリックし、Google Classroom の UI に戻って、課題のタイトルが変更されていることを確認します。

おめでとうございます!チュートリアル シリーズを完了しました。