はじめに
こんにちは。人差し指の詰めが割れ欠けてて、でも皮膚はくっついているので困っています、筆者です
さて、今回はGoogleドライブのフォルダ構造とそのフォルダに属しているファイルたちをスプレッドシートで見れるようにGASを書いていこうと思います。
参考になれば幸いです
方針
- Googleドライブのルートフォルダから所属するフォルダがある場合再帰的に掘っていく.
- フォルダ掘ると同時にファイルがあれば配列に追加していく.
- 配列の長さを調整
- ヘッダー追加
- スプレッドシート書き込み
Googleドライブのフォルダ・ファイル構造をスプレッドシートで一覧化する
スクリプトはこちらです。
const sheet = SpreadsheetApp.getActiveSheet()
const headers = ['No.', 'id', 'size', 'mimeType', 'url']
const main = () => {
let folder = DriveApp.getRootFolder(),
folder_name = folder.getName(),
data = []
data = seek(data, folder, [folder_name])
data = convert(data)
// スプレッドシートに書き込み.
sheet.getRange(1, 1, data.length, data[0].length).setValues(data)
}
const seek = (data, folder, ancestors) => {
const files = folder.getFiles()
let file
while (files.hasNext()) {
file = files.next()
// ファイル数やフォルダ数が少ない場合はここで直接スプレッドシートに書き込んでもいいけど処理遅いからおすすめしないよ.
data.push([
'=row() - 1',
file.getId(),
file.getSize(),
file.getMimeType(),
file.getUrl(),
...ancestors,
file.getName()
])
}
const folders = folder.getFolders()
let folder_descendant
while (folders.hasNext()) {
folder_descendant = folders.next()
seek(data, folder_descendant, [...ancestors, folder_descendant.getName()])
}
return data
}
const convert = data => {
// 深さの最大値を取得.
const depth_max = data.map(e => e.length).reduce((a, b) => a > b ? a : b)
// 深さを基に配列の長さを統一する.
// 配列の長さが異なると後続でスプレッドシートに値を書き込む際に怒られる...
for (let i in data) {
if (data[i].length < depth_max) {
data[i] = [
...data[i],
...[...Array(depth_max - data[i].length).keys()].map(i => '')
]
}
}
// header追加.
data = [
[
...headers,
...[...Array(depth_max - headers.length).keys()].map(i => `Layer${i + 1}`)
],
...data
]
return data
}
実行結果
所感
convert関数で処理している最後にスプレッドシートに渡す配列の長さを調整する処理ですが、内部的にかなりfor文が回っています...
ただ、data配列にpushせずそのタイミングでスプレッドシートに1行ずつ書き込むとAPIのリクエストがその数発生し、書き込み速度が遅くなるのでそれはもっとおすすめしないです。
そのため、スマートではないですが、GAS側で少し負荷掛かってますが、ご留意を
総じてこっちの方が高速処理できますので
おわりに
意外と600ファイルくらいあってビビりました。
いらないファイル消していこうと思います
それでは!