質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

88.58%

PHP headerとCookie

解決済

回答 2

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 1,210

snowman

score 25

ツイッターAPIを使ってユーザーの情報を持ってきてそれをcookieに保存したいのですがエラーになってしまいます。
調べてみるとcookieはheaderの前に書かなきゃいけないとか・・・。
しかし、headerを使ってからじゃないとcookieに保存したいユーザーの情報が手元に来ません。
どうしたら解決できますか?

$api_key = "" ;    // API Key
$api_secret = "" ;
$access_token = "" ;    // アクセストークン
$access_token_secret = "" ;    // アクセストークンシークレット

//情報の取得
$request_url = 'https://api.twitter.com/1.1/account/settings.json' ;        // エンドポイント
$request_method = 'GET' ;

// パラメータA (オプション)
$params_a = array(
) ;

// キーを作成する (URLエンコードする)
$signature_key = rawurlencode( $api_secret ) . '&' . rawurlencode( $access_token_secret ) ;

// パラメータB (署名の材料用)
$params_b = array(
    'oauth_token' => $access_token ,
    'oauth_consumer_key' => $api_key ,
    'oauth_signature_method' => 'HMAC-SHA1' ,
    'oauth_timestamp' => time() ,
    'oauth_nonce' => microtime() ,
    'oauth_version' => '1.0' ,
    ) ;

// パラメータAとパラメータBを合成してパラメータCを作る
$params_c = array_merge( $params_a , $params_b ) ;

// 連想配列をアルファベット順に並び替える
ksort( $params_c ) ;

// パラメータの連想配列を[キー=値&キー=値...]の文字列に変換する
$request_params = http_build_query( $params_c , '' , '&' ) ;

// 一部の文字列をフォロー
$request_params = str_replace( array( '+' , '%7E' ) , array( '%20' , '~' ) , $request_params ) ;

// 変換した文字列をURLエンコードする
$request_params = rawurlencode( $request_params ) ;

// リクエストメソッドをURLエンコードする
// ここでは、URL末尾の[?]以下は付けないこと
$encoded_request_method = rawurlencode( $request_method ) ;

// リクエストURLをURLエンコードする
$encoded_request_url = rawurlencode( $request_url ) ;

// リクエストメソッド、リクエストURL、パラメータを[&]で繋ぐ
$signature_data = $encoded_request_method . '&' . $encoded_request_url . '&' . $request_params ;

// キー[$signature_key]とデータ[$signature_data]を利用して、HMAC-SHA1方式のハッシュ値に変換する
$hash = hash_hmac( 'sha1' , $signature_data , $signature_key , TRUE ) ;

// base64エンコードして、署名[$signature]が完成する
$signature = base64_encode( $hash ) ;

// パラメータの連想配列、[$params]に、作成した署名を加える
$params_c['oauth_signature'] = $signature ;

// パラメータの連想配列を[キー=値,キー=値,...]の文字列に変換する
$header_params = http_build_query( $params_c , '' , ',' ) ;

// リクエスト用のコンテキスト
$context = array(
    'http' => array(
        'method' => $request_method , // リクエストメソッド
        'header' => array(              // ヘッダー
            'Authorization: OAuth ' . $header_params ,
        ) ,
    ) ,
) ;

// パラメータがある場合、URLの末尾に追加
if( $params_a ) {
    $request_url .= '?' . http_build_query( $params_a ) ;
}

// オプションがある場合、コンテキストにPOSTフィールドを作成する (GETの場合は不要)
//    if( $params_a ) {
//        $context['http']['content'] = http_build_query( $params_a ) ;
//    }

// cURLを使ってリクエスト
$curl = curl_init() ;
curl_setopt( $curl, CURLOPT_URL , $request_url ) ;
curl_setopt( $curl, CURLOPT_HEADER, 1 ) ; 
curl_setopt( $curl, CURLOPT_CUSTOMREQUEST , $context['http']['method'] ) ;    // メソッド
curl_setopt( $curl, CURLOPT_SSL_VERIFYPEER , false ) ;    // 証明書の検証を行わない
curl_setopt( $curl, CURLOPT_RETURNTRANSFER , true ) ;    // curl_execの結果を文字列で返す
curl_setopt( $curl, CURLOPT_HTTPHEADER , $context['http']['header'] ) ;    // ヘッダー
//    if( isset( $context['http']['content'] ) && !empty( $context['http']['content'] ) ) {        // GETの場合は不要
//        curl_setopt( $curl , CURLOPT_POSTFIELDS , $context['http']['content'] ) ;    // リクエストボディ
//    }
curl_setopt( $curl , CURLOPT_TIMEOUT , 5 ) ;    // タイムアウトの秒数
$res1 = curl_exec( $curl ) ;
$res2 = curl_getinfo( $curl ) ;
curl_close( $curl ) ;

// 取得したデータ
$json = substr( $res1, $res2['header_size'] ) ;        // 取得したデータ(JSONなど)
$header = substr( $res1, 0, $res2['header_size'] ) ;    // レスポンスヘッダー (検証に利用したい場合にどうぞ)

