> For the complete documentation index, see [llms.txt](https://catenoid-support.gitbook.io/kollus-dev-jp/llms.txt). Markdown versions of documentation pages are available by appending `.md` to page URLs; this page is available as [Markdown](https://catenoid-support.gitbook.io/kollus-dev-jp/callback-gai-yao/drm-callback.md).

# DRM Callback

### 概要 <a href="#drmcallback-gai-yao" id="drmcallback-gai-yao"></a>

KollusモバイルダウンロードDRMを利用してダウンロードしたコンテンツの「再生回数」、「再生有効期限」、「再生時間」を設定してコンテンツの再生を制御する機能について説明する文書です。サービス提供形態によって重複制限の設定として使用されることもあります。

#### Expire option <a href="#drmcallback-expireoption" id="drmcallback-expireoption"></a>

設定項目のdata-typeや値が範囲から外れた場合、エンドユーザーのコンテンツの利用に問題が発生する可能性があります。また、誤った設定値による使用回数の過剰などを回収する方法はありませんので設定に注意してください。

* Expire count : 再生回数の制限
  * data-type : integer
  * range : 0 \~ 1000
  * 0 : unlimited (無制限)
* Expire date : 有効期限の制限
  * data-type : integer, unixtime stamp
  * コンテンツの有効期限(再生可能期限)
    * 終了日(日時)\
      例) 2014. 3. 3.  5時 45分 30秒 GMT → 1393825531
    * 0 : unlimited (無制限)
    * 最大値 : 2029年 12月 31日 23時 59分 59秒 (1893455999)
* Expire playtime : 再生時間(倍速再生の場合は倍速が適用された時間で計算)
  * data-type : integer
  * range : 0, 60 \~ 604800 (単位: 秒, min : 60秒, max:1週間)
  * 0 : unlimited (無制限)

#### DRM callback <a href="#drmcallback-drmcallback" id="drmcallback-drmcallback"></a>

DRM ポリシーを適用するためには、チャンネルにDRM Callback URLを設定する必要があります。

チャンネルにDRM Callback URLを設定すると、コンテンツをダウンロードする時点にDRM Callback URLを呼び出して、リターン値をDRMポリシーとして使用します。ダウンロードされるDRMポリシー情報はJWT EncodeされてHttp Bodyにリターンしなければなりません。

注意:

1. DRM Callback URLがレスポンスしないとダウンロードできません。
2. アルゴリズムはHS256のみ対応しており、Httpヘッダに指定された(X-KOLLUS-USERKEY)ヘッダに“ユーザーキー”を共に転送しなければなりません。

#### **Callback ​flow**

<figure><img src="/files/84VEQMT2yCQCPwxWb4u9" alt=""><figcaption></figcaption></figure>

1. 配信チャンネルに DRM CallbackURLを設定します。
   * ex> <http://www.foo.com/auth.php> を顧客のDRM認証サーバーとする
2. JSONデータを生成してJWTでエンコードする
3. Kollus mobile playerが<http://www.foo.com/auth.php> に以下の情報をPOST転送します。
   * session\_key : Playerが生成したリクエスト確認用のセッションキー
     * kind3のcontent\_expire\_resetの場合、リクエストしたsession\_keyを確認します。
     * Callback v2が適用されるv1.6以降から使用可能です。
   * kind : 1 - 3
   * client\_user\_id : ユーザID(サービス会員情報)
     * Media token 生成の際に含まれたID
   * player\_id : ユーザのデバイスID
   * device\_name : ユーザのデバイス名
     * Android, iOS別に転送される端末名が異なります。
     * iOS：事前にApple社が定義した文字列で転送されます。
     * Android：デバイス名、モデル名が転送されます。※該当する情報がない場合NULLで転送
   * media\_content\_key : ダウンロードするコンテンツキー
     * チャンネルに登録されたコンテンツのメディアコンテンツキー(ユニーク)
     * 同じコンテンツを複数のチャンネルに登録する場合、それぞれのmedia\_content\_keyは全て異なります。
