はじめに
会社やプロジェクトによっては、各種ファイルの扱いにルールが設けられている場合があるかと思います。ファイル管理にGoogle ドライブを使っている場合、Google Apps Scriptを使用することでこれらのルールを自動的にチェックし、ファイルを適正管理することができます。
今回は、
- ファイルの命名規則が正しいかを自動チェックし、
- 不正なファイルは自動削除後、管理者に通知する
スクリプトを組んでみます。これにより、
- 「案件A」ドライブには、「案件A」をファイル名に含むファイルのみが格納されるようになり、
- 関係のないファイルが混在することを防ぐ
ような使い方もできます。
参照
参考記事
本記事は下記3つの記事を組み合わせています。
【ソースコード付き】GoogleDriveでファイルが追加・更新されたらメールで通知する
Google Apps Scriptでファイルフォルダの探索
GASを使って、スプレッドシート更新時にSlack通知を飛ばしてくれるbotを作ってみた
(念の為)Googleドライブ/GAS(Google Apps Scrip)とは
やりたいこと
- ①ユーザがある共有ドライブに、プロジェクトで設けられた命名規則に則っていない資料を格納したら、
- ②自動で検知して、
- ③そのファイルを削除する。
- ④正しく格納されたら、それを通知する。
やったこと(ざっくり)
- フォルダ内のファイル情報の検索をしたら、配列に保存する
- 親フォルダの名称にある企業名と、取得したファイル名にある企業名を突き合わせて、異なっていたら削除する
- スプレッドシートの情報を取得して、その情報と突き合わせて新しく格納されたファイルがあるか確認し、あれば通知する
※下記の前提条件を置きます
- 親フォルダ名に企業名、格納するファイル名にも企業名を記載する、という命名規則を置くこと
やったこと(詳細)
トリガーの設定
理想としてはファイルが格納されたタイミングで親フォルダ名とスプシ情報との突合を行いたかったのですが、ファイルの格納を検知する方法が思いつかず、今回はトリガーを利用して実現しました。
出来る限りファイル格納をリアルタイムで検知するため、トリガータイプは分ベースのタイマーとし、1分おきにスクリプトが走るようにしています。
スクリプトの処理フロー
設定したトリガーが起動するスクリプトは下記フローで処理を行います。
- フォルダID、スプレッドシート(以下スプシ)ID、スプシシート名、Slack通知用URLを取得(今回はスクリプトに直接記入)
- 共有フォルダ内にあるフォルダ情報を取得する
- 各フォルダ直下にあるファイル情報を取得する
- 親フォルダ名とファイル名を突合し、異なる名称であればファイルを削除し、任意のSlackチャンネルに通知する
- スプシに記載されているファイル情報を取得する
- フォルダ内にあるファイル名と、スプシに記載されているファイル名を比較する
- 既に記載されている場合、何もしない
- フォルダにあってスプシにない場合、格納した旨を任意のSlackチャンネルに通知する
- スプシにあってフォルダにない場合、スプシから削除する
フォルダの階層・スプシの配置はこんな感じです。
- 親フォルダ名→このフォルダ名に記載の企業名を利用して4.の処理を行います。
- スプシはどこでも良いですが、今回はとりあえず親フォルダ直下へ
- 他社の情報を削除しないよう、明示的に自社/他社フォルダを分ける
- スプシのファイル名やシート名は何でも良い
その他
スプレッドシートからスクリプトの作成・実行の方法、トリガーの設定方法などはこちらをご参考ください。
処理の詳細
1. フォルダID、スプレッドシート(以下スプシ)ID、スプシシート名、Slack通知用URLを取得(今回はスクリプトに直接記入)
//Slack通知
var url = "https://hooks.slack.com/services/xxxxxx"; //xxxxxxには任意のIDを記入
//対象とするGoogleDriveフォルダのID
var FOLDER_ID = "xxxxxxx"; //xxxxxxには任意のIDを記入
//更新日時を記録するのスプレッドシートのID
var UPDATE_SHEET_ID = "xxxxxx"; //xxxxxxには任意のIDを記入
//スプレッドシートのシート名
var UPDATE_SHEET_NAME = "xxxxxx";//xxxxxxには任意のシート名を記入
GoogleDriveフォルダのIDは任意のGoogleドライブのURLのfolders/
以降の文字列です。
スプレッドシートのIDはhttps://docs.google.com/spreadsheets/d/
ここ/edit#gid=0
です。
スプレッドシートのシート名は
ここです。
2. 共有フォルダ内にあるフォルダ情報を取得する
// スプレッドシートに記載されているフォルダ名と更新日時を取得。
var spreadsheet = SpreadsheetApp.openById(UPDATE_SHEET_ID);
var sheet = spreadsheet.getSheetByName(UPDATE_SHEET_NAME);
//Drive APIで特定のファイルをリストとして取得
var key = FOLDER_ID;
var nowdir = key;
var totalcount=0;
var totalFilesCount=0;
//folderlistシートに移動し内容をクリアする
var Cell = sheet.getRange("A2");//offsetの参照点セルを設定
Cell.activate();
//フラグ類
var subfolderflag = 0; //1なら有り、0なら無し。
//親フォルダ(スプシが格納されているフォルダ)名を取得
var psheetfolder = DriveApp.getFileById(UPDATE_SHEET_ID).getParents();
var sfolder = psheetfolder.next();
let splitTextpFolder = sfolder.getName().split("_");
var companyName = splitTextpFolder[0];
//フォルダがあるか検索し、あった場合にはフォルダリスト配列に格納する
//フォルダ直下のファイル情報を配列に格納する。totalcountを利用して配列番号を更新する。
var folderlist = new Array();
var folders = DriveApp.searchFolders("'"+key+"' in parents");
var myFile = {};
3. 各フォルダ直下にあるファイル情報を取得する
4. 親フォルダ名とファイル名を突合し、異なる名称であればファイルを削除し、任意のSlackチャンネルに通知する
情報取得しながらフォルダ名とファイル名の突合を行います。突合し誤っていればその時点で削除&通知を送ります。
//フォルダ直下にファイルがあるか確認する
var folderDirect = DriveApp.getFolderById(FOLDER_ID); //フォルダIDなので、このフォルダ名自体は別途取得が必要
var filesDirect = folderDirect.getFiles(); //あくまでfolder直下にあるファイルのみ探索
while(filesDirect.hasNext()){
var fileDirect = filesDirect.next();
let splitTextFile = fileDirect.getName().split("_");
if(splitTextFile[0] == companyName){
myFile[totalFilesCount] = {
name: fileDirect.getName(),
lastUpdate: fileDirect.getLastUpdated(),
url: fileDirect.getUrl(),
diff: 0
}
totalFilesCount++;
}else{
fileDirect.setTrashed(true);
var options = {
'method': 'post',
'headers': {'Content-type': 'application/json'},
'payload' : JSON.stringify({
'channel' : 'xxxxxx', //xxxxxxには任意のSlackチャンネル名を記入
'attachments':[
{
'fallback': 'Googleドライブへ格納されたファイルが削除されました',
'color': '#36a64f',
'title': 'Gドライブファイル削除',
'title_link': 'https://docs.google.com/spreadsheets/d/xxxxxx/edit#gid=0', //xxxxxxには任意のスプシIDを記入
'text': 'test',
}
]
})
};
UrlFetchApp.fetch(url, options);
}
}
if(folders.length == 0){
//普通に指定フォルダ内のファイルのリストを書き出して終了
}else{
//サブフォルダフラグを立てる
subfolderflag=1;
//folderlistシートへ書き込み
while(folders.hasNext()){
var folder = folders.next();
folderlist.push(folder.getId());
var files= folder.getFiles(); //あくまでfolder直下にあるファイルのみ探索
while(files.hasNext()){
var file = files.next();
let splitTextFile = file.getName().split("_");
if(splitTextFile[0] == companyName){
myFile[totalFilesCount] = {
name: file.getName(),
lastUpdate: file.getLastUpdated(),
url: file.getUrl(),
diff: 0
}
totalFilesCount++;
}else{
file.setTrashed(true);
var options = {
'method': 'post',
'headers': {'Content-type': 'application/json'},
'payload' : JSON.stringify({
'channel' : 'xxxxxx', //xxxxxxには任意のSlackチャンネル名を記入
'attachments':[
{
'fallback': 'Googleドライブへ格納されたファイルが削除されました',
'color': '#36a64f',
'title': 'Gドライブファイル削除',
'title_link': 'https://docs.google.com/spreadsheets/d/xxxxxx/edit#gid=0', //xxxxxxには任意のスプシIDを記入
'text': 'test',
}
]
})
};
UrlFetchApp.fetch(url, options);
}
}
totalcount++;
}
}
//サブフォルダ以降のフォルダの探索ルーチン
var sub = 1;
while (sub == 1) {
var nowponyo = folderlist.length;
for(var i = 0;i<nowponyo;i++){
//0番目の配列のkeyを取得する
key = folderlist[0];
//そのkeyのフォルダにぶら下がるフォルダをリストアップ
folders = DriveApp.searchFolders("'"+key+"' in parents");
nowdir = key;
//folderlistシートへ書き込み
if(folders.hasNext()==true){
while(folders.hasNext()){
var folder = folders.next();
folderlist.push(folder.getId());
totalcount++;
var files= folder.getFiles(); //あくまでfolder直下にあるファイルのみ探索
while(files.hasNext()){
var file = files.next();
let splitTextFile = file.getName().split("_");
if(splitTextFile[0] == companyName){
myFile[totalFilesCount] = {
name: file.getName(),
lastUpdate: file.getLastUpdated(),
url: file.getUrl(),
diff: 0
}
totalFilesCount++;
}else{
file.setTrashed(true);
var options = {
'method': 'post',
'headers': {'Content-type': 'application/json'},
'payload' : JSON.stringify({
'channel' : 'xxxxxx', //xxxxxxには任意のSlackチャンネル名を記入
'attachments':[
{
'fallback': 'Googleドライブへ格納されたファイルが削除されました',
'color': '#36a64f',
'title': 'Gドライブファイル削除',
'title_link': 'https://docs.google.com/spreadsheets/d/xxxxxx/edit#gid=0', //xxxxxxには任意のスプシIDを記入
'text': 'test',
}
]
})
};
UrlFetchApp.fetch(url, options);
}
}
}
}
//配列0番目を削除と詰める
folderlist.splice(0,1);
}
//folderlistが空ならsubfolderflagを0にする
if(folderlist.length == 0){
sub=0;
}
}
5. スプシに記載されているファイル情報を取得する
//スプシに記載されているファイル名を取得する 取得したデータをMapに変換。
var data = sheet.getDataRange().getValues();
var sheetData = {};
// headerがあるので2から開始
for (var i = 1; i < data.length; i++) {
sheetData[data[i][0]] = {
name: data[i][0],
lastUpdate: data[i][1],
url: data[i][2],
rowNo: i + 1
};
}
6. フォルダ内にあるファイル名と、スプシに記載されているファイル名を比較する
// 実際のファイルとスプレッドシート情報を比較。
var updateFileList = [];
for (key in myFile) {
if (sheetData[myFile[key].name]){
var test = sheetData[myFile[key].name].name;
if (test == myFile[key].name) {
// ファイル名がシートに存在する場合。特に何もしない
}
} else {
// ファイル名がシートに存在しない場合。
var lowno = sheet.getLastRow() + 1;
sheet.getRange(lowno, 1).setValue(myFile[key].name);
sheet.getRange(lowno, 2).setValue(myFile[key].lastUpdate);
sheet.getRange(lowno, 3).setValue(myFile[key].url);
updateFileList.push(key);
var options = {
'method': 'post',
'headers': {'Content-type': 'application/json'},
'payload' : JSON.stringify({
'channel' : 'xxxxxx', //xxxxxxには任意のSlackチャンネル名を記入
'attachments':[
{
'fallback': 'Googleドライブへファイルが格納されました',
'color': '#36a64f',
'title': 'Gドライブファイル格納',
'title_link': 'https://docs.google.com/spreadsheets/d/xxxxxx/edit#gid=0', //xxxxxxには任意のスプシIDを記入
'text': 'test',
}
]
})
};
UrlFetchApp.fetch(url, options);
}
}
// 削除されたファイルをチェックして、スプシから削除
var deleteFileList = [];
var count =0;
for (key in sheetData){
for (var i=0; i < totalFilesCount; i++){
if(key == myFile[i].name){
count++;
}
}
if (count == 0){
sheet.deleteRow(sheetData[key].rowNo)
deleteFileList.push(key);
}
count = 0;
}
//終了処理
SpreadsheetApp.flush();
ソースコード
function driveSearchPutArray(){
//Slack通知
var url = "https://hooks.slack.com/services/xxxxxx"; //xxxxxxには任意のIDを記入
//対象とするGoogleDriveフォルダのID
var FOLDER_ID = "xxxxxxx"; //xxxxxxには任意のIDを記入
//更新日時を記録するのスプレッドシートのID
var UPDATE_SHEET_ID = "xxxxxx"; //xxxxxxには任意のIDを記入
//スプレッドシートのシート名
var UPDATE_SHEET_NAME = "xxxxxx"; //xxxxxxには任意のシート名を記入
// スプレッドシートに記載されているフォルダ名と更新日時を取得。
var spreadsheet = SpreadsheetApp.openById(UPDATE_SHEET_ID);
var sheet = spreadsheet.getSheetByName(UPDATE_SHEET_NAME);
//Drive APIで特定のファイルをリストとして取得
var key = FOLDER_ID;
var nowdir = key;
var totalcount=0;
var totalFilesCount=0;
//folderlistシートに移動し内容をクリアする
var Cell = sheet.getRange("A2");//offsetの参照点セルを設定
Cell.activate();
//フラグ類
var subfolderflag = 0; //1なら有り、0なら無し。
//親フォルダ(スプシが格納されているフォルダ)名を取得
var psheetfolder = DriveApp.getFileById(UPDATE_SHEET_ID).getParents();
var sfolder = psheetfolder.next();
let splitTextpFolder = sfolder.getName().split("_");
var companyName = splitTextpFolder[0];
//フォルダがあるか検索し、あった場合にはフォルダリスト配列に格納する
//フォルダ直下のファイル情報を配列に格納する。totalcountを利用して配列番号を更新する。
var folderlist = new Array();
var folders = DriveApp.searchFolders("'"+key+"' in parents");
var myFile = {};
//フォルダ直下にファイルがあるか確認する
var folderDirect = DriveApp.getFolderById(FOLDER_ID); //フォルダIDなので、このフォルダ名自体は別途取得が必要
var filesDirect = folderDirect.getFiles(); //あくまでfolder直下にあるファイルのみ探索
while(filesDirect.hasNext()){
var fileDirect = filesDirect.next();
let splitTextFile = fileDirect.getName().split("_");
if(splitTextFile[0] == companyName){
myFile[totalFilesCount] = {
name: fileDirect.getName(),
lastUpdate: fileDirect.getLastUpdated(),
url: fileDirect.getUrl(),
diff: 0
}
totalFilesCount++;
}else{
fileDirect.setTrashed(true);
var options = {
'method': 'post',
'headers': {'Content-type': 'application/json'},
'payload' : JSON.stringify({
'channel' : 'xxxxxx', //xxxxxxには任意のSlackチャンネル名を記入
'attachments':[
{
'fallback': 'Googleドライブへ格納されたファイルが削除されました',
'color': '#36a64f',
'title': 'Gドライブファイル削除',
'title_link': 'https://docs.google.com/spreadsheets/d/xxxxxx/edit#gid=0', //xxxxxxには任意のスプシIDを記入
'text': 'test',
}
]
})
};
UrlFetchApp.fetch(url, options);
}
}
if(folders.length == 0){
//普通に指定フォルダ内のファイルのリストを書き出して終了
}else{
//サブフォルダフラグを立てる
subfolderflag=1;
//folderlistシートへ書き込み
while(folders.hasNext()){
var folder = folders.next();
folderlist.push(folder.getId());
var files= folder.getFiles(); //あくまでfolder直下にあるファイルのみ探索
while(files.hasNext()){
var file = files.next();
let splitTextFile = file.getName().split("_");
if(splitTextFile[0] == companyName){
myFile[totalFilesCount] = {
name: file.getName(),
lastUpdate: file.getLastUpdated(),
url: file.getUrl(),
diff: 0
}
totalFilesCount++;
}else{
file.setTrashed(true);
var options = {
'method': 'post',
'headers': {'Content-type': 'application/json'},
'payload' : JSON.stringify({
'channel' : 'xxxxxx', //xxxxxxには任意のSlackチャンネル名を記入
'attachments':[
{
'fallback': 'Googleドライブへ格納されたファイルが削除されました',
'color': '#36a64f',
'title': 'Gドライブファイル削除',
'title_link': 'https://docs.google.com/spreadsheets/d/xxxxxx/edit#gid=0', //xxxxxxには任意のスプシIDを記入
'text': 'test',
}
]
})
};
UrlFetchApp.fetch(url, options);
}
}
totalcount++;
}
}
//サブフォルダ以降のフォルダの探索ルーチン
var sub = 1;
while (sub == 1) {
var nowponyo = folderlist.length;
for(var i = 0;i<nowponyo;i++){
//0番目の配列のkeyを取得する
key = folderlist[0];
//そのkeyのフォルダにぶら下がるフォルダをリストアップ
folders = DriveApp.searchFolders("'"+key+"' in parents");
nowdir = key;
//folderlistシートへ書き込み
if(folders.hasNext()==true){
while(folders.hasNext()){
var folder = folders.next();
folderlist.push(folder.getId());
totalcount++;
var files= folder.getFiles(); //あくまでfolder直下にあるファイルのみ探索
while(files.hasNext()){
var file = files.next();
let splitTextFile = file.getName().split("_");
if(splitTextFile[0] == companyName){
myFile[totalFilesCount] = {
name: file.getName(),
lastUpdate: file.getLastUpdated(),
url: file.getUrl(),
diff: 0
}
totalFilesCount++;
}else{
file.setTrashed(true);
var options = {
'method': 'post',
'headers': {'Content-type': 'application/json'},
'payload' : JSON.stringify({
'channel' : 'xxxxxx', //xxxxxxには任意のSlackチャンネル名を記入
'attachments':[
{
'fallback': 'Googleドライブへ格納されたファイルが削除されました',
'color': '#36a64f',
'title': 'Gドライブファイル削除',
'title_link': 'https://docs.google.com/spreadsheets/d/xxxxxx/edit#gid=0', //xxxxxxには任意のスプシIDを記入
'text': 'test',
}
]
})
};
UrlFetchApp.fetch(url, options);
}
}
}
}
//配列0番目を削除と詰める
folderlist.splice(0,1);
}
//folderlistが空ならsubfolderflagを0にする
if(folderlist.length == 0){
sub=0;
}
}
//スプシに記載されているファイル名を取得する 取得したデータをMapに変換。
var data = sheet.getDataRange().getValues();
var sheetData = {};
// headerがあるので2から開始
for (var i = 1; i < data.length; i++) {
sheetData[data[i][0]] = {
name: data[i][0],
lastUpdate: data[i][1],
url: data[i][2],
rowNo: i + 1
};
}
// 実際のファイルとスプレッドシート情報を比較。
var updateFileList = [];
for (key in myFile) {
if (sheetData[myFile[key].name]){
var test = sheetData[myFile[key].name].name;
if (test == myFile[key].name) {
// ファイル名がシートに存在する場合。特に何もしない
}
} else {
// ファイル名がシートに存在しない場合。
var lowno = sheet.getLastRow() + 1;
sheet.getRange(lowno, 1).setValue(myFile[key].name);
sheet.getRange(lowno, 2).setValue(myFile[key].lastUpdate);
sheet.getRange(lowno, 3).setValue(myFile[key].url);
updateFileList.push(key);
var options = {
'method': 'post',
'headers': {'Content-type': 'application/json'},
'payload' : JSON.stringify({
'channel' : '#p_test-asai',
'attachments':[
{
'fallback': 'Googleドライブへファイルが格納されました',
'color': '#36a64f',
'title': 'Gドライブファイル格納',
'title_link': 'https://docs.google.com/spreadsheets/d/xxxxxx/edit#gid=0',//xxxxxxには任意のスプシIDを記入
'text': 'test',
}
]
})
};
UrlFetchApp.fetch(url, options);
}
}
// 削除されたファイルをチェックして、スプシから削除
var deleteFileList = [];
var count =0;
for (key in sheetData){
for (var i=0; i < totalFilesCount; i++){
if(key == myFile[i].name){
count++;
}
}
if (count == 0){
sheet.deleteRow(sheetData[key].rowNo)
deleteFileList.push(key);
}
count = 0;
}
//終了処理
SpreadsheetApp.flush();
}
まとめ
Googleドライブは非常に便利で、GASを使うことで要件毎に様々な自動処理をカスタマイズ出来るので、積極的に利用していければと思います。