SQLの窓

2013年09月16日


POST oauth/request_token で、PHP の cURL 関数が 400 Bad Request になる原因( Content-Length: -1 )

昔はうまくいっていたコードが動かなくて、GET で処理すると正しく動作しましたが、POST で行う場合に POST データを空文字で指定する必要がありました。

実際ログ取って見てみると、CURLOPT_POSTFIELDS を指定しないと Content-Length: -1 になります。これを Bad Request とする仕様であると言う事みたいです。

この理由から、GET で処理すると問題無く動くという事です。

▼ エラーになるコード
	$curl = curl_init();
	curl_setopt($curl, CURLOPT_CONNECTTIMEOUT, 30);
	curl_setopt($curl, CURLOPT_HEADER, false);
	curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
	curl_setopt($curl, CURLOPT_BINARYTRANSFER, true);
	curl_setopt($curl, CURLOPT_URL, $twitter_url);
	curl_setopt($curl, CURLOPT_POST, 1);

▼ 正しく動作するコード
	$curl = curl_init();
	curl_setopt($curl, CURLOPT_CONNECTTIMEOUT, 30);
	curl_setopt($curl, CURLOPT_HEADER, false);
	curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
	curl_setopt($curl, CURLOPT_BINARYTRANSFER, true);
	curl_setopt($curl, CURLOPT_URL, $twitter_url);
	curl_setopt($curl, CURLOPT_POST, 1);
	curl_setopt($curl, CURLOPT_POSTFIELDS, "");

この違いで考えられる事はだいたい想像は付きますが、実際に WWW サーバーに送ってログ見ないと解らないです。


タグ:トラブル
posted by lightbox at 2013-09-16 14:41 | PHP + Twitter | このブログの読者になる | 更新情報をチェックする

2013年08月28日


PHP で GET statuses/user_timeline を使って何か作る為のテンプレート

GET statuses/user_timeline



基本的には、API 制限があるので( 300/app )たいした事はできませんが、個人的に楽しむぶんには十分使えると思います。投稿に関しては weetDeck 等の公式クライアントを使ったほうがいいと思いますが、PHP で投稿する方法はこちら

使用方法は、アプリを作って『Consumer key、Consumer secret、Access token、Access token secret』を手に入れてテキストエリアに1行づつ入力し、5行目にユーザの表示名( ログイン用の文字列 ) を入力します
<?php
header("Content-type: text/html; charset=utf-8");
header( "pragma: no-cache" );
header( "Expires: Wed, 31 May 2000 14:59:58 GMT" );
header( "Cache-control: no-cache" );

$result = '';
if ( $_SERVER['REQUEST_METHOD'] == 'POST' ) {
	$token = explode("\r\n", $_POST['text']);
	$result = twitter_user_timeline(
		$token[0],
		$token[1],
		$token[2],
		$token[3],
		$token[4]
	);
}

?>
<!doctype html>
<html lang="ja">
<head>
<title>SkyDrive Word</title>
<meta charset="utf-8">
<style>
body {
	margin: 0px;
	padding:10px 0 30px 30px;
}
.entry {
	width: 600px;
	border-top: solid 1px #000000;
	padding: 5px;
	font-size:16px;
	font-family: Arial, Helvetica, Verdana, "ヒラギノ角ゴPro W3", "Hiragino Kaku Gothic Pro", Osaka, "メイリオ", Meiryo, "MS Pゴシック", sans-serif;
}
textarea {
	width:540px;
	height:100px;
}
</style>
</head>
<body>

<form method="post">
<textarea name="text" style='float:left'><?= $_POST['text'] ?></textarea>
<input type="submit" value="送信" style='float:left;'>
</form>
<br style='clear:left'><br>

<?
if ( $result != '' ) {
	$obj = json_decode($result);
	$cnt = count($obj);
	for( $i = 0; $i < $cnt; $i++ ) {
		if ( $i == 0 ) {
			print "<div class='entry'>";
			print "<img src='{$obj[$i]->user->profile_image_url}'>";
			print "</div>";
		}
		print "<div class='entry'";

		// ***************************************************
		// リツイート
		// ***************************************************
		if ( $obj[$i]->retweeted_status ) {
			print  " style='background-color:#D6B6CE'>";
			print "<div style='float:left;width:60px;'><img style='width:48px;' src='{$obj[$i]->retweeted_status->user->profile_image_url}'></div>";
			// URL をアンカーに変換
			$work = url2a($obj[$i]->text);
			print "<div style='float:left;width:530px'>{$work}</div>";
		}
		else {
			// *******************************************
			// 返信
			// *******************************************
			if ( $obj[$i]->in_reply_to_screen_name ) {
				print  " style='background-color:#B6D6D6'>";
			}
			// *******************************************
			// 通常
			// *******************************************
			else {
				print  ">";
			}
			// URL をアンカーに変換
			print url2a($obj[$i]->text);
			print "<br>";
		}

		// ***************************************************
		// 画像がある場合
		// ***************************************************
		if ( $obj[$i]->entities->media ) {
			print "<br>";
			print "<div style='clear:left;'><img src='{$obj[$i]->entities->media[0]->media_url}' style='width:200px;'></div>";
		}
		else {
			print "<br style='clear:left;'>";
		}

		print "</div>";
	}

print "<pre>";
//print_r($obj);
print "</pre>";

}
?>
</body>
</html>
<?

