我想推广使用App Engine,所以用Docker Compose快速搭建了Go + Datastore的开发环境的经历

使用Docker Compose轻松搭建最新版本的Google App Engine Go运行时开发环境的故事,或者传播可成本合理运营服务器的App Engine的故事。

太长了,没看。

    • 環境構築

https://github.com/pistatium/appgengine-go112-datastore-docker-compose
動くサンプルはこちら。

docker-compose up -d で開発環境が立ち上がります。
プロジェクトのテンプレートにするなりご自由に。

App Engine + Datastore の構成

非常に安価でのサーバー運用が可能。
第一世代から第二世代の移行はちょっとつらい。
今からやるなら第二世代の Go 1.12 を使うと良さそう。


应用引擎琐事

如果只对本地开发环境感兴趣的读者可以略过本章节。

为什么要使用 App Engine?

2011年ごろから自前のブログシステムを App Engine(Python) 上に構築して運用しています。
サーバレスでメンテいらず、とか急なアクセス増でも対応しやすい、などなどメリットはありますが、一番の利点は 運用コストの低さ です。
App Engine はサーバレス構成が一般的になる前から従量課金方式でサーバーを提供していて、用途によっては月額のレンタルサーバを借りるより安く利用することができました。学生の頃から活用していたのでランニングコストを抑えるかは重要でした。

image.png

自分のブログ の運用にかかっている費用推移なのですが、だいたい 2000PV/月 のリクエストをさばいて数十円/月程度に抑えられています。
レンタルブログサービスでも広告消すだけで数百円掛かってしまうので圧倒的なコスパですね。
PVや機能によっては App Engine が提供する 無料枠内での運用 も可能だったりします。

只需要一个选项。

Webサーバーリソース (App Engine)

データベース (Datastore)

