1
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 3 years have passed since last update.

【PHP】Google広告APIでオフラインコンバージョンを自動アップロードしたい!

Last updated at Posted at 2021-11-18

#はじめに

以前YahooAPIでオフラインコンバージョンを自動でアップロードする方法について、
調べて記事にしたんですけど、それじゃGoogleは?というおはなし。

#そもそもオフラインコンバージョンてなんぞや。

Googleさんが広告タグに「gclid」という長い文字列をくっつけてくれるので、
コンバージョンに結びついたデータからそのIDを引っこ抜いて、
ついでに何時何分何秒にいくらでコンバージョンに至ったかの情報を
まるっと渡すと、GoogleさんがAIを駆使して分析してより細分化された広告設定ができるようになる。らしい。

(とは担当者談なので、実際のところよく知らないんだけど)

オフラインコンバージョンについての詳細はココがわかりやすい

※ちなみにYahooの場合は「yclid」というタグが付与されて同様の機能があります。
 APIでのデータの渡し方は、上述した私の過去記事で紹介してあります。

#なぜ今まで自動アップロードに対応していなかったか

Googleのオフラインコンバージョンは、
もちろんAPIアップロードが可能ではあるのですが、
管理画面の設定によって、スプレッドシートを時間とともに設定しておけば、
自動で読み込んでアップロードしてくれるという便利機能があるらしく。

担当者の方からは

「指定時間に、スプレッドシートに必要なデータを出力するプログラム作って」

と言われたのです。

APIから直接アップしたほうがはやくね?
とは頭をよぎりましたが、
しがない雇われエンジニアとしては