// **********************************************************
// AOuth 用の urlencode 関数
// **********************************************************
function urle( $str ) {
	// php 5.3.x 〜 ではこの変換は必要無い
	return str_replace('%7E', '~', rawurlencode($str));
}

function twitter_user_timeline( $apikey, $secret, $token, $token_secret, $userid ) {
	// **********************************************************
	// API
	// **********************************************************
	$twitter_url = 'https://api.twitter.com/1.1/statuses/user_timeline.json';

	// **********************************************************
	// 認証データ
	// **********************************************************
	$oauth_consumer_key = $apikey;
	$oauth_consumer_secret = $secret;
	$oauth_token = $token;
	$oauth_secret = $token_secret;
	
	// 毎回変化するランダムな文字列
	$mt = microtime();
	$rand = mt_rand();
	$oauth_nonce = md5($mt . $rand);
	
	$oauth_signature_method = "HMAC-SHA1";
	$oauth_timestamp = time();
	$oauth_version = "1.0";
	
	// *********************************************************
	// シグネチャ用ベース文字列作成
	/*
	  httpMethod + "&" +
	  url_encode(  base_uri ) + "&" +
	  sorted_query_params.each  { | k, v |
	      url_encode ( k ) + "%3D" +
	      url_encode ( v )
	  }.join("%26")
	*/
	// *********************************************************
	$base_string = "GET";
	$base_string .= "&" . urle($twitter_url);
	$base_string .= "&";
	
	$base_string .= urle("oauth_consumer_key")."%3D".urle($oauth_consumer_key)."%26";
	$base_string .= urle("oauth_nonce")."%3D".urle($oauth_nonce)."%26";
	$base_string .= urle("oauth_signature_method")."%3D".urle($oauth_signature_method)."%26";
	$base_string .= urle("oauth_timestamp")."%3D".urle($oauth_timestamp)."%26";
	$base_string .= urle("oauth_token")."%3D".urle($oauth_token)."%26";
	$base_string .= urle("oauth_version")."%3D".urle($oauth_version);
	$base_string .= "%26screen_name%3D".urle($userid);
	
	// *********************************************************
	// シグネチャ作成
	/*
	url_encode( consumer_secret ) + "&" +
	url_encode( oauth_token_secret || nil )
	*/
	// *********************************************************
	$oauth_signature = 
	base64_encode( hash_hmac(
		"sha1",
		$base_string,
		$oauth_consumer_secret . "&" . $oauth_secret,
		true
	));
	
	// *********************************************************
	// curl 処理
	// *********************************************************
	$curl = curl_init();
	curl_setopt($curl, CURLOPT_CONNECTTIMEOUT, 30);
	curl_setopt($curl, CURLOPT_HEADER, false);
	curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
	curl_setopt($curl, CURLOPT_BINARYTRANSFER, true);
	curl_setopt($curl, CURLOPT_URL, $twitter_url . "?screen_name=" . urle($userid));

	// *********************************************************
	// http ヘッダ作成
	// *********************************************************
	$header = array();
	$header[] = 'Expect:';
	
	$header[] = "Authorization: OAuth ".
	urle("oauth_consumer_key")."=\"".urle($oauth_consumer_key)."\",".
	urle("oauth_token")."=\"".urle($oauth_token)."\",".
	urle("oauth_nonce")."=\"".urle($oauth_nonce)."\",".
	urle("oauth_timestamp")."=\"".urle($oauth_timestamp)."\",".
	urle("oauth_signature_method")."=\"".urle($oauth_signature_method)."\",".
	urle("oauth_version")."=\"".urle($oauth_version)."\",".
	urle("oauth_signature")."=\"".urle($oauth_signature)."\"";
	
	curl_setopt($curl, CURLOPT_HTTPHEADER, $header);
	
	// *********************************************************
	// https 用 ( https://api.twitter.com 利用時に必要 )
	// *********************************************************
	curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false);
	curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, 1);
	
	// *********************************************************
	// 戻された http ヘッダの出力
	// *********************************************************
	$handle = fopen("./header.txt", "w");
	curl_setopt($curl, CURLOPT_WRITEHEADER, $handle);
	
	// *********************************************************
	// 送信
	// *********************************************************
	$result = curl_exec($curl);
	
	// *********************************************************
	// 結果
	// *********************************************************
	$ret = true;
	if($result === false) {
		$ret = false;
	}
	curl_close($curl);

	return $result;

}

