GASからGoogleカレンダーのmeetを取得してGeminiで議事録作成まで
public.icon
完成フロー
https://gyazo.com/6af337ba4816f9471ed963c0a03996e8
自動でドキュメントを作成して紐づけてくれる
https://gyazo.com/5539e3e708632fcc3dfc25762eb5bafa
https://gyazo.com/eda9a6e750f3e61a20ded18f47fa7586
カレンダーに紐づいたmeetで会議を行う
meet内で録画を行うと、meetのidでdriveに動画が保存される(動画がアップロードされるのには10分程度かかる)
https://gyazo.com/94578b7c2f08e9ea65f43364434c82bb
https://gyazo.com/bdefaf512dc336771b693556d71bed07
1時間に1回の定期実行で、録画を発見→文字起こし、要約→agenda docに追記
https://gyazo.com/6974408bf21e02a580ffe22364725b86
以下作業ログ
https://gyazo.com/92dd967444393dc3944579f1c3dbf94d
そのままやろうとすると、event.getHangoutLink is not a functionでそもそも会議リンクが取得できなかった
どうやら、Gas標準のカレンダー機能ではmeetのリンクは取得できない
Calendar APIを使うしかなさそう
Calendar APIの追加はサービスからCalendarを選択するだけ
Calendar APIの使い方
https://gyazo.com/8fb0fb7455529a28ee38560135119f7a
Calendar.xxxxで使うことができる
Calendar APIのリファレンスはこちら
https://gyazo.com/6eb7e3970f26db199ffc687d27a04806
https://gyazo.com/4be0342790ba3f789481c97f47ff9b83
検索の書き方を発見
そのままmeetのURLが取得できた
そのまま、動画の取得とblobの取得に成功
geminiのキーを発行
https://gyazo.com/708ee1a4e87ee18436ece62167f5505f
https://gyazo.com/5306977f2b777db48f2f03c2e69d29b6
しかし、エラー。endopointNotFoundなので多分APIの記述が間違ってる
geminiのgasの叩き方を調べる
https://gyazo.com/2118a1d8223b546bbe4dc3f9b64a32bb
あっさり動いた
多分これをベースに音声ファイルを付け加えてあげればいけるはず
https://gyazo.com/e94cf14e3b54958ba45bdabdb79ce2dc
500エラーに変わった。。
しばらくしてからもう一度お試しください。再試行しても問題が解決しない場合は、Google AI Studio の フィードバックを送信 ボタンを使用して報告してください。 試しにmp3でやってみた時は問題なく動いた。
次はfileAPI経由を試してみる
https://gyazo.com/e23557a0a1ad60e476f0b6b569164949
file経由でいけたっぽい!
gasでfileAPIにdriveのファイルをアップロードするほうほう
code:main.gs
const CHUNK_SIZE = 8 * 1024 * 1024; // 8 MiB
/**
* メイン関数:ファイルの検索とアップロードを実行
* @return {Object|null} アップロード結果のJSONオブジェクト、またはnull(失敗時)
*/
function uploadFileToGenAI() {
const apiKey = getApiKey();
if (!apiKey) return null;
const file = findFileToUpload();
if (!file) return null;
const uploadUrl = startResumableUpload(apiKey, file);
if (!uploadUrl) return null;
const uploadResult = uploadFileInChunks(file, uploadUrl);
if (uploadResult) {
Logger.log('アップロード結果:');
Logger.log(JSON.stringify(uploadResult, null, 2));
}
return uploadResult;
}
// getApiKey() と findFileToUpload() 関数は変更なし
/**
* レジューマブルアップロードを開始
*/
function startResumableUpload(apiKey, file) {
const url = ${BASE_URL}/upload/v1beta/files?key=${apiKey};
const options = {
method: 'post',
headers: {
'X-Goog-Upload-Protocol': 'resumable',
'X-Goog-Upload-Command': 'start',
'X-Goog-Upload-Header-Content-Length': file.getSize().toString(),
'X-Goog-Upload-Header-Content-Type': file.getMimeType(),
'Content-Type': 'application/json'
},
payload: JSON.stringify({ file: { display_name: file.getName() } }),
muteHttpExceptions: true
};
try {
const response = UrlFetchApp.fetch(url, options);
if (response.getResponseCode() === 200) {
Logger.log(レジューマブルアップロードURL: ${uploadUrl});
return uploadUrl;
} else {
Logger.log(エラー: レジューマブルアップロードの開始に失敗しました。ステータスコード: ${response.getResponseCode()});
Logger.log(レスポンス: ${response.getContentText()});
}
} catch (e) {
Logger.log(例外が発生しました: ${e.toString()});
}
return null;
}
/**
* ファイルをチャンクに分けてアップロード
* @return {Object|null} アップロード結果のJSONオブジェクト、またはnull(失敗時)
*/
function uploadFileInChunks(file, uploadUrl) {
const fileSize = file.getSize();
const numChunks = Math.ceil(fileSize / CHUNK_SIZE);
const fileBlob = file.getBlob();
for (let i = 0; i < numChunks; i++) {
const offset = i * CHUNK_SIZE;
const chunkSize = Math.min(CHUNK_SIZE, fileSize - offset);
const chunkBlob = fileBlob.getBytes().slice(offset, offset + chunkSize);
const isLastChunk = (i === numChunks - 1);
Logger.log(チャンク ${i + 1}/${numChunks} をアップロード中 (${offset} - ${offset + chunkSize} / ${fileSize})...);
const chunkResult = uploadChunk(uploadUrl, chunkBlob, offset, chunkSize, fileSize, isLastChunk);
if (!chunkResult) {
Logger.log(エラー: チャンク ${i + 1}/${numChunks} のアップロードに失敗しました。);
return null;
}
if (isLastChunk) {
return chunkResult; // 最後のチャンクの結果を返す
}
}
Logger.log('アップロードが完了しましたが、最終レスポンスを取得できませんでした。');
return null;
}
/**
* 単一のチャンクをアップロード
* @return {Object|true|false} 最後のチャンクの場合はJSONオブジェクト、それ以外はboolean
*/
function uploadChunk(uploadUrl, chunkData, offset, chunkSize, totalSize, isLastChunk) {
const chunkBlob = Utilities.newBlob(chunkData, 'application/octet-stream');
const options = {
method: 'put',
headers: {
'X-Goog-Upload-Offset': offset.toString(),
'X-Goog-Upload-Command': isLastChunk ? 'upload, finalize' : 'upload',
'Content-Range': bytes ${offset}-${offset + chunkSize - 1}/${totalSize}
},
payload: chunkBlob,
contentLength: chunkSize,
muteHttpExceptions: true
};
try {
const response = UrlFetchApp.fetch(uploadUrl, options);
const responseCode = response.getResponseCode();
if (isLastChunk && responseCode === 200) {
Logger.log('最終チャンクのアップロードに成功しました。レスポンスを解析します。');
try {
const jsonResponse = JSON.parse(response.getContentText());
return jsonResponse;
} catch (e) {
Logger.log(JSONの解析に失敗しました: ${e.toString()});
return null;
}
} else if (!isLastChunk && responseCode === 308) {
Logger.log(チャンクのアップロードに成功しました。レスポンスコード: ${responseCode});
return true;
} else {
Logger.log(エラー: チャンクのアップロードに失敗しました。ステータスコード: ${responseCode});
Logger.log(エラーレスポンス: ${response.getContentText()});
return false;
}
} catch (e) {
Logger.log(チャンクのアップロード中に例外が発生しました: ${e.toString()});
return false;
}
}
/**
* API キーを取得
*/
function getApiKey() {
const apiKey = PropertiesService.getScriptProperties().getProperty('GEMINI_API_KEY');
if (!apiKey) {
Logger.log('エラー: GEMINI_API_KEY がスクリプトプロパティに設定されていません。');
return null;
}
return apiKey;
}
/**
* アップロードするファイルを検索
*/
function findFileToUpload() {
const files = DriveApp.searchFiles("title contains 'iom-ropo-sqm' and (mimeType contains 'audio/' or mimeType contains 'video/')");
if (!files.hasNext()) {
Logger.log('エラー: 条件に一致するファイルが見つかりません。');
return null;
}
const file = files.next();
Logger.log(ファイルが見つかりました: ${file.getName()} (${file.getMimeType()}, ${file.getSize()} バイト));
return file;
}
/**
* ファイルをアップロードし、アップロードされたファイルのURIを返す
* @return {string|null} アップロードされたファイルのURI、またはnull(失敗時)
*/
function uploadFileAndGetUri() {
const uploadResult = uploadFileToGenAI();
if (!uploadResult) {
Logger.log('ファイルのアップロードに失敗しました。');
return null;
}
try {
const fileInfo = uploadResult.file;
if (fileInfo && fileInfo.uri) {
Logger.log(アップロードされたファイルのURI: ${fileInfo.uri});
return fileInfo.uri;
} else {
Logger.log('アップロード結果にURIが含まれていません。');
Logger.log('完全なアップロード結果:', JSON.stringify(uploadResult, null, 2));
return null;
}
} catch (e) {
Logger.log(URIの抽出中にエラーが発生しました: ${e.toString()});
Logger.log('完全なアップロード結果:', JSON.stringify(uploadResult, null, 2));
return null;
}
}
そのままアップロードしたurlを紐づけて、動画の文字起こしに成功😺
ZennにTipsを書いた