mirror of
https://github.com/syuilo/ai.git
synced 2025-03-25 21:12:56 +00:00
enhance: aichat機能(画像対応)
This commit is contained in:
parent
47e17052c5
commit
c30659e526
2 changed files with 79 additions and 12 deletions
|
@ -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
16
src/utils/url2base64.ts
Normal 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;
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue