LoginSignup
1
2

More than 1 year has passed since last update.

【GAS】毎月XX日(土日祝日の場合は前倒し)にスクリプトを実行する

Posted at

はじめに

職場の人に頼まれてGoogle Drive操作の自動化プログラムをGASで作成した際,定期実行のタイミングが「毎月20日(ただし20日が土日祝日の場合は20日以前の最終営業日)の午前9~10時」という複雑なもので,単純にトリガーを設定するだけでは要件を満たせなかったため,少し工夫しました。

Python&crontabでも同じようなことをやっているので,よかったら下の記事もご覧ください。

毎月XX日(土日祝日の場合は前倒し)にcronジョブを実行する - Qiita

やり方

GASのトリガーを「毎日午前9~10時」で作成し,エントリーポイントに設定するtrigger関数でその日が定期実行実施日かどうか判定します。判定の仕方は以下の通りです。

  1. 当月20日が営業日であれば,20日を実施日とする
  2. 営業日でなければ1日ずつ遡り,最初に出現した営業日を実施日とする
  3. スクリプトを実行した日付と実施日の日付を比較する
  4. 両者が一致すればmain関数を呼び出す
  5. 一致しなければ何もせずに終了する

コード

function trigger() {
  var base = new Date(); // インクリメント用Dateオブジェクト
  var fixed; // 算出した実施日のDateオブジェクトを格納する変数
  base.setDate(20);
  if (isHoliday(base)) {
    while (true) {
      base.setDate(base.getDate() - 1);
      if (!(isHoliday(base))) {
        break
      }
    }
  }
  fixed = base;

  var today = new Date();
  if (today.getDate() == fixed.getDate()) {
    main();
  }
}

function isHoliday(day){
  //土日か判定
  var weekInt = day.getDay();
  if(weekInt <= 0 || 6 <= weekInt){
    return true;
  }

  //祝日か判定
  var calendarId = "ja.japanese#holiday@group.v.calendar.google.com";
  var calendar = CalendarApp.getCalendarById(calendarId);
  var events = calendar.getEventsForDay(day);
  if(events.length > 0){
    return true;
  }

  return false;
}

function main() {
  // 省略
}

whileループは1日から20日まで全て休日だった場合エラーを吐いてしまいそうですが,流石にそんなことは起こらないと思うのでご容赦ください。

土日祝日を判定するコードは以下のサイトで紹介されているものを利用させていただいたので,ここでの解説は省略させていただきます。

【GAS】土日・祝日・特定休日を判定する|もりさんのプログラミング手帳

別解

今回はGASなので,特にリソース消費を気にせず毎日トリガー実行でも問題ないと思いますが,前回と同様なるべくスクリプトの実行回数を減らそうとしたは以下のような手順になります。

  1. (手動)毎月1日にsetTrigger関数を実行するトリガーを作成する
  2. setTriggermain関数を実行する既存のトリガーを削除
  3. setTriggermain関数の実行日を算出する
  4. setTrigger)算出した日付でmain関数を実行するトリガーを作成

コード

function setTrigger() {
  delTrigger();

  var executionDate = new Date();
  // 挙動確認用
  // executionDate.setMonth(10);
  executionDate.setDate(20);
  if (isHoliday(executionDate)) {
    while (true) {
      executionDate.setDate(executionDate.getDate() - 1);
      if (!(isHoliday(executionDate))) {
        break
      }
    }
  }
  executionDate.setHours(9);
  executionDate.setMinutes(0);
  executionDate.setSeconds(0);

  ScriptApp.newTrigger('main').timeBased().at(executionDate).create();
}

// 既存のトリガーを削除
function delTrigger() {
  const triggers = ScriptApp.getProjectTriggers();
  for(const trigger of triggers){
    if(trigger.getHandlerFunction() == "main"){
      ScriptApp.deleteTrigger(trigger);
    }
  }
}

function isHoliday(day){
  // 省略
}

function main() {
  // 省略
}

delTrigger関数は以下のサイトで紹介されているものをお借りしました。

Google Apps Scriptで呼び出したトリガーを削除する方法

試しに2022年10月の設定でsetTriggerを実行すると,
image.png
このようにトリガーが作成されました。そのまま設定を2022年11月にして再実行すると,
image.png
image.png
トリガーの総数は変わりませんが,mainを呼び出すトリガーの実行日時が変わっており,前のトリガーが削除されて新たなトリガーが作成されたことがわかります。

おわりに

運用面を考えたら,祝日の判定にGoogleカレンダーを使えるのは安心感があっていいですね。
別解の方はコード行数が増えるものの,日時を秒単位で指定するので実行タイミングがかっちり決まるのはいいかもしれません。
行数が気になる場合はsetTriggerを別のファイルやプロジェクトに切り出すのもアリですね。
今回紹介した以外にもっとスマートな解法があったら是非教えてください。

参考

【GAS】正確な時間に起動するトリガーの設定 | ネコニキの開発雑記

1
2
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
2