// [cURL]ではなく、[file_get_contents()]を使うには下記の通りです…
// $json = file_get_contents( $request_url , false , stream_context_create( $context ) ) ;

// JSONをオブジェクトに変換
$obj = json_decode( $json ) ;

$arr = json_decode($json,true);

echo $arr[screen_name] ;

//cookie
setcookie("name",$arr[screen_name],time()+60*5);

echo $_COOKIE['name']."さんがログインしました。";

// HTML用
$html = '' ;

// タイトル
$html .= '<h1 style="text-align:center; border-bottom:1px solid #555; padding-bottom:12px; margin-bottom:48px; color:#D36015;">GET account/settings</h1>' ;

// エラー判定
if( !$json || !$obj ) {
    $html .= '<h2>エラー内容</h2>' ;
    $html .= '<p>データを取得することができませんでした…。設定を見直して下さい。</p>' ;
}

// 検証用
$html .= '<h2>取得したデータ</h2>' ;
$html .= '<p>下記のデータを取得できました。</p>' ;
$html .=     '<h3>ボディ(JSON)</h3>' ;
$html .=     '<p><textarea style="width:80%" rows="8">' . $json . '</textarea></p>' ;
$html .=     '<h3>レスポンスヘッダー</h3>' ;
$html .=     '<p><textarea style="width:80%" rows="8">' . $header . '</textarea></p>' ;

// 検証用
$html .= '<h2>リクエストしたデータ</h2>' ;
$html .= '<p>下記内容でリクエストをしました。</p>' ;
$html .=     '<h3>URL</h3>' ;
$html .=     '<p><textarea style="width:80%" rows="8">' . $context['http']['method'] . ' ' . $request_url . '</textarea></p>' ;
$html .=     '<h3>ヘッダー</h3>' ;
$html .=     '<p><textarea style="width:80%" rows="8">' . implode( "\r\n" , $context['http']['header'] ) . '</textarea></p>' ;

// フッター
$html .= '<small style="display:block; border-top:1px solid #555; padding-top:12px; margin-top:72px; text-align:center; font-weight:700;">プログラムの説明: <a href="https://syncer.jp/Web/API/Twitter/REST_API/GET/account/settings/" target="_blank">SYNCER</a></small>' ;

// 出力 (本稼働時はHTMLのヘッダー、フッターを付けよう)
echo $html ;
  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

質問への追記・修正、ベストアンサー選択の依頼

  • CHERRY

    2018/11/30 13:23

    お使いのプログラムコードを質問に追記してください。 APIKEY やパスワード等は、APY-KEY 等の適当な文字列に置き換えてください。

    キャンセル

  • snowman

    2018/11/30 13:39

    すみません、今追加しました

    キャンセル

回答 2

checkベストアンサー

+2

echo $arr[screen_name] ;

//cookie
setcookie("name",$arr[screen_name],time()+60*5);

echo $_COOKIE['name']."さんがログインしました。";

この部分の最初の echo が原因ですね。

header や setcookie の前に、画面出力のコマンドを書くとその時点で、header が確定されて変更できなくなりますので、エラーになります。

setcookie を echo の前に持ってくると エラーにならないと思います。

//cookie
setcookie("name",$arr[screen_name],time()+60*5);

echo $arr[screen_name] ;

echo $_COOKIE['name']."さんがログインしました。";

参考:

header より

覚えておいて頂きたいのは、header() 関数は、 通常の HTML タグまたは PHP からの出力にかかわらず、すべての実際の 出力の前にコールする必要があることです。 頻出するエラーとして、include または require 関数、他のファイルをアクセスする関数に 空白または空行があり、header() の前に出力が 行われてしまうというものがあります。

setcookieより

setcookie() は、その他のヘッダ情報と共に 送信するクッキーを定義します。 ほかのヘッダ情報と同様に、 クッキーは、スクリプトによる他のあらゆる出力よりも前に 送信される必要があります(これはHTTPプロトコルの制約です)。 <html> や <head> タグはもちろん 空白も含め、あらゆる出力よりも前にこの関数をコールするようにしなければなりません。

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2018/12/01 01:11

    できました!ありがとうございます!

    キャンセル

+2

headerを使ってからじゃないとcookieに保存したいユーザーの情報が手元に来ません。

論理的におかしいです。header出力用の関数なので、「headerを使わないとCookieに必要な情報が揃わない」ということがありえません。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2018/11/30 13:40

    知識不足ですみません・・・

    キャンセル

  • 2018/11/30 13:50 編集

    上で設定しているheaderは、Twitterと通信するためのものなので、PHP自体が返すheader()とは無関係です。

    書かれている範囲内のコードはこのままで問題ありません。←echoを見落としていました(追記

    キャンセル

  • 2018/11/30 13:52

    1つの可能性として、開く方の<?phpの手前に、BOMあるいはスペースなど、なにか出力されるものが入ってしまっていることが考えられます。

    一度出力を始めてしまえば、Cookieの出力は手遅れになります。

    キャンセル

  • 2018/12/01 01:12

    ありがとうございます!

    キャンセル

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 88.58%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る