4. 顧客のDRM認証サーバーは転送された上記の情報に基づいて以下のjson フォーマットのdataをJWT payloadに追加してEncoding します。ヘッダーに指定された“ユーザキー(X-KOLLUS-USERKEY)"を一緒に転送します。\
   \&#xNAN;**(顧客が認証データをPlayerに転送する全てのデータは必ずinteger型で転送しなければなりません。)**

#### **device\_name 追加説明**

* Andoid

  Androidアプリケーションの開発に使用される Build.DEVICE, Build.MODELは /(スラッシュ)で区分した文字列を作成して使用してください。

  * Build.DEVICE+”/”+Build.MODEL
  * デバイスによって該当情報はNULLで表示される可能性があります。
* iOS

  iOSから提供するdevice-nameを使用します。

| Device Type | Product Name                           |
| ----------- | -------------------------------------- |
| iPhone1,1   | iPhone                                 |
| iPhone1,2   | iPhone 3G                              |
| iPhone2,1   | iPhone 3GS                             |
| iPhone3,1   | iPhone 4 (GSM)                         |
| iPhone3,3   | iPhone 4 (CDMA)                        |
| iPhone4,1   | iPhone 4S                              |
| iPhone5,1   | iPhone 5 (A1428)                       |
| iPhone5,2   | iPhone 5 (A1429)                       |
| iPhone5,3   | iPhone 5c (A1456/A1532)                |
| iPhone5,4   | iPhone 5c (A1507/A1516/A1529)          |
| iPhone6,1   | iPhone 5s (A1433/A1453)                |
| iPhone6,2   | iPhone 5s (A1457/A1518/A1530)          |
| iPhone7,1   | iPhone 6 Plus                          |
| iPhone7,2   | iPhone 6                               |
| iPhone8,1   | iPhone 6s                              |
| iPhone8,2   | iPhone 6s Plus                         |
| iPhone8,4   | iPhone SE                              |
| iPhone9,1   | iPhone 7 (A1660/A1779/A1780)           |
| iPhone9,2   | iPhone 7 Plus (A1661/A1785/A1786)      |
| iPhone9,3   | iPhone 7 (A1778)                       |
| iPhone9,4   | iPhone 7 Plus (A1784)                  |
| iPhone10,1  | iPhone 8 (A1863/A1906)                 |
| iPhone10,2  | iPhone 8 Plus (A1864/A1898)            |
| iPhone10,3  | iPhone X (A1865/A1902)                 |
| iPhone10,4  | iPhone 8 (A1905)                       |
| iPhone10,5  | iPhone 8 Plus (A1897)                  |
| iPhone10,6  | iPhone X (A1901)                       |
| iPad1,1     | iPad                                   |
| iPad2,1     | iPad 2 (Wi-Fi)                         |
| iPad2,2     | iPad 2 (GSM)                           |
| iPad2,3     | iPad 2 (CDMA)                          |
| iPad2,4     | iPad 2 (Wi-Fi, revised)                |
| iPad2,5     | iPad mini (Wi-Fi)                      |
| iPad2,6     | iPad mini (A1454)                      |
| iPad2,7     | iPad mini (A1455)                      |
| iPad3,1     | iPad (3rd gen, Wi-Fi)                  |
| iPad3,2     | iPad (3rd gen, Wi-Fi+LTE Verizon)      |
| iPad3,3     | iPad (3rd gen, Wi-Fi+LTE AT\&T)        |
| iPad3,4     | iPad (4th gen, Wi-Fi)                  |
| iPad3,5     | iPad (4th gen, A1459)                  |
| iPad3,6     | iPad (4th gen, A1460)                  |
| iPad4,1     | iPad Air (Wi-Fi)                       |
| iPad4,2     | iPad Air (Wi-Fi+LTE)                   |
| iPad4,3     | iPad Air (Rev)                         |
| iPad4,4     | iPad mini 2 (Wi-Fi)                    |
| iPad4,5     | iPad mini 2 (Wi-Fi+LTE)                |
| iPad4,6     | iPad mini 2 (Rev)                      |
| iPad4,7     | iPad mini 3 (Wi-Fi)                    |
| iPad4,8     | iPad mini 3 (A1600)                    |
| iPad4,9     | iPad mini 3 (A1601)                    |
| iPad5,1     | iPad mini 4 (Wi-Fi)                    |
| iPad5,2     | iPad mini 4 (Wi-Fi+LTE)                |
| iPad5,3     | iPad Air 2 (Wi-Fi)                     |
| iPad5,4     | iPad Air 2 (Wi-Fi+LTE)                 |
| iPad6,3     | iPad Pro (9.7 inch) (Wi-Fi)            |
| iPad6,4     | iPad Pro (9.7 inch) (Wi-Fi+LTE)        |
| iPad6,7     | iPad Pro (12.9 inch, Wi-Fi)            |
| iPad6,8     | iPad Pro (12.9 inch, Wi-Fi+LTE)        |
| iPad6,11    | iPad 9.7-Inch 5th Gen (Wi-Fi Only)     |
| iPad6,12    | iPad 9.7-Inch 5th Gen (Wi-Fi/Cellular) |
| iPad7,1     | iPad Pro (12.9 inch, A1670)            |
| iPad7,2     | iPad Pro (12.9 inch, A18219)           |
| iPad7,3     | iPad Pro (10.5 inch, A1701)            |
| iPad7,4     | iPad Pro (10.5 inch, A1709)            |
| iPad7,5     | iPad (6th gen, A1893)                  |
| iPad7,6     | iPad (6th gen, A1954)                  |
| iPod1,1     | iPod touch                             |
| iPod2,1     | iPod touch (2nd gen)                   |
| iPod3,1     | iPod touch (3rd gen)                   |
| iPod4,1     | iPod touch (4th gen)                   |
| iPod5,1     | iPod touch (5th gen)                   |
| iPod7,1     | iPod touch (6th gen)                   |