ドメイン + SSL (https://〇〇.appspot.com)

が含まれています。
サブドメインとはいえドメインが含まれてるのが強いですね。
Google Cloud Run や Amazon API Gateway が吐き出す機械的なドメインよりも整ったドメイン名を使うことができます。
appspot.com というルートドメインも個人の成果物のアプリケーションが詰まってる感じがして悪くないです。

如果你想以低廉的成本运营博客,我强烈推荐考虑使用App Engine。
通过自己搭建博客,你也能自然地掌握网页应用程序的知识。

云数据存储(Firestore数据存储模式)

App Engine を利用する上で欠かせないのが Datastore と呼ばれる NoSQL データベースです。
Cloud SQL などを使って RDB と接続することもできますが、いつアクセスされるかわからない状況で常時インスタンスを起動して置かなければならないのは費用的に不利です。
Datastore であれば読み書きのオペレーションとストレージのみの課金額で済みます。

ちなみに Datastore 、昔は App Engine 専用のデータベースだったのですが、データベースサービスとして独立し、最近では Cloud Firestore に統合されてしまいました。

image.png

GCP 上でプロジェクトを立ち上げた際は Cloud Firestore をネイティブモードで動かすか、Datastoreモードで動かすか聞かれます。この選択肢は排他的なので一度設定してしまうと変更することができません。
すごく雑にいえばユーザーが直接データベースを参照することがなければ Datastore モードを使えばよさそうです。ドキュメント単位のセキュリティルールが使えない代わりに性能面でメリットがあります。

应用引擎运行时环境的变化

長く運用していると App Engine の実行環境の変化にも対応しなくてはなりません。
サーバレスだからといってメンテなしでずっと動くわけではないようです。
自分が経験したところでいうと Python 2.5 の環境が停止したり、デプロイのコマンドやオプションが変わったりということが時々発生しています。先日もデプロイを実行しようとしたら10,000ファイルアップロード上限が追加されてて焦りました…。

そして Python2 系は Python 自体のサポートが今年いっぱいで終了してしまいます。
これに対する App Engine 側からのアナウンスが見当たらないですが、たとえ利用できても使わない方が賢明です。

となると Python 3系へ乗り換える必要があるのですが、Python 3.7 のランタイムは App Engine の 第二世代 と呼ばれる環境で実行されるようになります。
つまり 仕様がガラッと変わってしまう わけですね。
今までは App Engine 用にライブラリや実行環境がカスタマイズされていたのですが、第二世代は素の Python に近い形で動くようになります。
App Engine に依存しない状態で開発できるようになるため、Dockerや別のシステムへの移植性があがり嬉しい反面、過去の App Engine のソースを移行しようとするとつらみがあります。
個人的には Memcache と呼ばれるメモリキャッシュ機能が消えてしまうのが厳しいですね。
第一世代のランタイムであればキャッシュ用のサービスが組み込まれてた安価に利用できたのですが、第二世代では SDK 側に組み込まれていないので Cloud Memorystore などのサービスを利用する必要が出てきました。
費用抑えて運用するためにはしばらくはキャッシュ機構持たないほうが良さそうですね…。
一応、App Engine のプログラムが呼ばれたとき、しばらくはインスタンスが生存して使いまわしが行われるので、整合性が不要なキャッシュ目的であればグローバル変数(リクエストのライフサイクルの外側)に入れておくことでキャッシュすることが可能です。

前往 Go 的运行时系统

因为最近正在学习Go语言,所以我考虑如果重写的话,不要用Python,而是用Go来写。
它的编译速度很快,生成的构建结果也很小,所以在服务器端使用很方便。虽然表达能力有点弱,有时候描述会比较困难,但它很容易上手,逐渐变得熟悉。

关于 Go 的运行时环境,第一代和第二代的交替已经发生,1.9 和 1.11 是第一代,1.12 是第二代。虽然 1.9 和 1.11 是同一代,但内部结构相当不同,所以在参考文档时需要注意使用哪个版本的运行时环境。

虽然 Go 1.11 并不像 Python 一样即将到达语言本身的消耗期限,所以在短期内应该还可以使用,但考虑到以后会逐渐被第二代取代,如果现在开始,可能只能选择 1.12。

建立开发环境

image.png

所以,回到正题。

使用Docker-compose来设置App Engine + Go 1.12 + Datastore的本地开发环境。
旧版App Engine包含了Datastore的模拟器和管理界面等功能,可以通过专用命令来使用。然而,自从它被整合到gcloud命令中后,只支持部署功能而已。
幸运的是,Datastore的模拟器本身作为gcloud命令的Beta版已经准备好了,我们将使用它来创建开发环境。

我想要实现的目标

    1. Datastore のエミュレータを起動する

 

    1. Go 1.12 の開発用サーバーを立ち上げる

 

    1. Go から Datastore のエミュレータへアクセスできるようにする

 

    アプリケーションから読み書きができるか確認

ローカルの開発環境としてやりたいことはこの4つです。
これらが docker-compose up -d でシュッと起動できれば開発が捗りそうですね。

让我们一点点看看需要什么。

启动 Datastore 的模拟器。

bash gcloud beta emulators datastore start

只要能使用gcloud命令,就可以使用这个命令启动模拟器。
由于官方提供了google/cloud-sdk的Docker,所以使用它应该很容易。

2. Go 1.12 の開発用サーバーを立ち上げる

Go 1.12 のランタイムでは普通の Webアプリケーションと変わらない形で Go のアプリケーションを作れます。
なので標準の net/http パッケージを使ったり、外部のフレームワークを使ったりしてアプリケーションを組めます。
今回は Gin というフレームワークを使ってみることにしました。
一点注意するところは PORT という環境変数を読み取って、その ポート番号でサーバーを立ち上げる必要がある点ですね。
CloudRun でも同じ仕様なので GCP 上にアプリケーション立ち上げる際はこの仕様にしておくと便利そうです。

将Go程序配置为可以访问Datastore模拟器。

立ち上げた Web サーバーでエミュレータの Datastore を見に行くように修正をします。
Google の SDK を使う場合、環境変数に DATASTORE_EMULATOR_HOST DATASTORE_PROJECT_ID の2つを渡してあげればデータストアへの向き先を変更することができる みたいです。
アプリケーション側の変更がいらないので本番にデプロイするときも簡単ですね。
ここで設定する プロジェクトID はローカルで開発する場合は、エミュレータの起動オプションとして渡したプロジェクトIDと揃えれば何でも良さそうです。

4. アプリケーションから読み書きができるか確認

    ctx := context.Background()

    projectID := os.Getenv(EnvKeyDatastoreProjectId)
    client, err = datastore.NewClient(ctx, projectID)

    if err != nil {
            // エラー処理
    }

    // 最新10件取得
    q := datastore.NewQuery("Entry").Order("-created_at").Limit(10)

アプリケーションからはこんな形で使えば取得できるみたいです。
詳細はサンプルソース を参照してください。
環境変数で SDK にプロジェクトID を渡してますが、プログラムからも再度プロジェクトIDを渡す必要がありました。

将其整合为Docker Compose

既有所需的信息,我们可以使用Docker Compose整合开发环境。


version: '3'
services:
  env:
    image: golang:1.12
    environment:
      - GO111MODULE=on
      - DATASTORE_EMULATOR_HOST=datastore:8059  # datastore サービスの `$ホスト名:$ポート` を指定
      - DATASTORE_PROJECT_ID=local-app  # 開発環境用のプロジェクト名
      - PORT=8000
    working_dir: /opt/app
    volumes:
      - .:/opt/app  # カレントディレクトリを同期
    ports:
      - 8000:8000  # ホストマシンから localhost:8000 でアクセスできるように
    links:
      - datastore:datastore  # datastore というホスト名で datastore サービスへアクセスできるように
    command: bash -c "go get github.com/pilu/fresh && fresh -c .fresh.conf"  # ホットリロードできるように Fresh を導入

  datastore:
    image: google/cloud-sdk
    environment:
      - PROJECT_ID=local-app  # 開発環境用のプロジェクト名 (上と同じであれば何でも可)
    command: /bin/bash -c "gcloud beta emulators datastore start --host-port 0.0.0.0:8059 --project $$PROJECT_ID --data-dir /data"
    volumes:
      - datastore_volume:/data
volumes:
  datastore_volume:  # 永続化用にボリュームを定義

这就是已经完成的 docker-compose.yml 文件。

我正在定义两个服务,一个是名为env的服务,用于承载Go应用程序,另一个是名为datastore的服务,用于运行Datastore模拟器。

由于每次重新启动Docker Compose时,数据存储服务中的数据丢失会造成困扰,因此我已定义了一个专用volume,并指定将数据写入该volume中。项目ID使用临时标识为local-app。

Go のアプリケーション側からは links 機能を使ってエミュレータにアクセスできるように内部用のホスト名を設定してます。先程の環境変数にこのホスト名を渡してあげれば通信することが可能です。
あとはソース変更のたびにアプリケーションを再起動し直すのは手間なので、Fresh というホットリロードの仕組みを導入してます。

これで docker-compose up -d を実行すれば開発環境がシュッと立ち上がる仕組みが完成しました。
App Engine への実際のデプロイも gcloud app deploy コマンドを叩けばそのまま実行できます。
(DATASTORE_PROJECT_ID を本番環境でも設定してあげる必要があります。)

2019/11/06 追記:
https://cloud.google.com/datastore/docs/activate?hl=ja#datastore-permissions-for-app-engine
App Engine から Datastore へアクセスする Permission の設定も必要でした。


就差要大力地写应用程序了。请一定要试试看使用App Engine,它很有趣!(必须在年底前重新设计博客…)

更新:2019年11月07日

更新App Engine以支持更多新的运行时环境:Node.js 12、Go 1.13、PHP 7.3和Python 3.8。看起来Go 1.13的运行时环境已经发布了测试版。在语言上进行了更新,但在规格上变化似乎不是很大吧。

请参考

    • Cloud Datastore エミュレータの実行

 

    • GAE第二世代+Go v1.12でechoサーバーを構築する(2019年度版)

 

    GAE/SE Pythonランタイムアップデートへの道
截至2019年11月5日
广告
将在 10 秒后关闭
bannerAds