ふーん( ´_ゝ`)

てなもんで、おおせのとおりに。

せこせことスプレッドシートAPIの仕様を調べて
(ちゃっかり記事は残した)
スプレッドシートにコンバージョン情報を出力する仕組みを作ったわけです。
CRONで朝起動設定にしてました。

ところが最近の月末(だったかな?)
なんだかスプレッドシート出力のエラーが。出た。

それ以降ちらほらとなんかエラー起こる。
なんなら一つのスプレッドシートで1つのシートだけエラー落ちしてたりする。
起こらない日もある。

エラーになってたプログラムを手動起動したら、
エラー起こらず正常終了しちゃうし。

ナニコレ!?

謎に包まれたまま連日エラーログとにらめっこしながらあーだこーだしてたわけです。

そしたら運用担当者さんからですね、

「なんかもうAPIから直接アップロードしちゃったほうが早くね?」

とまあ、ぶっちゃけちゃったねキミィという感じの連絡をいただきました。
そうですねそのとおりですね。実は最初から思ってました。。

という経緯により、
スプレッドシート出力はやめて直接API叩こう!という方針に変わったのでした。

なんかスプレッドシートAPIがヤクタタズみたいな書き方になってしまいましたが
SheetsAPIたいへん便利です。使いみちいろいろ。

ただアレです、単に私のエンジニアとしての能力不足だったのです。
エラー処理だいじ。

ログ吐け。なんでもいいから全部ログに吐き出せ。
出すもの全部出して、必要なものを必要なときに自分で精査しな!ってなもんです。
これに尽きます。

裏で動かすものはなにか起きたときにとにかく追えるようにせねばならぬ。
頭では分かってたのにできてませんでした。深く反省。

APIを呼んだらRESPONSEをそのままログに出力するようにいろいろ直しました。

#コンバージョンアップロードメソッドuploadClickConversions

最初にググって出てきたのが、
Google更新ドキュメントのサンプルコードでした。

おう!もうやりたいことここ全部やんけ!!

という感じなのですが、まあ例によって
「意地でもサンプル使わず自力でやる」がモットーなので(なんかもうヤケクソ)

今回も自力でやりますよ。いつもの。

上記のドキュメントを解読すると

image.png

大事なのここっぽい。。。

調べたらリファレンスにありました。
customerに紐づくメソッドとのこと。

image.png

ClickConversionというオブジェクトに必要な情報が入ってて、
それを渡す感じのようです。ふむふむ。

ClickConversionとはなんぞや?まで調べてみると
リファレンスはここ

image.png

gclidとかDateとかValueとか必要な情報があって、めっちゃこれっぽい!!!

ここまで親切に教えてくれるともうゴールは見えてきました。
慣れって怖いですね。

#前提:認証(いつもの)

まずはおなじみのoAuth2認証。

認証に関しては、過去記事参照のこと。
【PHP】Google Ads APIをサンプルプログラム無しで使いたい!!

リフレッシュトークンとクライアントID、クライアントシークレットを使ってoAuth2認証を行い、
アクセストークンを取るところまで。

アクセストークンが取れたら、ヘッダーに認証情報をセットします。

//ヘッダー情報をセット
$_header = [
    "Content-Type: application/json",
    "Accept: application/json",
    "Authorization: Bearer $access_token", 
    "developer-token: $developer_token",
    "login-customer-id: $login_customer_id",
];

準備おーけー。

#ConversionActionを取得する

ClickConversionの中に、ひとつだけ見覚えのないものがありました。
それがConversionActionというもの。

image.png
Google翻訳:
この変換に関連付けられている変換アクションのリソース名。
注:このリソース名は顧客IDと変換アクションIDで構成されますが、検証では顧客IDが無視され、
変換アクションIDが変換アクションの唯一の識別子として使用されます。

りそーす!!?
なんだっけそれ←

わからないのでConversionActionで調べてみると
なるほどリソースの中にありましたconversion_action

image.png

これアレですね、いつもの、SQLみたいなクエリで引っ張ってこれるデータですね。
過去記事でも使ってるようなやつ

で、リソース名って書いてあったから、これだね
image.png

というわけで、クエリ言語でconversionactionを取ってみます。

$_query_str = "
SELECT
   conversion_action.id,
   conversion_action.name,
   conversion_action.type,
   conversion_action.status,
   conversion_action.resource_name
FROM conversion_action
";

//クエリをセット
$_query = ["query" => $_query_str];

//URLをセット
//※versionやcustomeridは環境に合わせてセットすること 2021/11現在の最新versionはv9
$_url = "https://googleads.googleapis.com/v9/customers/$customerid/googleAds:search";

//curl START
$_curl = curl_init();

//OPTIONをセット
curl_setopt_array($_curl, array(
    CURLOPT_URL => $_url,
    CURLOPT_POST => true,
    CURLOPT_RETURNTRANSFER => true,
    CURLOPT_TIMEOUT => 120,
    CURLOPT_HTTPHEADER => $_header,
    CURLOPT_POSTFIELDS => json_encode($_query),
));

//curl EXEC(文字列で取得)
$_resp = curl_exec($_curl);

//エラーハンドリング用
$_errno = curl_errno($_curl);

//curl END
curl_close($_curl);

//エラーハンドリング
if ($_errno !== CURLE_OK) {
    //エラー処理
}

//エラーでなければjsonを連想配列化
$_result_json = json_decode($_resp, true); 
$_result = is_array($_result_json["results"]) ? $_result_json["results"] : [];

//履歴情報を整理
$_return = [];
foreach ($_result as $_idx => $_lineval) {
    if (!is_array($_lineval)) continue;

    $_id = $_lineval["conversionAction"]["id"];//コンバージョンアクションID
    $_name = $_lineval["conversionAction"]["name"];//コンバージョンアクション名
    $_type = $_lineval["conversionAction"]["type"];//コンバージョンアクションタイプ
    $_status = $_lineval["conversionAction"]["status"];//コンバージョンアクションステータス
    $_resourcename = $_lineval["conversionAction"]["resourceName"];//コンバージョンアクションリソース名

    $_return[] = [
        $_id, $_name, $_type, $_status, $_resourcename,
    ];
}

conversionActionはこんなかんじのリソースが取れます:

customers/1234567890/conversionActions/123456789

ではでは、これをClickConversionの形に当てはめてアップロードしてみませう。

#実際にアップロードしてみる

まずはコンバージョンデータを整理します。
ClickConversionのライブラリを参考に、必要な項目をセットしましょう。

配列に入れて複数件を一気にアップロードできます。

$ClickConversion = [];
//1件目
$ClickConversion[] = [
    "gclid" => "ABcdefjHIjklmnOPqrstuVWxyz_1234567890",//gclid
    "conversionAction" => "customers/1234567890/conversionActions/123456789",//取得したConversionActionのリソース名
    "conversionDateTime" => "2021-11-01 15:01:11+09:00",//書式注意GMT入れる必要あり
    "conversionValue" => "10000",//
];
//2件目
$ClickConversion[] = [
    "gclid" => "1212ABcdefjHI3434-jklmn5656OPqrstu_789VWxyz0",//gclid
    "conversionAction" => "customers/1234567890/conversionActions/123456789",//取得したConversionActionのリソース名
    "conversionDateTime" => "2021-11-01 18:24:54+09:00",//書式注意GMT入れる必要あり
    "conversionValue" => "5000",//
];

ところで、
uploadClickConversionsメソッドのリクエストパラメータの中に、
validateOnlyというフラグがひっそりとおります。そう、ひっそりと。

image.png
Google翻訳:
trueの場合、要求は検証されますが実行されません。 結果ではなく、エラーのみが返されます。

こいつがテスト用か本番用かをしっかり分けてくれます。えらい。

validateOnlyをtrueにしておけば、実際にアップロードはされずエラーだけ教えてくれるのです。
なので、テストのときやエラーがないかをチェックしたいときはtrueにしましょう。

ということを踏まえた上で、
今度はClickConversionをアップロードする処理です。

//URLをセット
//※versionやcustomeridは環境に合わせてセットすること 2021/11現在の最新versionはv9
$_url = "https://googleads.googleapis.com/v9/customers/$customerid:uploadClickConversions";

//パラメータをセット
$_param = [
	"conversions" => $ClickConversion,
	"partialFailure" => true,
	"validateOnly" => true, //検証の場合はtrue,実行する場合はfalse
];

//curl START
$_curl = curl_init();

//OPTIONをセット
curl_setopt_array($_curl, array(
    CURLOPT_URL => $_url,
    CURLOPT_POST => true,
    CURLOPT_RETURNTRANSFER => true,
    CURLOPT_TIMEOUT => 120,
    CURLOPT_HTTPHEADER => $this->header,
	CURLOPT_POSTFIELDS => json_encode($_param),
));

//curl EXEC(文字列で取得)
$_resp = curl_exec($_curl);

//エラーハンドリング用
$_errno = curl_errno($_curl);

//curl END
curl_close($_curl);

//エラーハンドリング
if ($_errno !== CURLE_OK) {
    throw new Exception("Failed to GoogleAPI Access:".$_resp);
}

//エラーでなければjsonを連想配列化        
$_result = json_decode($_resp, true); 

validateOnlyをtrueにして実行した場合、こんなかんじのエラーが返ってきます。

{"partialFailureError":{
"code":3,"message":"This Google Click ID could not be decoded., at conversions[31].gclid",
"details":[{"@type":"type.googleapis.com/google.ads.googleads.v9.errors.GoogleAdsFailure",
"errors":[{"errorCode":{"conversionUploadError":"UNPARSEABLE_GCLID"},
"message":"This Google Click ID could not be decoded.",
"trigger":{"stringValue":"CjwKCAiA7dKMBhBCEiwAO_crFPZY68nU8i8Z0dkAd0GHCu8ENIAG-ofVv8ft7dhhoCiHQQAvD_BwE"},
"location":{"fieldPathElements":[{"fieldName":"conversions","index":31},
{"fieldName":"gclid"}]}}]}]},
"results":{}}

エラーがあれば教えてくれるけど、実行はしていないのでresultは空です。

validateOnlyをfalseにすると、resultの中にアップロードした内容が入ります。
たとえばこんなかんじ:

"results":[
{"gclid":"ABcdefjHIjklmnOPqrstuVWxyz_1234567890",
"conversionAction":"customers/1234567890/conversionActions/123456789",
"conversionDateTime":"2021-11-01 15:01:11+09:00"},
{"gclid":"1212ABcdefjHI3434-jklmn5656OPqrstu_789VWxyz0",
"conversionAction":"customers/1234567890/conversionActions/123456789",
"conversionDateTime":"2021-11-01 18:24:54+09:00"}}]

gclidとDateTimeを見てアップロード済みかどうかを判別しているようですね。

#アップロード確認

YahooAPIでのオフラインコンバージョンアップロードでは、
プログラム内でCSVファイルを作成⇒CSVファイルをアップロード
という手順を踏むためか、
アップロードした内容が管理画面で確認できるのですが。

Googleは直接アップロードするため、
アップロード内容が管理画面では確認できないようです。

担当者のかたによると、半日くらいかけてじわじわと反映されていくらしく、
コンバージョン値が反映されているかをじわじわ確認していくしかないとのことでした。

がんばれ。。。。

#おわりに

GoogleAPIも11月でV9がリリースされました。
β版のV1から触ってる身としては、感慨深いものがあります。

GoogleAPIとYahooAPIをあれこれ調べまくって試した結果、
「APIってこんなかんじ」というのがなんとなく分かってきた気がします。
まだぜんぜんへなちょこですが。

「これをすれば、あとはスムーズにできる」というのがなんとなくわかる。

APIの知識など完全ゼロベースから始めたことを考えると、
けっこう成長できたのかなと思います。

何事も継続は力なり、です。

次じつはFacebookAPIについての調査などが始まるのですが、
なんとか乗り越えていきたいと!
思っております!

おわり!!

1
1
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
1
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?