// URL をアンカーに変換
function url2a( $target ){

	$target = str_replace("http", " http", $target );

	if ( preg_match( "/http(s)?:\/\/([\w-]+\.)+[\w-]+(\/[\w- .\/?%&=]*)?/", $target ) ) {
		preg_match_all( "/http(s)?:\/\/([\w-]+\.)+[\w-]+(\/[\w- .\/?%&=]*)?/", $target, $pattarn );
		foreach ( $pattarn[0] as $key => $val ){
			$replace[] = "<a href=\"{$val}\" target=\"_blank\">{$val}</a>";
		}
		$target = str_replace( $pattarn[0], $replace, $target );
	}
	return $target;
}
?>

関連する記事

PHP + cURL で、Twitter API 1.1 の search/tweets を呼び出して、RSS に変換する



posted by lightbox at 2013-08-28 16:34 | PHP + Twitter | このブログの読者になる | 更新情報をチェックする

2013年07月11日


PHP : Twitter 投稿関数( twitter_update ) / cURL 関数

cURL 関数を使った単純な投稿用です。( windows では、php.ini で extension=php_curl.dll を有効にして下さい )

<?php
header("Content-type: text/html; charset=utf-8");

$result = twitter_update(
		"Consumer key",
		"Consumer secret",
		"Access token",
		"Access token secret",
		"PHPより投稿しています"
);

print "<pre>";
print_r(json_decode($result));
print "</pre>";

// **********************************************************
// AOuth 用の urlencode 関数
// **********************************************************
function urle( $str ) {
	// php 5.3.x 〜 ではこの変換は必要無い
	return str_replace('%7E', '~', rawurlencode($str));
}

function twitter_update( $apikey, $secret, $token, $token_secret, $text ) {
	// **********************************************************
	// API
	// **********************************************************
	$twitter_url = 'https://api.twitter.com/1.1/statuses/update.json';
	
	// **********************************************************
	// 認証データ
	// **********************************************************
	$oauth_consumer_key = $apikey;
	$oauth_consumer_secret = $secret;
	$oauth_token = $token;
	$oauth_secret = $token_secret;
	
	// 毎回変化するランダムな文字列
	$mt = microtime();
	$rand = mt_rand();
	$oauth_nonce = md5($mt . $rand);
	
	$oauth_signature_method = "HMAC-SHA1";
	$oauth_timestamp = time();
	$oauth_version = "1.0";
	
	// *********************************************************
	// シグネチャ用ベース文字列作成
	/*
	  httpMethod + "&" +
	  url_encode(  base_uri ) + "&" +
	  sorted_query_params.each  { | k, v |
	      url_encode ( k ) + "%3D" +
	      url_encode ( v )
	  }.join("%26")
	*/
	// *********************************************************
	$base_string = "POST";
	$base_string .= "&" . urle($twitter_url);
	$base_string .= "&";
	
	$base_string .= urle("oauth_consumer_key")."%3D".urle($oauth_consumer_key)."%26";
	$base_string .= urle("oauth_nonce")."%3D".urle($oauth_nonce)."%26";
	$base_string .= urle("oauth_signature_method")."%3D".urle($oauth_signature_method)."%26";
	$base_string .= urle("oauth_timestamp")."%3D".urle($oauth_timestamp)."%26";
	$base_string .= urle("oauth_token")."%3D".urle($oauth_token)."%26";
	$base_string .= urle("oauth_version")."%3D".urle($oauth_version);
	
	$base_string .= "%26status" . "%3D" . urle(urle($text));
	
	// *********************************************************
	// シグネチャ作成
	/*
	url_encode( consumer_secret ) + "&" +
	url_encode( oauth_token_secret || nil )
	*/
	// *********************************************************
	$oauth_signature = 
	base64_encode( hash_hmac(
		"sha1",
		$base_string,
		$oauth_consumer_secret . "&" . $oauth_secret,
		true
	));
	
	// *********************************************************
	// curl 処理
	// *********************************************************
	$curl = curl_init();
	curl_setopt($curl, CURLOPT_CONNECTTIMEOUT, 30);
	curl_setopt($curl, CURLOPT_HEADER, false);
	curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
	curl_setopt($curl, CURLOPT_BINARYTRANSFER, true);
	curl_setopt($curl, CURLOPT_URL, $twitter_url);
	curl_setopt($curl, CURLOPT_POST, 1);
	curl_setopt($curl, CURLOPT_POSTFIELDS, "status=" . urle($text));
	
	// *********************************************************
	// http ヘッダ作成
	// *********************************************************
	$header = array();
	$header[] = 'Expect:';
	
	$header[] = "Authorization: OAuth ".
	urle("oauth_consumer_key")."=\"".urle($oauth_consumer_key)."\",".
	urle("oauth_token")."=\"".urle($oauth_token)."\",".
	urle("oauth_nonce")."=\"".urle($oauth_nonce)."\",".
	urle("oauth_timestamp")."=\"".urle($oauth_timestamp)."\",".
	urle("oauth_signature_method")."=\"".urle($oauth_signature_method)."\",".
	urle("oauth_version")."=\"".urle($oauth_version)."\",".
	urle("oauth_signature")."=\"".urle($oauth_signature)."\"";
	
	curl_setopt($curl, CURLOPT_HTTPHEADER, $header);
	
	// *********************************************************
	// https 用 ( https://api.twitter.com 利用時に必要 )
	// *********************************************************
	curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false);
	curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, 1);
	
	// *********************************************************
	// 戻された http ヘッダの出力
	// *********************************************************
	$handle = fopen("./header.txt", "w");
	curl_setopt($curl, CURLOPT_WRITEHEADER, $handle);
	
	// *********************************************************
	// 送信
	// *********************************************************
	$result = curl_exec($curl);
	
	// *********************************************************
	// 結果
	// *********************************************************
	$ret = true;
	if($result === false) {
		$ret = false;
	}
	curl_close($curl);

	return $result;

}
?>

