GCPインスタンスをNode.js8のCloud Pub/Subでスケジュール起動停止

2019-09-29

GCPでインスタンスをスケジューラーで自動起動・自動停止したいことがある。業務で使用している開発サーバーなどを週7日24h稼働するのはコスト的にも勿体無い。サービスによっては本番公開サーバーであっても稼働する必要がない時間帯があることもあるかもしれない。

GCEで自動起動・自動停止をする場合、Cloud functions(Cloud Pub/Sub)とCloud Schedulerを使用する。
Cloud Scheduler を使用した Compute インスタンスのスケジュール設定

基本的にはここに書かれている通りにすればインスタンスのスケジューリング起動停止ができる。

Nodejs8でのCloud Scheduler & Cloud Pub/Sub

上記URLに書かれている通りにすると正常に動作するのだが「Node6」での話だ。(このリファレンスにもNode6を選択するようにと書かれている)

しかし、Cloud functionsでNode6を選択しようとするとGoogleさんが以下のように注意してきます。

gcp-function

つまり、今時、Node.js6でデプロイするなということですな。なので、私はNode.js8でこのインスタンス起動停止スクリプトをデプロイした。

ところがなんとNode8ではこの起動停止スクリプトが動作しない。ログを見てみたら以下のエラーがキャッチされている。

	TypeError: First argument must be a string, Buffer, ArrayBuffer, Array, or array-like object. 
	at Function.Buffer.from (buffer.js:183:11) at exports.startInstancePubSub (/srv/index.js:20:25) at /worker/worker.js:825:24 at  at process._tickDomainCallback (internal/process/next_tick.js:229:7)

GCPの公式リファレンスの通りにしてもNode8だとこのエラーが出てしまうのだ。デバッグしてみたらpubsubMessage.dataがundefindになっていた。

	const pubsubMessage = event.data;
    const payload = _validatePayload(
      // pubsubMessage.data が undefind!
      JSON.parse(Buffer.from(pubsubMessage.data, 'base64').toString())
    );

なぜこうなるのかわからないが、以下のようにするとeventのmessageを展開でき、正常に動くようだ。

	const pubsubMessage = event.data;
    const payload = _validatePayload(
      JSON.parse(Buffer.from(pubsubMessage, 'base64').toString())
    );

だがだがしかし、これ正常に動作しながらも以下のエラーをはいているようだ。

	TypeError: callback is not a function at compute.zone.vm.stop.then.then.catch.err
	 (/srv/index.js:39:9) at  at process._tickDomainCallback (internal/process/next_tick.js:229:7)

正常にインスタンスを起動 or 停止した後にfunctionの完了のcallbackでこけているようだ。解決策は「GCPインスタンスをNode.js8のCloud Pub/Subでスケジュール起動停止(2)

というかGoogleさんがこの公式リファレンスのコードを修正して欲しいです。Node8では明らかに動作しない。