現在のブラウザが JavaScript でヘッドレス ブラウザであるかどうかを検出する方法

現在のブラウザが JavaScript でヘッドレス ブラウザであるかどうかを検出する方法

ヘッドレスブラウザとは何ですか?

ヘッドレス ブラウザは、グラフィカル インターフェイスで実行できるブラウザです。ヘッドレス ブラウザをプログラムで制御して、テストの実行、Web ページのスクリーンショットの取得など、さまざまなタスクを自動的に実行できます。

なぜ「ヘッドレス」ブラウザと呼ばれるのでしょうか?

「ヘッドレス」という言葉は、元々は「ヘッドレス コンピュータ」から来ています。 「ヘッドレス コンピュータ」に関する Wikipedia のエントリ:

ヘッドレス システムとは、モニター (つまり「ヘッド」)、キーボード、マウスなしで動作するように構成されたコンピュータ システムまたはデバイスです。ヘッドレス システムは通常、ネットワーク接続を介して制御されますが、一部のヘッドレス システム デバイスでは、デバイス管理のために RS-232 シリアル接続も必要です。運用コストを削減するために、サーバーはヘッドレス モードで実行されることがよくあります。

ヘッドレスブラウザを検出するのはなぜですか?

前述の 2 つの無害な使用例の他に、ヘッドレス ブラウザを使用して悪意のあるタスクを自動化することもできます。最も一般的な形式は、Web クローラー、トラフィックの偽装、または Web サイトの脆弱性の検出です。

非常に人気のあるヘッドレス ブラウザは Phantomjs です。Qt フレームワークをベースとしているため、一般的なブラウザと比較してさまざまな機能があり、識別方法も多数あります。

しかし、Chrome 59 以降、Google はヘッドレス Google Chrome ブラウザをリリースしました。 Phantomjsとは異なり、他のフレームワークではなくオーソドックスなGoogle Chromeをベースに開発されているため、プログラムが通常のブラウザなのかヘッドレスブラウザなのかを区別することが困難です。

以下では、プログラムが通常のブラウザで実行されているか、ヘッドレスブラウザで実行されているかを判断するためのいくつかの方法を紹介します。

ヘッドレスブラウザの検出

注: これらの方法は 4 台のマシン (Linux 2 台、Mac 2 台) でのみテストされていますが、ヘッドレス ブラウザーを検出する方法は他にもたくさんあります。

ユーザーエージェント

まず、ブラウザの種類を判断する最も一般的な方法である、ユーザーエージェントを確認する方法を紹介します。 Linux コンピュータ上の Chrome バージョン 59 ヘッドレス ブラウザのユーザー エージェント値は次のとおりです。

「Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (Khtml、Gecko に類似) HeadlessChrome/59.0.3071.115 Safari/537.36」

したがって、次のようにしてヘッドレス Chrome ブラウザかどうかを検出できます。

if (/HeadlessChrome/.test(window.navigator.userAgent)) {
  console.log("Chrome ヘッドレスが検出されました");
 }

ユーザーエージェントは HTTP ヘッダーから取得することもできます。ただし、これらのシナリオは両方とも簡単に偽造できます。

プラグイン

navigator.plugins は、現在のブラウザのプラグイン情報を含む配列を返します。通常、通常の Chrome ブラウザには、Chrome PDF ビューアや Google Native Client などのデフォルトのプラグインがいくつか用意されています。対照的に、プラグインのないヘッドレス モードでは、空の配列が返されます。

(navigator.plugins.length == 0)の場合{
  console.log("Chrome ヘッドレスである可能性があります");
}

言語

Google Chrome には、現在のブラウザの言語設定を取得できる JavaScript プロパティが 2 つあります。navigator.language と navigator.languages です。最初のものはブラウザインターフェースの言語を参照し、2 番目のものはブラウザユーザーが選択したすべての二次言語を格納する配列を返します。ただし、ヘッドレス モードでは、navigator.languages は空の文字列を返します。

if (navigator.languages ​​== "") {
  console.log("Chrome ヘッドレスが検出されました");
}

ウェブGL

WebGL は、HTML キャンバスで 3D レンダリングを実行できる一連の API を提供します。これらの API を通じて、グラフィックス ドライバーのベンダーとレンダラーを照会できます。

Linux 上の通常の Google Chrome ブラウザでは、レンダラーとベンダーの値は「Google SwiftShader」と「Google Inc.」として取得されます。

ヘッドレス モードでは、「Mesa OffScreen」 (ウィンドウ システムを使用しないレンダリング テクノロジの名前) と、オープン ソースの Mesa グラフィック ライブラリのオリジナル プログラマーである「Brian Paul」が表示されます。

 var キャンバス = document.createElement('キャンバス');
 var gl = canvas.getContext('webgl');
  
 var debugInfo = gl.getExtension('WEBGL_debug_renderer_info');
 var ベンダー = gl.getParameter(debugInfo.UNMASKED_VENDOR_WEBGL);
 var レンダラー = gl.getParameter(debugInfo.UNMASKED_RENDERER_WEBGL);
  
 if (ベンダー == "Brian Paul" && レンダラー == "Mesa OffScreen") {
  console.log("Chrome ヘッドレスが検出されました");
 }