関連する記事

Twitter API の自分のアプリのトークンを使って投稿するだけの class VS2010_Twitter

Twitter API の自分のアプリのトークンを使って投稿するだけの class Android_Twitter

Twitter API の自分のアプリのトークンを使って投稿するだけの class VS2012_Twitter

WSH : VBScript と JavaScript で Twitter に投稿する

Twitter アプリの登録方法と、API キーの利用


posted by lightbox at 2013-07-11 21:11 | PHP + Twitter | このブログの読者になる | 更新情報をチェックする

2013年06月02日


PHP + cURL で、Twitter API 1.1 の search/tweets を呼び出して、RSS に変換する

自分のアプリケーションで検索するなら、制限的には 450 となっているので、『自分専用』として運用も可能なので、他のアプリから呼び出す為のテスト用として作成しました。

search.twitter.com/search は、https://dev.twitter.com/blog/api-v1-retirement-date-extended-to-june-11 では、6月11日には、使えなくなるとの事です。
Requests to search.twitter.com/search.* will receive HTTP 410 Gone. Use GET search/tweets instead.
<?php
// **********************************************************
// Twitter API 1.1 search/tweets => RSS
// **********************************************************
header( "Content-Type: text/xml; Charset=utf-8" );
header( "pragma: no-cache" );
header( "Expires: Wed, 31 May 2000 14:59:58 GMT" );
header( "Cache-control: no-cache" );

// **********************************************************
// OAuth 用の urlencode 関数
// **********************************************************
function url_rfc3986( $str ) {
	// php 5.3.x 〜 ではこの変換は必要無く、rawurlencode で OK
	return str_replace('%7E', '~', rawurlencode($str));
}

// **********************************************************
// API
// **********************************************************
$twitter_url = 'https://api.twitter.com/1.1/search/tweets.json';

// **********************************************************
// 認証データ
// **********************************************************
$oauth_consumer_key = "";
$oauth_consumer_secret = "";
$oauth_token = "";
$oauth_secret = "";

// 毎回変化するランダムな文字列
$mt = microtime();
$rand = mt_rand();
$oauth_nonce = md5($mt . $rand);

$oauth_signature_method = "HMAC-SHA1";
$oauth_timestamp = time();
$oauth_version = "1.0";

if ( $_POST['q'] != '' ) {
	$query = $_POST['q'];
}
else {
	$query = "進撃の巨人";
}
if ( $_POST['c'] != '' ) {
	$count = $_POST['c'];
}
else {
	$count = 20;
}

// *********************************************************
// シグネチャ用ベース文字列作成
// *********************************************************
$base_string = "GET";
$base_string .= "&" . url_rfc3986($twitter_url);
$base_string .= "&count%3D{$count}%26";

$base_string .= url_rfc3986("oauth_consumer_key")."%3D".url_rfc3986($oauth_consumer_key)."%26";
$base_string .= url_rfc3986("oauth_nonce")."%3D".url_rfc3986($oauth_nonce)."%26";
$base_string .= url_rfc3986("oauth_signature_method")."%3D".url_rfc3986($oauth_signature_method)."%26";
$base_string .= url_rfc3986("oauth_timestamp")."%3D".url_rfc3986($oauth_timestamp)."%26";
$base_string .= url_rfc3986("oauth_token")."%3D".url_rfc3986($oauth_token)."%26";
$base_string .= url_rfc3986("oauth_version")."%3D".url_rfc3986($oauth_version)."%26";
$base_string .= "q%3D". url_rfc3986(url_rfc3986($query));

// *********************************************************
// シグネチャ作成
// *********************************************************
$oauth_signature = 
base64_encode( hash_hmac(
	"sha1",
	$base_string,
	url_rfc3986($oauth_consumer_secret) . "&" . url_rfc3986($oauth_secret),
	true
));