### DRM Callback新規バージョン v2

バージョン1.4まで提供していた「kind1, kind2, kind3」の個別で呼び出しを行われたことを、一回の呼出で統合しました。ユーザー側の呼出が増えることで対応が難しいとの多数の顧客からの意見に対応しました。新しいバージョンを"v2"で、以前バージョンを"v1"で記載します。

#### **Reuest**

| 区分    | Description                     |
| ----- | ------------------------------- |
| POST  | Http POSTでリクエスト (parameterではない) |
| items | JsonArrayで構成されたstring           |

items項目はkind1, kind2, kind3の全てのデータが含まれた呼出にすることができます。

#### **items (JsonArray)**

#### **kind1, kind2**

<table><thead><tr><th width="237">区分</th><th>Description</th></tr></thead><tbody><tr><td>kind</td><td>1,2</td></tr><tr><td>client_user_id</td><td>ユーザーID, media_token 生成に使用されたclient_user_idと同一です。</td></tr><tr><td>player_id</td><td>ユーザーのデバイスID</td></tr><tr><td>hardware_id</td><td>デバイスのhardware ID(PC, 内容がある場合)</td></tr><tr><td>device_name</td><td>ユーザーデバイスのモデル名</td></tr><tr><td>media_content_key</td><td>コンテンツのメディアコンテンツキー</td></tr><tr><td>uservalues</td><td>JSON format (VideoGatewayの呼出に使用されたuservalue0~9)</td></tr><tr><td>localtime</td><td>デバイスの時刻 (UTC)</td></tr></tbody></table>

#### **kind3**