ヘッドレス ブラウザのすべてのバージョンで、これら 2 つの値が同じであるわけではありません。ただし、現在ヘッドレスブラウザでは値は「Mesa Offscreen」と「Brian Paul」です

ブラウザの機能

Modernizr は、HTML および CSS のさまざまな機能に対する現在のブラウザのサポートを検出できます。通常のChromeとヘッドレスChromeの唯一の違いは、ヘッドレスモードではHIDPI/Retinaのヘアラインがサポートされているかどうかを検出するために使用されるヘアライン機能がないことです。

if (!Modernizr["ヘアライン"]) {
  console.log("Chrome ヘッドレスである可能性があります");
}

画像の読み込みに失敗しました

最後に、私が見つけた最後の方法であり、最も効果的と思われる方法は、ブラウザで正しく読み込まれていない画像の高さと幅を確認することです。

通常の Chrome では、正常に読み込めなかった画像のサイズはブラウザのズームに関係しますが、絶対にゼロにはなりません。ヘッドレス Chrome ブラウザでは、この画像の幅と高さはどちらも 0 です。

var body = document.getElementsByTagName("body")[0];
var image = document.createElement("img");
画像を拡大
image.setAttribute("id", "偽画像");
body.appendChild(画像);
image.onerror = 関数(){
	画像の幅が 0 の場合、画像の高さは 0 になります。
		console.log("Chrome ヘッドレスが検出されました");
	}
}

上記は、JavaScript を使用して現在のブラウザがヘッドレス ブラウザであるかどうかを検出する方法の詳細です。JavaScript の詳細については、123WORDPRESS.COM の他の関連記事に注目してください。

以下もご興味があるかもしれません:
  • NodeJSとブラウザにおけるこのキーワードの違い
  • JavaScriptはブラウザがIEかどうかを判定します
  • ブラウザのウェブページの自動スクロールとクリックを実現する JavaScript のサンプル コード
  • JavaScript を使用してブラウザでウェブカメラを使用する方法
  • JavaScript を使用してブラウザ履歴 API を操作する方法
  • JSクロスブラウザXMLアプリケーションプロセスの詳細な説明
  • モバイルブラウザがWeChat共有を呼び出す(JS)
  • ブラウザのJavaScriptデバッグ機能は使用できません。解決策
  • よく使われるJavaScriptツールの機能まとめ(ブラウザ環境)
  • js に基づいてブラウザの種類を判断する例

<<:  シェルを使用してMySQLデータバックアップスクリプトを作成する

>>:  Linux で固定 IP を設定する方法 (テスト済みで効果的)

推薦する

Linux で SpringBoot jar プログラム デプロイメント シェル スクリプトを起動および停止する方法

では早速、コードをお見せしましょう。具体的なコードは次のとおりです。 #!/bin/bash cd ...

CSS3で実装されたダイナミックな星空の背景

結果:実装コードhtml <link href='https://fonts.goog...

初心者がHTMLタグを学ぶ(1)

初心者は、いくつかの HTML タグを理解することで HTML を学習できます。この入門書は、初心者...

CSS フロート(float, clear)の人気の解説と体験談

私はかなり昔に CSS に触れましたが、フローティングについてはいつも混乱していました。私の理解が浅...

Vue 名前付きスロットの基本的な使用例

序文名前付きスロットは、スロット内の「name」属性を使用して要素にバインドされます。知らせ: 1....

docker に nacos をインストールしてデータベースを構成する詳細なチュートリアル

環境の準備 Docker環境 MySQL 5.7 (公式イメージはmysql8をサポートしていません...

ダッシュボードを実装するためのjQueryプラグイン

jQueryプラグインは、参考のためにダッシュボードを実装します。具体的な内容は次のとおりです。一般...

MySQL での数値のフォーマットの詳細な説明

最近、仕事の都合で、MySQL で数字をフォーマットする必要がありましたが、インターネット上にはほと...

Kubernetesでポッドを作成する方法

目次ポッドを作成するには? kubectl ツールポッドを作成するには?前回の記事では、コンテナとポ...

HTMLページの文字セットを指定する2つの方法

1. HTMLページの文字セットを指定する2つの方法方法1: <メタ文字セット="u...

MySQL公式パフォーマンステストツールmysqlslapの使い方の紹介

目次導入説明書実際の経験まとめ導入MySQL は最も人気のあるオープンソース データベースとして、さ...

JavaScriptプロトタイプと例の詳細な説明

目次コンストラクタインスタンスとプロトタイプの関係プロトタイププロパティ属性またはメンバーの検索原則...

mysql データベースの作成、ユーザーの追加、ユーザー認証の実用的な方法

1. MySQLデータベースを作成する1. データベース構文を作成する --「testdb」という名...

React Stateの状態とライフサイクルの実装方法

1. コンポーネントの実装方法:組件名稱首字母必須大寫1. JS関数を通じてコン​​ポーネントを実装...

Linux で at および cron スケジュールタスクをカスタマイズする方法

Linux システムには 2 種類のスケジュールされたタスクがあります。1 つは 1 回だけ実行され...