// *********************************************************
// curl 処理
// *********************************************************
$curl = curl_init();
curl_setopt($curl, CURLOPT_CONNECTTIMEOUT, 30);
curl_setopt($curl, CURLOPT_HEADER, false);
curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
curl_setopt($curl, CURLOPT_BINARYTRANSFER, true);
curl_setopt($curl, CURLOPT_URL, $twitter_url . "?q=". url_rfc3986($query) . "&count={$count}" );

// *********************************************************
// http ヘッダ作成
// *********************************************************
$header = array();
//$header[] = 'Expect:';
$header[] = 'Authorization: OAuth '.
url_rfc3986("oauth_consumer_key")."=\"".url_rfc3986($oauth_consumer_key)."\",".
url_rfc3986("oauth_nonce")."=\"".url_rfc3986($oauth_nonce)."\",".
url_rfc3986("oauth_signature")."=\"".url_rfc3986($oauth_signature)."\",".
url_rfc3986("oauth_signature_method")."=\"".url_rfc3986($oauth_signature_method)."\",".
url_rfc3986("oauth_timestamp")."=\"".url_rfc3986($oauth_timestamp)."\",".
url_rfc3986("oauth_token")."=\"".url_rfc3986($oauth_token)."\",".
url_rfc3986("oauth_version")."=\"".url_rfc3986($oauth_version)."\"";

curl_setopt($curl, CURLOPT_HTTPHEADER, $header);

// *********************************************************
// https 用
// *********************************************************
curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false);
curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, 1);

// *********************************************************
// 送信
// *********************************************************
//curl_setopt($curl, CURLOPT_VERBOSE, true);	// デバッグ
//$handle = fopen("./debug.txt", "w");
//curl_setopt($curl, CURLOPT_STDERR, $handle);
//$handle2 = fopen("./ret_header.txt", "w");
//curl_setopt($curl, CURLOPT_WRITEHEADER, $handle2);

$result = curl_exec($curl);


// *********************************************************
// 結果
// *********************************************************
if($result === false) {
	// $json = 'Curl error: ' . curl_error($curl);
	$json = json_decode("{}");
}
else {
	// echo 'Operation completed without any errors';
	$json = json_decode($result);
}
curl_close($curl);
//fclose($handle2);
//fclose($handle);


print '<?xml version="1.0" encoding="UTF-8"?>';
?>
<rss xmlns:atom="http://www.w3.org/2005/Atom" xmlns:media="http://search.yahoo.com/mrss/" xmlns:dc="http://purl.org/dc/elements/1.1/" version="2.0">
<channel>
<?php

if ( $json->statuses ) {
foreach($json->statuses as $tweet) {

$str = <<<ITEM
<item>
	<title>{$tweet->user->name}</title>
	<link/>
	<description>{$tweet->text}</description>
	<pubDate>{$tweet->created_at}</pubDate>
	<guid>{$tweet->id_str}</guid>
	<dc:creator>{$tweet->user->screen_name}</dc:creator>
	<dc:date>{$dc_date}</dc:date>
	<media:thumbnail url="{$tweet->user->profile_image_url}" />
	<media:content url="{$tweet->user->profile_image_url}" />
</item>
ITEM;

	print $str;

}
}
else {

$str = <<<ITEM
<item>
	<title>error</title>
	<link/>
	<description/>
	<pubDate/>
	<guid/>
	<dc:creator/>
	<dc:date/>
	<media:thumbnail url="" />
	<media:content url="" />
</item>
ITEM;

	print $str;
}

?>
</channel>
</rss>



posted by lightbox at 2013-06-02 01:28 | PHP + Twitter | このブログの読者になる | 更新情報をチェックする

2011年06月12日


PHP : Twitter への投稿 : バージョン2

PHP : Twitter への投稿 : バージョン1 では、自分専用のアプリケーションとしてテストしていたので、トークンは直接 Twitter アカウント内から取りだして、「クライアント型」で実行していましたが、本来 Twitter アプリケートションは自分以外の人が使う為に、その人用のトークンを動的に取得するようになっています。

その為のパッケージです。


まず、アプリケーションの設定で「ブラウザ型」に変更します。その際、コールバックする為の http:// から始まる通常の URL を書く必要がありますが、実在している必要はありません。プログラムで変更して指定するので自分のサイトの好きなところでもかまいません

サンプルのプロセスは、3つになりますが、実際は二つです。この場合は途中の値を見る為に、コールバック以降を二つにしています。また、一旦取得したトークンを以降もWEB上で継続して使えるようにするために localStorage に保存しています。ここでは作成していませんが、ログアウトする処理も必要です。

トークンを PC に保存するので、盗まれる可能性はありますが、その場合は正式アカウントから、ユーザーがそのアプリケーションを削除すればそのトークンは無効になります。それでも心配な場合は、毎回 Twitter で認証させる必要があります。
<?php
session_start();
$_SESSION['oauth_token_secret'] = '';
// **********************************************************
// Twitter への投稿 : バージョン2
// **********************************************************
header( "Content-Type: text/html; Charset=utf-8" );
header( "pragma: no-cache" );
header( "Expires: Wed, 31 May 2000 14:59:58 GMT" );
header( "Cache-control: no-cache" );