| 区分                  | Description                                         |
| ------------------- | --------------------------------------------------- |
| kind                | 3                                                   |
| session\_key        | content\_expire\_resetリクエストの際に同じsession\_keyを確認します。 |
| client\_user\_id    | ユーザーID, media\_token 生成に使用されたclient\_user\_idと同一です。 |
| player\_id          | ユーザのデバイスID                                          |
| hardware\_id        | デバイスのhardware ID(PC, 内容がある場合)                       |
| device\_name        | ユーザーデバイスのモデル名                                       |
| media\_content\_key | コンテンツのメディアコンテンツキー                                   |
| start\_at           | <p>unixtimestamp (localtime)<br>- 転送リクエスト時刻</p>     |
| uservalues          | JSON format (VideoGatewayの呼出に使用されたuservalue0\~9)    |
| content\_expired    | コンテンツの有効期限を確認 flag ( 1: 再生不可, 0: 再生可能)              |
| check\_expired      | 更新チェックの有効期限を確認 flag ( 1: 再生不可, 0: 再生可能)             |
| reset\_req          | 一括更新なのかを確認 ( 0 (default), 1: 一括更新)                  |
| expiration\_date    | 有効期限終了日(unixtimestamp)                              |
| localtime           | デバイスの時刻 (UTC)                                       |

#### **Items Sample**

```
[
    {
        "kind": 1,
        "media_content_key" : "XXX-MEDIA_CONTENTKEY-XXX",
        "client_user_id": "XXXXXXX",
        "player_id": "xxxxxxxxxxxxxxxx",
        "device_name": "XXXXX",
        "uservalues": {
            "uservalue0": "value0"
        }
    },
    {
        "kind": 2,
        "media_content_key" : "XXX-MEDIA_CONTENTKEY-XXX",
        "client_user_id": "XXXXXXX",
        "player_id": "xxxxxxxxxxxxxxxx",
        "device_name": "XXXXX",
        "uservalues": {
            "uservalue0": "value0"
        }
    },
    {
        "kind": 3,
        "session_key" : "XXX-SESSION_KEY-XXX", ← content_expire_reset リクエストする際に確認します。
        "media_content_key" : "XXX-MEDIA_CONTENTKEY-XXX",
        "client_user_id": "XXXXXXX",
        "player_id": "xxxxxxxxxxxxxxxx",
        "device_name": "XXXXX",
        "uservalues": {
       		"uservalue1": "value1"
        }
    }
]
```

```
```

#### **Response**

Arrayは“data”フィールドで受け取る

#### **kind1**

<table><thead><tr><th width="274">区分</th><th width="74.33333333333331">必須</th><th>Description</th></tr></thead><tbody><tr><td>(int) kind</td><td>O</td><td>1</td></tr><tr><td>(string) media_content_key</td><td>O</td><td>コンテンツのメディアコンテンツキー</td></tr><tr><td>(int) expiration_date</td><td><br></td><td>有効期限のunixtime stamp<br>最大値 : 2029年 12月 31日 23時 59分 59秒 (1893455999)</td></tr><tr><td>(int) expiration_count</td><td><br></td><td>再生回数制限 例) 10 ← 10回まで再生可能</td></tr><tr><td>(int) expiration_playtime</td><td><br></td><td>再生時間制限 例) 3600 ← 3600秒(1時間)まで再生可能</td></tr><tr><td>(int) expiration_playtime_type</td><td><br></td><td>1の場合、play状態を起動時間として制限</td></tr><tr><td>(int) result</td><td>O</td><td>0 (エラー), 1 (正常)<br>0の場合ダウンロードできません。<br>0の場合リトライしません。</td></tr><tr><td>(string) message</td><td><br></td><td>0 (エラー)の場合、message内容を追加するとエラーが起きた際に内容が表示されます。</td></tr><tr><td>(int)<br>expiration_refresh_popup</td><td><br></td><td>有効期限終了後、ポリシー更新表示の有無<br>0の場合表示しない (基本値)<br>1の場合有効期限終了後にユーザの確認を求めるメッセージを表示します。</td></tr><tr><td>(int) vmcheck</td><td><br></td><td>virtual machine 有無確認 (PC) <br>0: 確認しない, 1: 確認する (基本値)</td></tr><tr><td>(int) check_abuse</td><td><br></td><td>DRM kind3 を常に確認 (0: 確認しない(基本値), 1: 確認する)</td></tr><tr><td>(int) offline_bookmark.download</td><td><br></td><td>オフライン状態でブックマークダウンロード使用有無<br>コンテンツをダウンロードした時点のデータのみ反映され、以降サーバーとのシンクはしない<br>0: しない(基本値)<br>1: ブックマークのみダウンロード</td></tr><tr><td>(int) offline_bookmark.readonly</td><td><br></td><td>オフライン状態でブックマーク追加/削除の使用有無 (0: 使用する(基本値), 1: 使用しない)</td></tr></tbody></table>

