Salesforce で、送信されたお問い合わせが簡単に消えてしまう話


Salesforce の Web-to-ケースで reCAPTCHA を有効にした場合1、ユーザの時計が1分ほど遅れているだけで、ケースが消えてなくなります。特にエラーなどは表示されず、ただ消失してしまうようです。Web-to-リード も同様です。

以下の手順で簡単に再現します。

再現手順

1. Salesforce にログインする

アカウントがない場合は Developer Edition でアカウントを作成します。

2. Web-to-ケースのセットアップを行う

ドキュメントのとおりに Web-to-ケースをセットアップします。

reCAPTCHA キー (v2 checkbox または invisible) を Salesforce に登録し、HTML を生成します。送信完了ページには適当な URL を設定すれば OK です。

reCAPTCHA 設定には、localhost など、テストに使うサーバのドメインを追加しておいてください。

3. 正常にケースを生成できることを確認する

localhost など reCAPTCHA キーに登録したドメインで生成した HTML を開き、お問い合わせを送信します。

送信完了画面にリダイレクトし、Salesforce にケースが作成されるのを確認します。

4. 時計を遅らせて、ケースが消失するのを確認する

次に、手元の時計を1分以上遅らせて、同じように送信し、送信完了画面にリダイレクトするのに、Salesforce にはケースが作成されていないことを確認します。

生成された HTML で、captcha_settingsts 値に時刻を書き込んでいる以下の箇所を書き換えることでも再現できます。

// 修正前
elems["ts"] = JSON.stringify(new Date().getTime());

// 修正後。ts 値を 90 秒遅らせる
elems["ts"] = JSON.stringify(new Date().getTime() - (1000 * 90));

原因について

Salesforce は、captcha_settings という値で、reCAPTCHA の設定を送信しています。

<!-- Salesforce が生成する HTML より引用、JSON の値を修正 -->
<input type=hidden name='captcha_settings' value='{"keyname":"reCAPTCHA キー名","fallback":"フォールバックするかどうか","orgId":"組織ID","ts":"タイムスタンプ"}'>

JavaScript が 0.5 秒ごとにこの JSON をパースし、ts 値をローカルのエポックタイムに設定、再び JSON としてエンコードして、captcha_settings に書き戻す、という処理を行っています。(この時点で、既にだいぶ微妙な感じはありますが)

// Salesforce が生成する HTML より引用
 function timestamp() { var response = document.getElementById("g-recaptcha-response"); if (response == null || response.value.trim() == "") {var elems = JSON.parse(document.getElementsByName("captcha_settings")[0].value);elems["ts"] = JSON.stringify(new Date().getTime());document.getElementsByName("captcha_settings")[0].value = JSON.stringify(elems); } } setInterval(timestamp, 500);

そして、Salesforce サーバは、ts 値が遅れていると、黙ってケース・リードを破棄する、という不思議な実装になっているようです。

サポートともやり取りをさせていただき、詳細を書くのは避けますが、これが設定の不備などではないのは間違いないようです。

タイムスタンプは reCAPTCHA 仕様で必要?

reCAPTCHA の Verification API を見なおしましたが、このような制約は必要無いと思います。

API で必須のパラメータは、シークレットキーとクライアントから送られてきたトークンの2つだけです。

API レスポンスには challenge_ts というトークン取得時のタイムスタンプが含まれています。仮にもしこの値を独自に検証するのだとしたら (それも必要ないと思いますが) サーバ時刻を使うのが適切でしょう。クライアント側で自由に書き換えられる値を使う意味がわかりません、、。

Google が Salesforce に向けて、独自に非公開の Validation API を提供している可能性はありえなくはないと思いますが、普通に使うよりも品質が下がっていますし、考えにくいのではと思います。

Salesforce の reCAPTCHA 実装の品質について

数年前までは、captcha_settings 値を送信しないことで、reCAPTCHA の検証を回避できる問題がありました。

なんとかしてください

このような制約は不要なはずです。もし本当に必要なら、詳細をドキュメントで説明してください。

Ideas というところにも書きました。(よかったら vote してください)

代替

単に問い合わせを受けられるだけでよいなら、Kwes Forms などのフォームサービスもあります。


  1. スパムケースを防止するための reCAPTCHA の有効化 に「Winter '19 より後に作成された組織では、この設定はデフォルトで有効になっています。無効にできますが、それはお勧めしません」とあり、デフォルトで有効になっているものと思われます。が、アカウントを新規に作成して確認はしていません。