// **********************************************************
// AOuth 用の urlencode 関数
// **********************************************************
function urle( $str ) {
	// php 5.3.x 〜 ではこの変換は必要無い
	return str_replace('%7E', '~', rawurlencode($str));
}

// **********************************************************
// 投稿 API
// **********************************************************
//$twitter_url = 'http://api.twitter.com/1/statuses/update.json';
// **********************************************************
// トークン要求 API ( SSL is recommended )
// **********************************************************
$twitter_url = 'https://api.twitter.com/oauth/request_token';

// **********************************************************
// 認証データ
// **********************************************************
$oauth_consumer_key = "アプリケーションページより取得";
$oauth_consumer_secret = "アプリケーションページより取得";
$oauth_token = "";
$oauth_secret = "";

// 毎回変化するランダムな文字列
$mt = microtime();
$rand = mt_rand();
$oauth_nonce = md5($mt . $rand);

$oauth_signature_method = "HMAC-SHA1";
$oauth_timestamp = mktime();
$oauth_version = "1.0";

// コールバックする URL
$path_parts = pathinfo($_SERVER['PHP_SELF']);
$oauth_callback = "http://localhost{$path_parts['dirname']}/callback_test.php";

// *********************************************************
// シグネチャ用ベース文字列作成
/*
  httpMethod + "&" +
  url_encode(  base_uri ) + "&" +
  sorted_query_params.each  { | k, v |
      url_encode ( k ) + "%3D" +
      url_encode ( v )
  }.join("%26")
*/
// *********************************************************
$base_string = "POST";
$base_string .= "&" . urle($twitter_url);
$base_string .= "&";

// 投稿用
//$base_string .= urle("oauth_consumer_key")."%3D".urle($oauth_consumer_key)."%26";
//$base_string .= urle("oauth_nonce")."%3D".urle($oauth_nonce)."%26";
//$base_string .= urle("oauth_signature_method")."%3D".urle($oauth_signature_method)."%26";
//$base_string .= urle("oauth_timestamp")."%3D".urle($oauth_timestamp)."%26";
//$base_string .= urle("oauth_token")."%3D".urle($oauth_token)."%26";
//$base_string .= urle("oauth_version")."%3D".urle($oauth_version);

$base_string .= urle("oauth_callback")."%3D".urle(urle($oauth_callback))."%26";
$base_string .= urle("oauth_consumer_key")."%3D".urle($oauth_consumer_key)."%26";
$base_string .= urle("oauth_nonce")."%3D".urle($oauth_nonce)."%26";
$base_string .= urle("oauth_signature_method")."%3D".urle($oauth_signature_method)."%26";
$base_string .= urle("oauth_timestamp")."%3D".urle($oauth_timestamp)."%26";
$base_string .= urle("oauth_version")."%3D".urle($oauth_version);

// **********************************************************
// 投稿データ
// **********************************************************
// POST データは2度 urlencode する
//$text = "Twitter への投稿 : バージョン1 : nonce=" . $oauth_nonce;
//$text .= " : timestamp=" . $oauth_timestamp;
//$base_string .= "%26status" . "%3D" . urle(urle($text));

// *********************************************************
// シグネチャ作成
/*
url_encode( consumer_secret ) + "&" +
url_encode( oauth_token_secret || nil )
*/
// *********************************************************
$oauth_signature = 
base64_encode( hash_hmac(
	"sha1",
	$base_string,
	$oauth_consumer_secret . "&",
	true
));

// *********************************************************
// curl 処理
// *********************************************************
$curl = curl_init();
curl_setopt($curl, CURLOPT_CONNECTTIMEOUT, 30);
curl_setopt($curl, CURLOPT_HEADER, false);
curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
curl_setopt($curl, CURLOPT_BINARYTRANSFER, true);
curl_setopt($curl, CURLOPT_URL, $twitter_url);
curl_setopt($curl, CURLOPT_POST, 1);

// *********************************************************
// http ヘッダ作成
// *********************************************************
$header = array();
$header[] = 'Expect:';
$header[] = 'Content-Length:';

$header[] = 'Authorization: OAuth '.
urle("oauth_nonce")."="".urle($oauth_nonce)."",".
urle("oauth_callback")."="".urle($oauth_callback)."",".
urle("oauth_signature_method")."="".urle($oauth_signature_method)."",".
urle("oauth_timestamp")."="".urle($oauth_timestamp)."",".
urle("oauth_consumer_key")."="".urle($oauth_consumer_key)."",".
urle("oauth_signature")."="".urle($oauth_signature)."",".
urle("oauth_version")."="".urle($oauth_version).""";

curl_setopt($curl, CURLOPT_HTTPHEADER, $header);

// *********************************************************
// https 用 ( https://api.twitter.co 利用時に必要 )
// *********************************************************
curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false);
curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, 1);