#### **kind2**

<table><thead><tr><th width="278">区分</th><th width="73.33333333333331">必須</th><th>Description</th></tr></thead><tbody><tr><td>(int) kind</td><td>O</td><td>2</td></tr><tr><td>(string) media_content_key</td><td>O</td><td>コンテンツのメディアコンテンツキー</td></tr><tr><td>(int) content_delete</td><td><br></td><td>0 (削除しない)、1 (ダウンロードしたファイルを削除)<br>ダウンロードしたコンテンツを削除するオプションです。</td></tr><tr><td>(string) message</td><td><br></td><td>0 (エラー)の場合、またはcontent_deleteが1(ダウンロードしたファイルを削除)の場合にmessageを追加すると状況によるメッセージが表示されます。(内容編集可能)</td></tr><tr><td>(int) check_expiration_date</td><td><br></td><td>使用しない(0), 更新チェックの有効期限終了日のunixtime stamp</td></tr><tr><td>(int) result</td><td>O</td><td>0 (エラー)、1 (正常)<br>0の場合、追加作業はありません。<br>0の場合、追加オプションは無視されます。<br>0の場合、再度リクエストされません。</td></tr></tbody></table>

#### **kind3**

<table><thead><tr><th width="282">区分</th><th width="69.33333333333331">必須</th><th>Description</th></tr></thead><tbody><tr><td>(int) kind</td><td>O</td><td>3</td></tr><tr><td>(string) session_key</td><td><br></td><td>content_expire_reset リクエストした際にsession_keyを確認します。</td></tr><tr><td>(string) media_content_key</td><td>O</td><td>コンテンツのメディアコンテンツキー</td></tr><tr><td>(int) start_at</td><td>O</td><td>Requestに含まれたstart_at</td></tr><tr><td>(int) content_expired</td><td><br></td><td>0 (再生可能), 1 (再生不可)<br>ダウンロードしたコンテンツを強制的に有効期限切れ状態にします。<br>content_expire_resetオプションで復旧することができます。<br>* expiredコンテンツは 0(再生可能)でレスポンスしても再生できません。<br>* 1の場合、content_expire_reset オプションを無視します。</td></tr><tr><td>(int) content_delete</td><td><br></td><td>0 (削除しない), 1 (削除する)<br>ダウンロードしたコンテンツを削除するオプション<br>content_expired オプションを無視してcontent_deleteが存在する場合削除する。</td></tr><tr><td>(int) content_expire_reset</td><td><br></td><td>0 (何もしない), 1 (expiredされたコンテンツ権限をリセット)<br>expiredされたコンテンツの権限をリセットするオプション</td></tr><tr><td>(int) expiration_date</td><td><br></td><td>有効期限終了日のunixtime stamp<br>最大値: 2029年 12月 31日 23時 59分 59秒 (1893455999)<br>* content_expire_reset オプションが1の場合、再設定されます。</td></tr><tr><td>(int) expiration_count</td><td><br></td><td>再生回数制限 例) 10 ← 10回再生可能<br>* content_expire_reset オプションが1の場合、再設定されます。</td></tr><tr><td>(int) expiration_playtime</td><td><br></td><td>再生時間制限 例) 3600 ← 3600秒(1時間)まで再生可能<br>* content_expire_reset オプションが1の場合、再設定されます。</td></tr><tr><td>(int) result</td><td>O</td><td>0 (エラー)、1 (正常)<br>0の場合、追加作業はありません。<br>0の場合、追加オプションは無視されます。<br>0の場合、再度リクエストされません。</td></tr><tr><td>(string) message</td><td><br></td><td>0 (비정상), content_expiredが 1(再生不可)・content_deleteが 1(コンテンツ削除)の場合にmessageを追加すると状況によるメッセージが表示されます。</td></tr><tr><td>(int) check_abuse</td><td><br></td><td>DRM kind3 常に確認 (0: 確認しない(基本値), 1: 確認する)</td></tr><tr><td>(int) check_expiration_date</td><td><br></td><td><p>更新チェックしない(0), 更新チェックの有効期限終了日のunixtime stamp</p><p><strong>* content_expire_reset オプションが 1の場合、再設定されます。</strong></p></td></tr></tbody></table>

