ai/src/modules/aichat/index.ts
tetsuya-ki fd76c424e8 aichat機能について修正(デフォルトをGemini API Flash 2.0 expに変更など)
- Gemini APIを基本Flash使用に変更
- Flashのバージョンを2.0 expに変更
- Flash, PRO指定時の不正な文章を修正
2025-01-05 18:02:34 +09:00

513 lines
15 KiB
TypeScript

import { bindThis } from '@/decorators.js';
import Module from '@/module.js';
import serifs from '@/serifs.js';
import Message from '@/message.js';
import config from '@/config.js';
import Friend from '@/friend.js';
import urlToBase64 from '@/utils/url2base64.js';
import got from 'got';
import loki from 'lokijs';
type AiChat = {
question: string;
prompt: string;
api: string;
key: string;
history?: { role: string; content: string }[];
};
type Base64Image = {
type: string;
base64: string;
};
type AiChatHist = {
postId: string;
createdAt: number;
type: string;
api?: string;
history?: {
role: string;
content: string;
}[];
};
const KIGO = '&';
const TYPE_GEMINI = 'gemini';
const GEMINI_PRO = 'gemini-pro';
const GEMINI_FLASH = 'gemini-flash';
const TYPE_PLAMO = 'plamo';
const GEMINI_20_FLASH_API = 'https://generativelanguage.googleapis.com/v1beta/models/gemini-2.0-flash-exp:generateContent';
// const GEMINI_15_FLASH_API = 'https://generativelanguage.googleapis.com/v1beta/models/gemini-1.5-flash:generateContent';
const GEMINI_15_PRO_API = 'https://generativelanguage.googleapis.com/v1beta/models/gemini-1.5-pro:generateContent';
const PLAMO_API = 'https://platform.preferredai.jp/api/completion/v1/chat/completions';
const RANDOMTALK_DEFAULT_PROBABILITY = 0.02;// デフォルトのrandomTalk確率
const TIMEOUT_TIME = 1000 * 60 * 60 * 0.5;// aichatの返信を監視する時間
const RANDOMTALK_DEFAULT_INTERVAL = 1000 * 60 * 60 * 12;// デフォルトのrandomTalk間隔
export default class extends Module {
public readonly name = 'aichat';
private aichatHist: loki.Collection<AiChatHist>;
private randomTalkProbability: number = RANDOMTALK_DEFAULT_PROBABILITY;
private randomTalkIntervalMinutes: number = RANDOMTALK_DEFAULT_INTERVAL;
@bindThis
public install() {
this.aichatHist = this.ai.getCollection('aichatHist', {
indices: ['postId']
});
// 確率は設定されていればそちらを採用(設定がなければデフォルトを採用)
if (config.aichatRandomTalkProbability != undefined && !Number.isNaN(Number.parseFloat(config.aichatRandomTalkProbability))) {
this.randomTalkProbability = Number.parseFloat(config.aichatRandomTalkProbability);
}
// ランダムトーク間隔(分)は設定されていればそちらを採用(設定がなければデフォルトを採用)
if (config.aichatRandomTalkIntervalMinutes != undefined && !Number.isNaN(Number.parseInt(config.aichatRandomTalkIntervalMinutes))) {
this.randomTalkIntervalMinutes = 1000 * 60 * Number.parseInt(config.aichatRandomTalkIntervalMinutes);
}
this.log('aichatRandomTalkEnabled:' + config.aichatRandomTalkEnabled);
this.log('randomTalkProbability:' + this.randomTalkProbability);
this.log('randomTalkIntervalMinutes:' + (this.randomTalkIntervalMinutes / (60 * 1000)));
// 定期的にデータを取得しaichatRandomTalkを行う
if (config.aichatRandomTalkEnabled) {
setInterval(this.aichatRandomTalk, this.randomTalkIntervalMinutes);
}
return {
mentionHook: this.mentionHook,
contextHook: this.contextHook,
timeoutCallback: this.timeoutCallback,
};
}
@bindThis
private async genTextByGemini(aiChat: AiChat, image:Base64Image|null) {
this.log('Generate Text By Gemini...');
let parts: ({ text: string; inline_data?: undefined; } | { inline_data: { mime_type: string; data: string; }; text?: undefined; })[];
const systemInstruction : {role: string; parts: [{text: string}]} = {role: 'system', parts: [{text: aiChat.prompt}]};
if (!image) {
// 画像がない場合、メッセージのみで問い合わせ
parts = [{text: aiChat.question}];
} else {
// 画像が存在する場合、画像を添付して問い合わせ
parts = [
{ text: aiChat.question },
{
inline_data: {
mime_type: image.type,
data: image.base64,
},
},
];
}
// 履歴を追加
let contents: ({ role: string; parts: ({ text: string; inline_data?: undefined; } | { inline_data: { mime_type: string; data: string; }; text?: undefined; })[]}[]) = [];
if (aiChat.history != null) {
aiChat.history.forEach(entry => {
contents.push({ role : entry.role, parts: [{text: entry.content}]});
});
}
contents.push({role: 'user', parts: parts});
let options = {
url: aiChat.api,
searchParams: {
key: aiChat.key,
},
json: {
contents: contents,
systemInstruction: systemInstruction,
},
};
this.log(JSON.stringify(options));
let res_data:any = null;
try {
res_data = await got.post(options,
{parseJson: (res: string) => JSON.parse(res)}).json();
{parseJson: res => JSON.parse(res)}).json();
this.log(JSON.stringify(res_data));
if (res_data.hasOwnProperty('candidates')) {
if (res_data.candidates.length > 0) {
if (res_data.candidates[0].hasOwnProperty('content')) {
if (res_data.candidates[0].content.hasOwnProperty('parts')) {
if (res_data.candidates[0].content.parts.length > 0) {
if (res_data.candidates[0].content.parts[0].hasOwnProperty('text')) {
const responseText = res_data.candidates[0].content.parts[0].text;
return responseText;
}
}
}
}
}
}
} catch (err: unknown) {
this.log('Error By Call Gemini');
if (err instanceof Error) {
this.log(`${err.name}\n${err.message}\n${err.stack}`);
}
}
return null;
}
@bindThis
private async genTextByPLaMo(aiChat: AiChat) {
this.log('Generate Text By PLaMo...');
let options = {
url: aiChat.api,
headers: {
Authorization: 'Bearer ' + aiChat.key
},
json: {
model: 'plamo-beta',
messages: [
{role: 'system', content: aiChat.prompt},
{role: 'user', content: aiChat.question},
],
},
};
this.log(JSON.stringify(options));
let res_data:any = null;
try {
res_data = await got.post(options,
{parseJson: (res: string) => JSON.parse(res)}).json();
this.log(JSON.stringify(res_data));
if (res_data.hasOwnProperty('choices')) {
if (res_data.choices.length > 0) {
if (res_data.choices[0].hasOwnProperty('message')) {
if (res_data.choices[0].message.hasOwnProperty('content')) {
return res_data.choices[0].message.content;
}
}
}
}
} catch (err: unknown) {
this.log('Error By Call PLaMo');
if (err instanceof Error) {
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;
}
@bindThis
private async mentionHook(msg: Message) {
if (!msg.includes([this.name])) {
return false;
} else {
this.log('AiChat requested');
}
// msg.idをもとにnotes/conversationを呼び出し、会話中のidかチェック
const conversationData = await this.ai.api('notes/conversation', { noteId: msg.id });
// aichatHistに該当のポストが見つかった場合は会話中のためmentionHoonkでは対応しない
let exist : AiChatHist | null = null;
if (conversationData != undefined) {
for (const message of conversationData) {
exist = this.aichatHist.findOne({
postId: message.id
});
// 見つかった場合はそれを利用
if (exist != null) return false;
}
}
// タイプを決定
let type = TYPE_GEMINI;
if (msg.includes([KIGO + TYPE_GEMINI])) {
type = TYPE_GEMINI;
} else if (msg.includes([KIGO + 'chatgpt4'])) {
type = 'chatgpt4';
} else if (msg.includes([KIGO + 'chatgpt'])) {
type = 'chatgpt3.5';
} else if (msg.includes([KIGO + TYPE_PLAMO])) {
type = TYPE_PLAMO;
}
const current : AiChatHist = {
postId: msg.id,
createdAt: Date.now(),// 適当なもの
type: type
};
// AIに問い合わせ
const result = await this.handleAiChat(current, msg);
if (result) {
return {
reaction: 'like'
};
}
return false;
}
@bindThis
private async contextHook(key: any, msg: Message) {
this.log('contextHook...');
if (msg.text == null) return false;
// msg.idをもとにnotes/conversationを呼び出し、該当のidかチェック
const conversationData = await this.ai.api('notes/conversation', { noteId: msg.id });
// 結果がnullやサイズ0の場合は終了
if (conversationData == null || conversationData.length == 0 ) {
this.log('conversationData is nothing.');
return false;
}
// aichatHistに該当のポストが見つからない場合は終了
let exist : AiChatHist | null = null;
for (const message of conversationData) {
exist = this.aichatHist.findOne({
postId: message.id
});
// 見つかった場合はそれを利用
if (exist != null) break;
}
if (exist == null) {
this.log('conversationData is not found.');
return false;
}
// 見つかった場合はunsubscribe&removeし、回答。今回のでsubscribe,insert,timeout設定
this.log('unsubscribeReply & remove.');
this.log(exist.type + ':' + exist.postId);
if (exist.history) {
for (const his of exist.history) {
this.log(his.role + ':' + his.content);
}
}
this.unsubscribeReply(key);
this.aichatHist.remove(exist);
// AIに問い合わせ
const result = await this.handleAiChat(exist, msg);
if (result) {
return {
reaction: 'like'
};
}
return false;
}
@bindThis
private async aichatRandomTalk() {
this.log('AiChat(randomtalk) started');
const tl = await this.ai.api('notes/local-timeline', {
limit: 30
});
const interestedNotes = tl.filter(note =>
note.userId !== this.ai.account.id &&
note.text != null &&
note.replyId == null &&
note.renoteId == null &&
note.cw == null &&
!note.user.isBot
);
// 対象が存在しない場合は処理終了
if (interestedNotes == undefined || interestedNotes.length == 0) return false;
// ランダムに選択
const choseNote = interestedNotes[Math.floor(Math.random() * interestedNotes.length)];
// msg.idをもとにnotes/conversationを呼び出し、会話中のidかチェック
const conversationData = await this.ai.api('notes/conversation', { noteId: choseNote.id });
// aichatHistに該当のポストが見つかった場合は会話中のためaichatRandomTalkでは対応しない
let exist : AiChatHist | null = null;
if (conversationData != undefined) {
for (const message of conversationData) {
exist = this.aichatHist.findOne({
postId: message.id
});
if (exist != null) return false;
}
}
// 確率をクリアし、親愛度が指定以上、かつ、Botでない場合のみ実行
if (Math.random() < this.randomTalkProbability) {
this.log('AiChat(randomtalk) targeted: ' + choseNote.id);
} else {
this.log('AiChat(randomtalk) is end.');
return false;
}
const friend: Friend | null = this.ai.lookupFriend(choseNote.userId);
if (friend == null || friend.love < 7) {
this.log('AiChat(randomtalk) end.Because there was not enough affection.');
return false;
} else if (choseNote.user.isBot) {
this.log('AiChat(randomtalk) end.Because message author is bot.');
return false;
}
const current : AiChatHist = {
postId: choseNote.id,
createdAt: Date.now(),// 適当なもの
type: TYPE_GEMINI
};
// AIに問い合わせ
let targetedMessage = choseNote;
if (choseNote.extractedText == undefined) {
const data = await this.ai.api('notes/show', { noteId: choseNote.id });
targetedMessage = new Message(this.ai, data);
}
const result = await this.handleAiChat(current, targetedMessage);
if (result) {
return {
reaction: 'like'
};
}
return false;
}
@bindThis
private async handleAiChat(exist: AiChatHist, msg: Message) {
let text: string, aiChat: AiChat;
let prompt: string = '';
if (config.prompt) {
prompt = config.prompt;
}
const reName = RegExp(this.name, 'i');
let reKigoType = RegExp(KIGO + exist.type, 'i');
const extractedText = msg.extractedText;
if (extractedText == undefined || extractedText.length == 0) return false;
// Gemini API用にAPIのURLと置き換え用タイプを変更
if (msg.includes([KIGO + GEMINI_FLASH])) {
exist.api = GEMINI_20_FLASH_API;
reKigoType = RegExp(KIGO + GEMINI_FLASH, 'i');
} else if (msg.includes([KIGO + GEMINI_PRO])) {
exist.api = GEMINI_15_PRO_API;
reKigoType = RegExp(KIGO + GEMINI_PRO, 'i');
}
const question = extractedText
.replace(reName, '')
.replace(reKigoType, '')
.trim();
switch (exist.type) {
case TYPE_GEMINI:
// geminiの場合、APIキーが必須
if (!config.geminiProApiKey) {
msg.reply(serifs.aichat.nothing(exist.type));
return false;
}
const base64Image: Base64Image | null = await this.note2base64Image(msg.id);
aiChat = {
question: question,
prompt: prompt,
api: GEMINI_20_FLASH_API,
key: config.geminiProApiKey,
history: exist.history
};
if (exist.api) {
aiChat.api = exist.api
}
text = await this.genTextByGemini(aiChat, base64Image);
break;
case TYPE_PLAMO:
// PLaMoの場合、APIキーが必須
if (!config.pLaMoApiKey) {
msg.reply(serifs.aichat.nothing(exist.type));
return false;
}
aiChat = {
question: msg.text,
prompt: prompt,
api: PLAMO_API,
key: config.pLaMoApiKey,
history: exist.history
};
text = await this.genTextByPLaMo(aiChat);
break;
default:
msg.reply(serifs.aichat.nothing(exist.type));
return false;
}
if (text == null) {
this.log('The result is invalid. It seems that tokens and other items need to be reviewed.')
msg.reply(serifs.aichat.error(exist.type));
return false;
}
this.log('Replying...');
msg.reply(serifs.aichat.post(text, exist.type)).then(reply => {
// 履歴に登録
if (!exist.history) {
exist.history = [];
}
exist.history.push({ role: 'user', content: question });
exist.history.push({ role: 'model', content: text });
// 履歴が10件を超えた場合、古いものを削除
if (exist.history.length > 10) {
exist.history.shift();
}
this.aichatHist.insertOne({
postId: reply.id,
createdAt: Date.now(),
type: exist.type,
api: aiChat.api,
history: exist.history
});
this.log('Subscribe&Set Timer...');
// メンションをsubscribe
this.subscribeReply(reply.id, reply.id);
// タイマーセット
this.setTimeoutWithPersistence(TIMEOUT_TIME, {
id: reply.id
});
});
return true;
}
@bindThis
private async timeoutCallback({id}) {
this.log('timeoutCallback...');
const exist = this.aichatHist.findOne({
postId: id
});
this.unsubscribeReply(id);
if (exist != null) {
this.aichatHist.remove(exist);
}
}
}