enhance: aichat機能(画像対応)

This commit is contained in:
tetsuya-ki 2024-03-08 23:25:17 +09:00
parent 47e17052c5
commit c30659e526
2 changed files with 79 additions and 12 deletions

View file

@ -3,6 +3,7 @@ import Module from '@/module.js';
import serifs from '@/serifs.js'; import serifs from '@/serifs.js';
import Message from '@/message.js'; import Message from '@/message.js';
import config from '@/config.js'; import config from '@/config.js';
import urlToBase64 from '@/utils/url2base64.js';
import got from 'got'; import got from 'got';
type AiChat = { type AiChat = {
@ -11,7 +12,12 @@ type AiChat = {
api: string; api: string;
key: string; key: string;
}; };
type Base64Image = {
type: string;
base64: string;
};
const GEMINI_API = 'https://generativelanguage.googleapis.com/v1beta/models/gemini-pro:generateContent'; const GEMINI_API = 'https://generativelanguage.googleapis.com/v1beta/models/gemini-pro:generateContent';
const GEMINI_VISION_API = 'https://generativelanguage.googleapis.com/v1beta/models/gemini-pro-vision:generateContent';
export default class extends Module { export default class extends Module {
public readonly name = 'aichat'; public readonly name = 'aichat';
@ -24,21 +30,34 @@ export default class extends Module {
} }
@bindThis @bindThis
private async genTextByGemini(aiChat: AiChat) { private async genTextByGemini(aiChat: AiChat, image:Base64Image|null) {
this.log('Generate Text By Gemini...'); this.log('Generate Text By Gemini...');
var options = { let parts: ({ text: string; inline_data?: undefined; } | { inline_data: { mime_type: string; data: string; }; text?: undefined; })[];
if (image === null) {
// 画像がない場合、メッセージのみで問い合わせ
parts = [{text: aiChat.prompt + aiChat.question}];
} else {
// 画像が存在する場合、画像を添付して問い合わせ
parts = [
{ text: aiChat.prompt + aiChat.question },
{
inline_data: {
mime_type: image.type,
data: image.base64,
},
},
];
}
let options = {
url: aiChat.api, url: aiChat.api,
searchParams: { searchParams: {
key: aiChat.key, key: aiChat.key,
}, },
json: { json: {
contents: [{ contents: {parts: parts}
parts:[{
text: aiChat.prompt + aiChat.question
}]
}]
}, },
}; };
this.log(JSON.stringify(options));
let res_data:any = null; let res_data:any = null;
try { try {
res_data = await got.post(options, res_data = await got.post(options,
@ -60,7 +79,35 @@ export default class extends Module {
} catch (err: unknown) { } catch (err: unknown) {
this.log('Error By Call Gemini'); this.log('Error By Call Gemini');
if (err instanceof Error) { if (err instanceof Error) {
this.log(`${err.name}\n${err.message}`); this.log(`${err.name}\n${err.message}\n${err.stack}`);
}
}
return null;
}
@bindThis
private async note2base64Image(notesId: string) {
const noteData = await this.ai.api('notes/show', { noteId: notesId });
let fileType: string | undefined,thumbnailUrl: string | undefined;
if (noteData !== null && noteData.hasOwnProperty('files')) {
if (noteData.files.length > 0) {
if (noteData.files[0].hasOwnProperty('type')) {
fileType = noteData.files[0].type;
}
if (noteData.files[0].hasOwnProperty('thumbnailUrl')) {
thumbnailUrl = noteData.files[0].thumbnailUrl;
}
}
if (fileType !== undefined && thumbnailUrl !== undefined) {
try {
const image = await urlToBase64(thumbnailUrl);
const base64Image:Base64Image = {type: fileType, base64: image};
return base64Image;
} catch (err: unknown) {
if (err instanceof Error) {
this.log(`${err.name}\n${err.message}\n${err.stack}`);
}
}
} }
} }
return null; return null;
@ -88,8 +135,8 @@ export default class extends Module {
.replace(kigo + type, '') .replace(kigo + type, '')
.trim(); .trim();
let text; let text:string, aiChat:AiChat;
let prompt = ''; let prompt:string = '';
if (config.prompt) { if (config.prompt) {
prompt = config.prompt; prompt = config.prompt;
} }
@ -99,13 +146,17 @@ export default class extends Module {
msg.reply(serifs.aichat.nothing(type)); msg.reply(serifs.aichat.nothing(type));
return false; return false;
} }
const aiChat = { const base64Image:Base64Image|null = await this.note2base64Image(msg.id);
aiChat = {
question: question, question: question,
prompt: prompt, prompt: prompt,
api: GEMINI_API, api: GEMINI_API,
key: config.geminiProApiKey key: config.geminiProApiKey
}; };
text = await this.genTextByGemini(aiChat); if (base64Image !== null) {
aiChat.api = GEMINI_VISION_API;
}
text = await this.genTextByGemini(aiChat, base64Image);
break; break;
default: default:
msg.reply(serifs.aichat.nothing(type)); msg.reply(serifs.aichat.nothing(type));

16
src/utils/url2base64.ts Normal file
View file

@ -0,0 +1,16 @@
import log from '@/utils/log.js';
import got from 'got';
export default async function(url: string): Promise<string> {
try {
const buffer = await got(url).buffer();
const base64Image = buffer.toString('base64');
return base64Image;
} catch (err: unknown) {
log('Error in url2base64');
if (err instanceof Error) {
log(`${err.name}\n${err.message}\n${err.stack}`);
}
throw err;
}
}