B-Teck!

お仕事からゲームまで幅広く

【雑記】毎日1コミットを目指す習慣を始めた

2分間コーディングのすすめ、コードを書く習慣のハードルを下げる を見て、2分間だと何も出来なかったので毎日最低1コミットをしようという活動を、年度頭あたりから始めてみた。

現状はこんな感じで、とりあえず1ヶ月以上継続できている。
f:id:beatdjam:20210512224736p:plain
↑の画像の作成には Grass-Graph / Imaging your GitHub Contributions Graph を利用しています。

内容は雑多で、

  • KotlinでPDFを扱うライブラリのサンプルコード
  • このブログのCSSを管理するもの
  • 業務で扱うことになったPythonの写経
  • Githubのプロフィールにしているmdのメンテナンス
  • 興味の出たScalaの本の写経

など。

もともと、家で開発をするというのがかなりのハードルになっていて、土日に気力を振り絞ってやるみたいになっていたところを、とりあえず毎日ネタ探してなんか書くぞ!となっていたり、積んでいたScalaの本を読み進められたりしていて、今のところはいい感じ。
これが義務的になっておざなりになってしまうとだめなんだろうけど…。
1日の作業時間は10分〜数時間で、明らかに無理そうな日用のネタを作っていたりする部分もある。

blog.beatdjam.com ↑でも書いたんだけど、勝手に自分の中に高めのハードルを設けて、それを超えられないことに苦しみを感じてしまいがちな部分があって、それを意識して低くして、成功体験を積んで自信を持てるようになれたらいいなという気持ちです。

【GAM/Python】Google Ad ManagerのAPIをPythonで叩く

サービスアカウント作成と連携

Google Ad Manager APIを利用するためにはOAuth2認証が必要。
いくつか方法があるけど今回はサービスアカウントを作成して認証されるようにします。

サービスアカウント作成

  • Google Cloud Platformのコンソールからサービスアカウントを作成します
  • 作成したサービスアカウントでjsonキーを作成します
    • このjsonは後にプログラムからAPIに疎通する際に利用します

Python用ライブラリを読み込む

googleads/googleads-python-lib

  • pipを推奨されてますがPoetryでも問題なく導入できました。
$ poetry add googleads
  • 先の手順で取得したjsonを credential.json として保存して参照しています。
from googleads import ad_manager
from googleads import oauth2
import os

# 認証ファイルのpathを取得
script_dir = os.path.dirname(__file__)
KEY_FILE = os.path.join(script_dir, 'credential.json')

# 任意のアプリケーション名
APPLICATION_NAME = 'sample'

def main(key_file, application_name):
    oauth2_client = oauth2.GoogleServiceAccountClient(
        key_file, oauth2.GetAPIScope('ad_manager'))

    ad_manager_client = ad_manager.AdManagerClient(
        oauth2_client, application_name)

    networks = ad_manager_client.GetService('NetworkService').getAllNetworks()
    for network in networks:
        print('Network with network code "%s" and display name "%s" was found.'
              % (network['networkCode'], network['displayName']))

if __name__ == '__main__':
    main(KEY_FILE, APPLICATION_NAME)

上記のコードを実行して
Network with network code "対象のネットワークコード" and display name "ネットワークの名称" was found.
が出力されれば疎通は成功です。
お疲れ様でした。

【Python】ResponderのQuickStartをやった

仕事で Responderを使うことになったのでQuick Startをやった。
合間に自分用のメモを書きつつ実装したので、メモがてら残しておく。

Responder導入

前の記事 でPoetryを導入しているので、下記コマンドを実行して導入した。

$ poetry add responder 

実装

# Declare a Web Service
import responder

api = responder.API()


# Hello World!
@api.route("/")
def hello_world(req, resp):
    resp.text = "hello, world!"


# Accept Route Arguments
@api.route("/hello/{who}")
def hello_to(req, resp, *, who):
    resp.text = f"hello, {who}!"


# Returning JSON / YAML
# resp.mediaに値を詰めるとJSONにシリアライズして返却する
# clientがyamlを要求してきた場合(Accept : application/x-yamlなど)はyamlを返す
@api.route("/hello/{who}/json")
def hello_to(req, resp, *, who):
    resp.media = {"hello": who}


# Rendering a Template
# テンプレートのhtmlはtemplates配下に入れる
# テンプレートの文字列は {{ variable }} の形で指定する
@api.route("/hello/{who}/html")
def hello_html(req, resp, *, who):
    resp.html = api.template('hello.html', who=who)


# Setting Response Status Code
@api.route("/416")
def teapot(req, resp):
    resp.status_code = api.status_codes.HTTP_416


# Setting Response Headers
@api.route("/pizza")
def pizza_pizza(req, resp):
    resp.headers['X-Pizza'] = '42'


# Receiving Data & Background Tasks
# process_dataをバックグラウンドで実行しながら即時レスポンスする
@api.route("/incoming")
async def receive_incoming(req, resp):

    @api.background.task
    def process_data(data):
        f = open('./{}'.format(data['file']['filename']), 'w')
        f.write(data['file']['content'].decode('utf-8'))
        f.close()

    data = await req.media(format='files')
    process_data(data)

    resp.media = {'success': True}


# Run the Server
if __name__ == '__main__':
    api.run()
    # address、 portを設定する場合
    # api.run(address='0.0.0.0', port=5042)

最後の Receiving Data & Background Tasks は、サンプルコードを /incoming に向くように変更して動作確認をした。

import requests
data = {'file': ('hello.txt', 'hello, world!', "text/plain")}
r = requests.post('http://127.0.0.1:5042/incoming', files=data)
print(r.text)

f:id:beatdjam:20210413203322p:plain