'系统消息', 'task-alert' => '任务提醒', 'check-in' => '签到打卡', 'anon-msg' => '匿名消息', 'approval-alert' => '审批', 'ai-openai' => 'ChatGPT', 'ai-claude' => 'Claude', 'ai-wenxin' => '文心一言', 'ai-qianwen' => '通义千问', 'ai-gemini' => 'Gemini', 'ai-zhipu' => '智谱清言', 'bot-manager' => '机器人管理', 'meeting-alert' => '会议通知', 'okr-alert' => 'OKR提醒', default => '', // 不是系统机器人时返回空(也可以拿来判断是否是系统机器人) }; return Doo::translate($name); } /** * 机器人菜单 * @param $email * @return array|array[] */ public static function quickMsgs($email) { switch ($email) { case 'check-in@bot.system': $menu = [ /*[ 'key' => 'it', 'label' => Doo::translate('IT资讯') ], [ 'key' => '36ke', 'label' => Doo::translate('36氪') ], [ 'key' => '60s', 'label' => Doo::translate('60s读世界') ], [ 'key' => 'joke', 'label' => Doo::translate('开心笑话') ], [ 'key' => 'soup', 'label' => Doo::translate('心灵鸡汤') ]*/ ]; $setting = Base::setting('checkinSetting'); if ($setting['open'] !== 'open') { return $menu; } if (in_array('locat', $setting['modes']) && Base::isEEUIApp()) { $menu[] = [ 'key' => 'locat-checkin', 'label' => Doo::translate('定位签到'), 'config' => [ 'key' => $setting['locat_bd_lbs_key'], 'lng' => $setting['locat_bd_lbs_point']['lng'], 'lat' => $setting['locat_bd_lbs_point']['lat'], 'radius' => $setting['locat_bd_lbs_point']['radius'], ] ]; } if (in_array('manual', $setting['modes'])) { $menu[] = [ 'key' => 'manual-checkin', 'label' => Doo::translate('手动签到') ]; } return $menu; case 'anon-msg@bot.system': return [ [ 'key' => 'help', 'label' => Doo::translate('使用说明') ], [ 'key' => 'privacy', 'label' => Doo::translate('隐私说明') ], ]; case 'meeting-alert@bot.system': if (!Base::judgeClientVersion('0.39.89')) { return []; } return [ [ 'key' => 'meeting-create', 'label' => Doo::translate('新会议') ], [ 'key' => 'meeting-join', 'label' => Doo::translate('加入会议') ], ]; case 'bot-manager@bot.system': return [ [ 'key' => '/help', 'label' => Doo::translate('帮助指令') ], [ 'key' => '/api', 'label' => Doo::translate('API接口文档') ], [ 'key' => '/list', 'label' => Doo::translate('我的机器人') ], ]; default: if (preg_match('/^ai-(.*?)@bot.system$/', $email)) { return [ [ 'key' => '%3A.clear', 'label' => Doo::translate('清空上下文') ] ]; } return []; } } /** * 签到机器人 * @param $command * @param $userid * @param $extra * @return string */ public static function checkinBotQuickMsg($command, $userid, $extra = []) { if (Cache::get("UserBot::checkinBotQuickMsg:{$userid}") === "yes") { return "操作频繁!"; } Cache::put("UserBot::checkinBotQuickMsg:{$userid}", "yes", Carbon::now()->addSecond()); // if ($command === 'manual-checkin') { $setting = Base::setting('checkinSetting'); if ($setting['open'] !== 'open') { return '暂未开启签到功能。'; } if (!in_array('manual', $setting['modes'])) { return '暂未开放手动签到。'; } if ($error = UserBot::checkinBotCheckin('manual-' . $userid, Base::time(), true)) { return $error; } return null; } elseif ($command === 'locat-checkin') { $setting = Base::setting('checkinSetting'); if ($setting['open'] !== 'open') { return '暂未开启签到功能。'; } if (!in_array('locat', $setting['modes'])) { return '暂未开放定位签到。'; } if (empty($extra)) { return '当前客户端版本低(所需版本≥v0.39.75)。'; } if ($extra['type'] === 'bd') { // todo 判断距离 } else { return '错误的定位签到。'; } if ($error = UserBot::checkinBotCheckin('locat-' . $userid, Base::time(), true)) { return $error; } return null; } else { return Extranet::checkinBotQuickMsg($command); } } /** * 签到机器人签到 * @param mixed $mac * - 多个使用,分隔 * - 支持:mac地址、(manual|locat|face|checkin)-userid * @param $time * @param bool $alreadyTip 签到过是否提示 * @return string|null 返回string表示错误信息,返回null表示签到成功 */ public static function checkinBotCheckin($mac, $time, $alreadyTip = false) { $setting = Base::setting('checkinSetting'); $times = $setting['time'] ? Base::json2array($setting['time']) : ['09:00', '18:00']; $advance = (intval($setting['advance']) ?: 120) * 60; $delay = (intval($setting['delay']) ?: 120) * 60; // $nowDate = date("Y-m-d"); $nowTime = date("H:i:s"); // $timeStart = strtotime("{$nowDate} {$times[0]}"); $timeEnd = strtotime("{$nowDate} {$times[1]}"); $timeAdvance = max($timeStart - $advance, strtotime($nowDate)); $timeDelay = min($timeEnd + $delay, strtotime("{$nowDate} 23:59:59")); if (Base::time() < $timeAdvance || $timeDelay < Base::time()) { return "不在有效时间内,有效时间为:" . date("H:i", $timeAdvance) . "-" . date("H:i", $timeDelay); } // $macs = explode(",", $mac); $checkins = []; foreach ($macs as $mac) { $mac = strtoupper($mac); $array = []; if (Base::isMac($mac)) { if ($UserCheckinMac = UserCheckinMac::whereMac($mac)->first()) { $array = [ 'userid' => $UserCheckinMac->userid, 'mac' => $UserCheckinMac->mac, 'date' => $nowDate, ]; $checkins[] = [ 'userid' => $UserCheckinMac->userid, 'remark' => $UserCheckinMac->remark, ]; } } elseif (preg_match('/^(manual|locat|face|checkin)-(\d+)$/i', $mac, $match)) { $type = str_replace('checkin', 'face', $match[1]); $mac = intval($match[2]); $remark = match ($type) { 'manual' => $setting['manual_remark'] ?: 'Manual', 'locat' => $setting['locat_remark'] ?: 'Location', 'face' => $setting['face_remark'] ?: 'Machine', default => '', }; if ($UserInfo = User::whereUserid($mac)->whereBot(0)->first()) { $array = [ 'userid' => $UserInfo->userid, 'mac' => '00:00:00:00:00:00', 'date' => $nowDate, ]; $checkins[] = [ 'userid' => $UserInfo->userid, 'remark' => $remark, ]; } } if ($array) { $record = UserCheckinRecord::where($array)->first(); if (empty($record)) { $record = UserCheckinRecord::createInstance($array); } $record->times = Base::array2json(array_merge($record->times, [$nowTime])); $record->report_time = $time; $record->save(); } } // if ($checkins && $botUser = User::botGetOrCreate('check-in')) { $getJokeSoup = function($type, $userid) { $pre = $type == "up" ? "每日开心:" : "心灵鸡汤:"; $key = $type == "up" ? "jokes" : "soups"; $array = Base::json2array(Cache::get(JokeSoupTask::keyName($key))); if ($array) { $item = $array[array_rand($array)]; if ($item) { Doo::setLanguage($userid); return Doo::translate($pre . $item); } } return null; }; $sendMsg = function($type, $checkin) use ($alreadyTip, $getJokeSoup, $botUser, $nowDate) { $cacheKey = "Checkin::sendMsg-{$nowDate}-{$type}:" . $checkin['userid']; $typeContent = $type == "up" ? "上班" : "下班"; if (Cache::get($cacheKey) === "yes") { if ($alreadyTip && $dialog = WebSocketDialog::checkUserDialog($botUser, $checkin['userid'])) { $text = "今日已{$typeContent}打卡,无需重复打卡。"; $text .= $checkin['remark'] ? " ({$checkin['remark']})": ""; WebSocketDialogMsg::sendMsg(null, $dialog->id, 'template', [ 'type' => 'content', 'content' => $text, ], $botUser->userid, false, false, true); } return; } Cache::put($cacheKey, "yes", Carbon::now()->addDay()); // if ($dialog = WebSocketDialog::checkUserDialog($botUser, $checkin['userid'])) { $hi = date("H:i"); $remark = $checkin['remark'] ? " ({$checkin['remark']})": ""; $subcontent = $getJokeSoup($type, $checkin['userid']); $title = "{$typeContent}打卡成功,打卡时间: {$hi}{$remark}"; WebSocketDialogMsg::sendMsg(null, $dialog->id, 'template', [ 'type' => 'content', 'title' => $title, 'content' => [ [ 'content' => $title ], [ 'content' => $subcontent, 'language' => false, 'style' => 'padding-top:4px;opacity:0.6', ] ], ], $botUser->userid, false, false, $type != "up"); } }; if ($timeAdvance <= Base::time() && Base::time() < $timeEnd) { // 上班打卡通知(从最早打卡时间 到 下班打卡时间) foreach ($checkins as $checkin) { $sendMsg('up', $checkin); } } if ($timeEnd <= Base::time() && Base::time() <= $timeDelay) { // 下班打卡通知(下班打卡时间 到 最晚打卡时间) foreach ($checkins as $checkin) { $sendMsg('down', $checkin); } } } return null; } /** * 隐私机器人 * @param $command * @return array */ public static function anonBotQuickMsg($command) { return match ($command) { "help" => [ "title" => "匿名消息使用说明", "content" => "使用说明:打开你想要发匿名消息的个人对话,点击输入框右边的 ⊕ 号,选择「匿名消息」即可输入你想要发送的匿名消息内容。" ], "privacy" => [ "title" => "匿名消息隐私说明", "content" => "匿名消息将通过「匿名消息(机器人)」发送给对方,不会记录你的身份信息。" ], default => [], }; } }