// *********************************************************
// 戻された http ヘッダの出力
// *********************************************************
$handle = fopen("./header1.txt", "w");
curl_setopt($curl, CURLOPT_WRITEHEADER, $handle);

// *********************************************************
// 送信
// *********************************************************
$result = curl_exec($curl);


// *********************************************************
// 結果
// *********************************************************
print "<br>";

if($result === false) {
	print '送信処理にエラーが発生しました: ' . curl_error($curl);
}
else {
	print '送信処理は正常に終了しました';
}
curl_close($curl);

// *********************************************************
// 結果表示
// *********************************************************
print "<br>";

parse_str($result,$arr);
print "<pre>";
print_r( $arr );
print "</pre>";

$_SESSION['oauth_token_secret'] = $arr['oauth_token_secret'];

?>
<a href="https://twitter.com/oauth/authorize?oauth_token=<?= $arr['oauth_token'] ?>"
>Twitterで許可する為のページ</a>

このコードは、Twitter で認証させる為のリンクを用意するページです。まだ一時的なリクエスト用のトークンしか取得できていません。'oauth_token_secret' は、コールバック先で使用する為に、セッションで引き渡しています。

コールバックするページはテストの為に localhost を使用していますが、通常は自分のサイト上( 他のユーザがアクセスできる場所 ) を指定します


Twitter で投稿する為のトークンを取得するコールバック後のデータ表示
Twitter で投稿する為のトークンを取得するコールバック後の Twitter アクセス



posted by lightbox at 2011-06-12 12:53 | PHP + Twitter | このブログの読者になる | 更新情報をチェックする

2011年04月16日


PHP : Twitter への投稿 : バージョン1

Twitter のアカウントが必要です。New Twitter Application | dev.twitter.com よりアプリケーションを登録します。


※ アプリケーションのWEBサイトは適当に自分゛管理する URLをセットします
( 省略できません )
※ この時点で、コールバックは必要ありません

登録されたアプリケーションの情報を元に、自分でアクセスします( 実行すると、連携アプリとして登録されます )。この情報は本来ログインで一時的に取得できるものが含まれていますが、自分のアプリケーションには登録時に一つ用意されています。それを使うのでログインは必要ありません。( 連携アプリを削除すると My Access Token はリセットされます )

以下のサンプルでは、$text に投稿データを UTF-8 でセットするようになっています。FORM の テキストエリアから投稿したい場合は、データをセットし直して下さい。

関連する記事

PHP をテストする為の初心者用フォーム
<?php
// **********************************************************
// Twitter への投稿 : バージョン1
// **********************************************************
header( "Content-Type: text/html; Charset=utf-8" );
header( "pragma: no-cache" );
header( "Expires: Wed, 31 May 2000 14:59:58 GMT" );
header( "Cache-control: no-cache" );

// **********************************************************
// AOuth 用の urlencode 関数
// **********************************************************
function urle( $str ) {
	// php 5.3.x 〜 ではこの変換は必要無い
	return str_replace('%7E', '~', rawurlencode($str));
}

// **********************************************************
// API
// **********************************************************
$twitter_url = 'http://api.twitter.com/1/statuses/update.json';

// **********************************************************
// 認証データ
// **********************************************************
$oauth_consumer_key = "アプリケーションページより取得";
$oauth_consumer_secret = "アプリケーションページより取得";
$oauth_token = "My Access Token ページより取得";
$oauth_secret = "My Access Token ページより取得";

// 毎回変化するランダムな文字列
$mt = microtime();
$rand = mt_rand();
$oauth_nonce = md5($mt . $rand);

$oauth_signature_method = "HMAC-SHA1";
$oauth_timestamp = mktime();
$oauth_version = "1.0";


// *********************************************************
// シグネチャ用ベース文字列作成
/*
  httpMethod + "&" +
  url_encode(  base_uri ) + "&" +
  sorted_query_params.each  { | k, v |
      url_encode ( k ) + "%3D" +
      url_encode ( v )
  }.join("%26")
*/
// *********************************************************
$base_string = "POST";
$base_string .= "&" . urle($twitter_url);
$base_string .= "&";

$base_string .= urle("oauth_consumer_key")."%3D".urle($oauth_consumer_key)."%26";
$base_string .= urle("oauth_nonce")."%3D".urle($oauth_nonce)."%26";
$base_string .= urle("oauth_signature_method")."%3D".urle($oauth_signature_method)."%26";
$base_string .= urle("oauth_timestamp")."%3D".urle($oauth_timestamp)."%26";
$base_string .= urle("oauth_token")."%3D".urle($oauth_token)."%26";
$base_string .= urle("oauth_version")."%3D".urle($oauth_version);

// **********************************************************
// 投稿データ
// **********************************************************
// POST データは2度 urlencode する
$text = "Twitter への投稿 : バージョン1 : nonce=" . $oauth_nonce;
$text .= " : timestamp=" . $oauth_timestamp;
$base_string .= "%26status" . "%3D" . urle(urle($text));