#### **Response ​Example**

```
{
    “data” : [
        {
            "kind": 1,
            "media_content_key": "XXX-MEDIA_CONTENT_KEY-XXX",
            "expiration_date": 1402444800,
            "expiration_count": 10,
            "expiration_playtime": 60,
            "result": 1
        },
        {
            "kind": 2,
            "media_content_key": "XXX-MEDIA_CONTENT_KEY-XXX",
            "content_delete": 1,
            "result": 1
        },
        {
            "kind": 3,
            "session_key" : "XXX-SESSION_KEY-XXX",
            "media_content_key": "XXX-MEDIA_CONTENT_KEY-XXX",
            "start_at": 140000000,
            "content_expired": 1,
            "content_delete": 1,
            "content_expire_reset": 1,
            "expiration_date": 1402444800,
            "expiration_count": 10,
            "expiration_playtime": 3600,
            "result": 1
        }
    ]
}
```

```
```

### JWT 対応 <a href="#drmcallbackjwt-dui-ying" id="drmcallbackjwt-dui-ying"></a>

DRMポリシーをJWTでレスポンスすることができます。JWT ウェブサイト(<https://jwt.io>) から公認されたライブラリーを使用して生成されたTokenに変更してレスポンスする機能が追加されました。

PlayerにJWT Tokenで返還する場合には指定されたヘッダにユーザーキーを含めて転送しなければなりません。

```
HTTP/1.1 200 OK
Date: Fri, 14 Oct 2016 04:12:46 GMT
Content-Type: text/html; charset=utf-8
Transfer-Encoding: chunked
Connection: keep-alive
X-Kollus-UserKey: 0993d76eb424a72f2005b874ac49405d44a6c
Content-Encoding: gzip
```

​指定した(X-Kollus-UserKey) ヘッダに転送されたユーザーキーの情報が一致する場合のみ再生されます。

#### **JWT ​Secret,​ ​JWT​ ​ユーザーキー**

**ユーザーキー(2)を**headerに追加することと共に、既存のJWT scpecにあるsecret keyには**セキュリティキー(1)**&#x3092;入力しなければなりません。Kollus CMSで以下のように関連情報を確認することができます。

設定 > サービスアカウント設定

<figure><img src="/files/3ejS6EQQX1hZ1wcUJmjP" alt=""><figcaption></figcaption></figure>

#### **JWT payload**

各 callbackの responseを JWT playloadに入れて転送します。

#### **Sample**

[ ](http://jwt.io)<http://jwt.io> から提供している様々な言語で具現されたライブラリーとサンプルを使用してください。JWT スペックと同じですが、レスポンスされるhttpヘッダに **X-Kollus-UserKey**でユーザーキーを一緒に転送しなければなりません。

(サンプル) php

```
define('KOLLUS_KEY', '**ユーザーキー**'); // X-Kollus-UserKey
define('JWT_KEY', '**セキュリティキー**'); // JWT VERIFY SIGNATURE

function encode_jwt($payload, $key, $alg = 'HS256') {
    $header = array('alg' => $alg, 'typ' => 'JWT' );
    $unsignedToken = base64_encode(json_encode($header)).'.'.
    base64_encode(json_encode($payload));
    $signature = hash_hmac('sha256', $unsignedToken, $key, TRUE);
    return $unsignedToken.'.'.base64_encode($signature);
}
$payload = array(
    'data' => array(
        array(
            'kind' => 1,
            'result' =>1,
            'expiration_date' => time()+3600,
            'expiration_playtime' => 30,
            'vmcheck' => 1,
        ),
        array(
            'kind' => 2,
            'result' =>1,
            'expiration_date' => time()+3600,
            'expiration_playtime' => 30,
            'vmcheck' => 1,
        ),
        array(
            'kind' => 3,
            'result' =>1,
            'expiration_date' => time()+3600,
            'expiration_playtime' => 30,
            'vmcheck' => 1,
        ),
    ),
);

$result = encode_jwt($payload, JWT_KEY);

//header('Content-Type: application/jwt');
header('X-Kollus-UserKey: '.KOLLUS_KEY);
echo $result;
```

#### Code sample (v1.4 基準) <a href="#drmcallbackcodesamplev1.4-ji-zhun" id="drmcallbackcodesamplev1.4-ji-zhun"></a>

以下のように顧客のDBが構成されたことを想定したサンプルコードになります。

<figure><img src="/files/AzrXUCV3fCTuTbQMjlcP" alt=""><figcaption></figcaption></figure>

#### **PHP ​sample**

```
<?php
/**
* PHP Version : 5.4 above
*/
function get_jwt($payload, $key, $alg = 'HS256')
{
	$header = array('alg' => $alg, 'typ' => 'JWT' );
	$unsignedToken = base64_encode(json_encode($header)).'.'.base64_encode(json_encode($payload));
	$signature = hash_hmac('sha256', $unsignedToken, $key, TRUE);
	return $unsignedToken.'.'.base64_encode($signature);
}

function print_kollus_jwt($data)
{
    define('KOLLUS_KEY', '**ユーザーキー**'); // X-Kollus-UserKey
    define('JWT_KEY', '**セキュリティキー**'); // JWT VERIFY SIGNATURE
    $payload = array( ‘data’ => $data );
    $result = encode_jwt($payload, JWT_KEY);
    //header('Content-Type: application/jwt');
    header('X-Kollus-UserKey: '.KOLLUS_KEY);
    echo $result;
}

/////////////////////////////////////////////////

include "./config.php";
// 注意 : kindが1の場合、自動でexpiration_dateが生成されるサンプルページとなります。
// 再生回数制限値
$_default_expiration_count = 3;
$_expired_duration = 60 * 60 * 24; // 1 day

// DB Connection
$_db_conn = mysqli_connect($_hostname, $_username, $_password);
if (!$_db_conn) {
	die('Could not connect: ' . mysql_error());
}

$_db_selected = mysqli_select_db($_database, $_db_conn);
if (!$_db_selected) {
	die ('Can\'t use database : ' . mysqli_error());
}

$_kind = isset($_POST['kind']) ? ((int) $_POST['kind']) : NULL;
$_media_content_key = isset($_POST['media_content_key']) ? $_POST['media_content_key'] : NULL;
$_client_user_id = isset($_POST['client_user_id']) ? $_POST['client_user_id'] : NULL;

$channel = NULL;
$_query = sprintf("SELECT * FROM `channels` WHERE `media_content_key` = '%s'",
mysqli_real_escape_string($_media_content_key, $_db_conn));
$_result = mysqli_query($_query, $_db_conn);

if ($_result) {
	$channel = mysqli_fetch_array($_result, MYSQL_ASSOC);
	if ($channel === FALSE) $channel = NULL;
} else {
	die('Invalid query: ' . mysqli_error());
}

$user = NULL;
$_query = sprintf("SELECT * FROM `users` WHERE `client_user_id` = '%s'",
mysqli_real_escape_string($_client_user_id, $_db_conn));
$_result = mysqli_query($_query, $_db_conn);
if ($_result) {
	$user = mysqli_fetch_array($_result, MYSQL_ASSOC);
	if ($user === FALSE) $user = NULL;
} else {
	die('Invalid query: ' . mysqli_error());
}

$channel_user = NULL;
if (!is_null($_media_content_key) && !is_null($_client_user_id)) {
	$_query = sprintf("SELECT * FROM `channel_users` WHERE `user_id` = '%s', `channel_id` = '%s'", $user['id'], $channel['id']);
	$_result = mysqli_query($_query, $_db_conn);
	if ($_result) {
		$channel_user = mysqli_fetch_array($_result, MYSQL_ASSOC);
		if ($channel_user === FALSE) $channel_user = NULL;
	} else {
		die('Invalid query: ' . mysqli_error());
	}
}

$_jwt_result = array('result' => 0);
switch($_kind) {
case 1:
	if (is_null($channel_user)){
		$_expiration_date = time() + $_expired_duration;
        $_expiration_count = $_default_expiration_count;
        $_query = sprintf("INSERT INTO `channel_users`(`user_id`, `channel_id`, `expiration_date`
        ,`expiration_count` , `created_at`, `updated_at`) VALUES('%s', '%s', '%s', '%s', UNIX_TIMESTAMP(),
        UNIX_TIMESTAMP())", $user['id'], $channel['id'], $_expiration_date, $_expiration_count);
        $_result = mysqli_query($_query, $_db_conn);
		if (!$_result) {
			die('Invalid query: ' . mysqli_error());
		}
	} else {
        $_expiration_date = $channel_user['expiration_date'];
        $_expiration_count = $channel_user['$expiration_count'];
	}
    $_jwt_result['expiration_date'] = (int) $_expiration_date;
    $_jwt_result['expiration_count'] = (int) $_expiration_count;
	break;
case 2:
    if (!is_null($channel_user)){
        $_download_times = ++((int) $channel_user['download_times']);
        $_query = sprintf("UPDATE `channel_users` SET `download_times` = '%s', `updated_at` =
        UNIX_TIMESTAMP() WHERE id = %s", $_download_times, $channel_user['id']);
        $_result = mysqli_query($_query, $_db_conn);
  	    if (!$_result) {
    		die('Invalid query: ' . mysql_error());
   		}
        $_jwt_result['result'] = 1;
    }
	break;
case 3:
    if (!is_null($channel_user) && $channel_user['is_expired']) {
    	$_jwt_result['content_expired'] = 1;
    }
    break;
}

// DB Close
mysqli_close($_db_conn);
// 結果の data arrayをJWTに変換して出力
print_kollus_jwt($_jwt_result);
```

### **サービスサポート**

> サンプルコードについてのお問い合わせは担当までご連絡ください。
>
> E-mail お問い合わせ > <jp_team@catenoid.net>
>
> 電話番号 > 03-4405-8462


---

# Agent Instructions
This documentation is published with GitBook. GitBook is the documentation platform designed so that both humans and AI agents can read, navigate, and reason over technical content effectively. Learn more at gitbook.com.

## Querying This Documentation
If you need additional information that is not directly available in this page, you can query the documentation dynamically by asking a question.

Perform an HTTP GET request on the current page URL with the `ask` query parameter:

```
GET https://catenoid-support.gitbook.io/kollus-dev-jp/callback-gai-yao/drm-callback.md?ask=<question>
```

The question should be specific, self-contained, and written in natural language.
The response will contain a direct answer to the question and relevant excerpts and sources from the documentation.

Use this mechanism when the answer is not explicitly present in the current page, you need clarification or additional context, or you want to retrieve related documentation sections.
