fetch() の概要

fetch() を使用すると、XMLHttpRequest(XHR)に似たネットワーク リクエストを行うことができます。主な違いは、Fetch API では Promise が使用されることです。この API は、XMLHttpRequest API の複雑なコールバックを回避するためのシンプルな API を備えています。

対応ブラウザ

  • Chrome: 42。
  • Edge: 14.
  • Firefox: 39.
  • Safari: 10.1。

ソース

Promise を使用したことがない場合は、JavaScript Promise の概要をご覧ください。

XMLHttpRequestfetch で実装された例を次に示します。URL をリクエストしてレスポンスを取得し、JSON として解析します。

XMLHttpRequest には、成功とエラーのケースを処理する 2 つのリスナーと、open()send() の呼び出しが必要です。MDN ドキュメントからの例

function reqListener () {
  const data = JSON.parse(this.responseText);
  console.log(data);
}

function reqError (err) {
  console.log('Fetch Error :-S', err);
}

const oReq = new XMLHttpRequest();
oReq.onload = reqListener;
oReq.onerror = reqError;
oReq.open('get', './api/some.json', true);
oReq.send();

フェッチ

取得リクエストは次のようになります。

fetch('./api/some.json')
  .then(response => {
    if (response.status !== 200) {
      console.log(`Looks like there was a problem. Status Code: ${response.status}`);

      return;
    }

    // Examine the text in the response
    response.json().then(function(data) {
      console.log(data);
    });
  })
  .catch(err => {
    console.log('Fetch Error :-S', err);
  });

XHR の例と同じ処理を実行するために、fetch() リクエストで必要な呼び出しは 1 回だけです。レスポンスを処理するには、まずレスポンスのステータスが 200 であることを確認し、次にレスポンスを JSON として解析します。fetch() リクエストに対するレスポンスは Stream オブジェクトです。つまり、json() メソッドを呼び出した後に Promise が返されます。ストリームは非同期で発生します。

レスポンス メタデータ

上記の例では、Response オブジェクトのステータスと、レスポンスを JSON として解析する方法を示しました。ヘッダーなど、アクセスする必要がある他のメタデータを処理する方法は次のとおりです。

fetch('users.json').then(response => {
  console.log(response.headers.get('Content-Type'));
  console.log(response.headers.get('Date'));

  console.log(response.status);
  console.log(response.statusText);
  console.log(response.type);
  console.log(response.url);
});

レスポンス タイプ

Google がフェッチ リクエストを行うと、レスポンスには「basic」、「cors」、または「opaque」の response.type が返されます。これらの types は、リソースの取得元を示しており、レスポンス オブジェクトの処理方法を決定するために使用できます。

ブラウザが同じオリジンのリソースをリクエストする場合、レスポンスは basic 型となり、レスポンスから表示できる内容が制限されます。

別のオリジンのリソースに対してリクエストが送信され、そのオリジンが CORs ヘッダーを返す場合、タイプは cors です。cors レスポンスは basic レスポンスに似ていますが、表示できるヘッダーが Cache-ControlContent-LanguageContent-TypeExpiresLast-ModifiedPragma に制限されます。

opaque レスポンスが、CORS ヘッダーを返さない別の生成元から送信されている。不透明なレスポンスでは、返されたデータを読み取ったり、リクエストのステータスを確認したりできないため、リクエストが成功したかどうかを確認できません。

フェッチ リクエストのモードを定義して、特定のリクエスト タイプのみが解決されるようにできます。設定できるモードは次のとおりです。

  • same-origin は、同じオリジンのアセットのリクエストでのみ成功し、他のすべてのリクエストを拒否します。
  • cors は、同じオリジンのアセットと、適切な COR ヘッダーを返す他のオリジンのアセットのリクエストを許可します。
  • cors-with-forced-preflight は、リクエストを行う前にプリフライト チェックを実行します。
  • no-cors は、CORS ヘッダーがない他のオリジンにリクエストを行い、opaque レスポンスを返すことを目的としていますが、前述のように、現時点ではウィンドウ グローバル スコープではできません。

モードを定義するには、fetch リクエストに 2 番目のパラメータとしてオプション オブジェクトを追加し、そのオブジェクト内でモードを定義します。

fetch('http://some-site.com/cors-enabled/some.json', {mode: 'cors'})
  .then(response => response.text())
  .then(text => {
    console.log('Request successful', text);
  })
  .catch(error => {
    log('Request failed', error)
  });

Promise の連鎖

Promise の優れた機能の一つは、それらを連結できることです。fetch() の場合、フェッチ リクエスト間でロジックを共有できます。

JSON API を使用している場合は、ステータスを確認し、各レスポンスの JSON を解析する必要があります。ステータスと JSON 解析を Promise を返す個別の関数で定義し、フェッチ リクエストを使用して最終データとエラーケースのみを処理することで、コードを簡素化できます。

function status (response) {
  if (response.status >= 200 && response.status < 300) {
    return Promise.resolve(response)
  } else {
    return Promise.reject(new Error(response.statusText))
  }
}

function json (response) {
  return response.json()
}

fetch('users.json')
  .then(status)
  .then(json)
  .then(data => {
    console.log('Request succeeded with JSON response', data);
  }).catch(error => {
    console.log('Request failed', error);
  });

この例では、response.status をチェックし、解決された Promise を Promise.resolve() として返す、または拒否された Promise を Promise.reject() として返す status 関数を定義します。これは fetch() チェーンで呼び出される最初のメソッドです。

Promise が解決されると、スクリプトは json() メソッドを呼び出します。このメソッドは response.json() 呼び出しから 2 番目の Promise を返し、解析された JSON を含むオブジェクトを作成します。解析に失敗した場合、Promise は拒否され、catch ステートメントが実行されます。

この構造により、すべてのフェッチ リクエストでロジックを共有できるため、コードの保守、読み取り、テストが容易になります。

POST リクエスト

ウェブアプリで POST メソッドを使用して API を呼び出し、リクエストの本文にパラメータを追加する必要がある場合があります。これを行うには、fetch() オプションで method パラメータと body パラメータを設定します。

fetch(url, {
    method: 'post',
    headers: {
      "Content-type": "application/x-www-form-urlencoded; charset=UTF-8"
    },
    body: 'foo=bar&lorem=ipsum'
  })
  .then(json)
  .then(data => {
    console.log('Request succeeded with JSON response', data);
  })
  .catch(error => {
    console.log('Request failed', error);
  });

取得リクエストで認証情報を送信する

Cookie などの認証情報を使用して取得リクエストを行うには、リクエストの credentials 値を "include" に設定します。

fetch(url, {
  credentials: 'include'
})