// *********************************************************
// シグネチャ作成
/*
url_encode( consumer_secret ) + "&" +
url_encode( oauth_token_secret || nil )
*/
// *********************************************************
$oauth_signature = 
base64_encode( hash_hmac(
	"sha1",
	$base_string,
	$oauth_consumer_secret . "&" . $oauth_secret,
	true
));

// *********************************************************
// curl 処理
// *********************************************************
$curl = curl_init();
curl_setopt($curl, CURLOPT_CONNECTTIMEOUT, 30);
curl_setopt($curl, CURLOPT_HEADER, false);
curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
curl_setopt($curl, CURLOPT_BINARYTRANSFER, true);
curl_setopt($curl, CURLOPT_URL, $twitter_url);
curl_setopt($curl, CURLOPT_POST, 1);
curl_setopt($curl, CURLOPT_POSTFIELDS, "status=" . urle($text));

// *********************************************************
// http ヘッダ作成
// *********************************************************
$header = array();
$header[] = 'Expect:';

$header[] = "Authorization: OAuth ".
urle("oauth_consumer_key")."="".urle($oauth_consumer_key)."",".
urle("oauth_token")."="".urle($oauth_token)."",".
urle("oauth_nonce")."="".urle($oauth_nonce)."",".
urle("oauth_timestamp")."="".urle($oauth_timestamp)."",".
urle("oauth_signature_method")."="".urle($oauth_signature_method)."",".
urle("oauth_version")."="".urle($oauth_version)."",".
urle("oauth_signature")."="".urle($oauth_signature).""";

curl_setopt($curl, CURLOPT_HTTPHEADER, $header);

// *********************************************************
// https 用 ( https://api.twitter.co 利用時に必要 )
// *********************************************************
//curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false);
//curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, 1);

// *********************************************************
// 戻された http ヘッダの出力
// *********************************************************
$handle = fopen("./header.txt", "w");
curl_setopt($curl, CURLOPT_WRITEHEADER, $handle);

// *********************************************************
// 送信
// *********************************************************
$result = curl_exec($curl);


// *********************************************************
// 結果
// *********************************************************
print "<br>";

if($result === false) {
	print '送信処理にエラーが発生しました: ' . curl_error($curl);
}
else {
	print '送信処理は正常に終了しました';
}
curl_close($curl);

// *********************************************************
// 結果表示
// *********************************************************
print "<br>";

$result = json_decode( $result );

print "<pre>";
print_r( $result );
print "</pre>";

?>


アプリケーション一覧ページ
※ 上のページでアプリケーションのアイコンをクリックすると個別のアプリケーションページです
※ そのページの右サイドのメニューに My Access Token へのリンクがあります



posted by lightbox at 2011-04-16 14:54 | PHP + Twitter | このブログの読者になる | 更新情報をチェックする
Seesaa の各ページの表示について
Seesaa の 記事がたまに全く表示されない場合があります。その場合は、設定> 詳細設定> ブログ設定 で 最新の情報に更新の『実行ボタン』で記事やアーカイブが最新にビルドされます。

Seesaa のページで、アーカイブとタグページは要注意です。タグページはコンテンツが全く無い状態になりますし、アーカイブページも歯抜けページはコンテンツが存在しないのにページが表示されてしまいます。

また、カテゴリページもそういう意味では完全ではありません。『カテゴリID-番号』というフォーマットで表示されるページですが、実際存在するより大きな番号でも表示されてしまいます。

※ インデックスページのみ、実際の記事数を超えたページを指定しても最後のページが表示されるようです

対処としては、このようなヘルプ的な情報を固定でページの最後に表示するようにするといいでしょう。具体的には、メインの記事コンテンツの下に『自由形式』を追加し、アーカイブとカテゴリページでのみ表示するように設定し、コンテンツを用意するといいと思います。


※ エキスパートモードで表示しています

アーカイブとカテゴリページはこのように簡単に設定できますが、タグページは HTML 設定を直接変更して、以下の『タグページでのみ表示される内容』の記述方法で設定する必要があります

<% if:page_name eq 'archive' -%>
アーカイブページでのみ表示される内容
<% /if %>

<% if:page_name eq 'category' -%>
カテゴリページでのみ表示される内容
<% /if %>

<% if:page_name eq 'tag' -%>
タグページでのみ表示される内容
<% /if %>
この記述は、以下の場所で使用します
container 終わり



フリーフォントで簡単ロゴ作成
フリーフォントでボタン素材作成
フリーフォントで吹き出し画像作成
フリーフォントではんこ画像作成
ほぼ自由に利用できるフリーフォント
フリーフォントの書体見本とサンプル
画像を大きく見る為のウインドウを開くボタンの作成

CSS ドロップシャドウの参考デモ
イラストAC
ぱくたそ
写真素材 足成
フリーフォント一覧
utf8 文字ツール
右サイド 終わり
base 終わり