is_default == 1) { self::query()->where("id", "!=", $model->id)->update(["is_default" => 0]); } self::saveUrlCache(); }); static::saving(function (TrackerUrl $model) { if ($model->is_default == 1) { $model->enabled = 1; } }); } public static function saveUrlCache(): void { //添加 id 与 URL 映射 $redis = NexusDB::redis(); $redis->unlink(self::TRACKER_URL_CACHE_KEY); $list = self::listAll(); $first = $list->first(); $hasDefault = false; foreach ($list as $item) { $redis->hset(self::TRACKER_URL_CACHE_KEY, $item->id, $item->url); if ($item->is_default == 1) { $hasDefault = true; $redis->set(self::TRACKER_URL_DEFAULT_CACHE_KEY, $item->url); } } if (!$hasDefault && $first) { $redis->set(self::TRACKER_URL_DEFAULT_CACHE_KEY, $first->url); } } public static function listAll() { return self::query() ->where("enabled", 1) ->orderBy("is_default", "desc") ->orderBy("priority", "desc") ->get(); } public static function getById(int $trackerUrlId) { $redis = NexusDB::redis(); $notFoundFlagKey = "TRACKER_URL_NOT_FOUND:$trackerUrlId"; if ($trackerUrlId == 0) { return self::getFromRedisWithRetry($redis, 'get', [self::TRACKER_URL_DEFAULT_CACHE_KEY], $notFoundFlagKey); } $result = self::getFromRedisWithRetry($redis, 'hGet', [self::TRACKER_URL_CACHE_KEY, $trackerUrlId], $notFoundFlagKey); if ($result !== false) { return $result; } do_log("No tracker url found for $trackerUrlId, try default", 'warning'); return self::getFromRedisWithRetry($redis, 'get', [self::TRACKER_URL_DEFAULT_CACHE_KEY], $notFoundFlagKey); } private static function getFromRedisWithRetry(\Redis $redis, string $command, array $params, string $notFoundFlagKey) { $result = call_user_func_array([$redis, $command], $params); if ($result !== false) { return $result; } //有不存在标记,不要再尝试,直接返回 false if ($redis->exists($notFoundFlagKey)) { return false; } $lockKey = "$notFoundFlagKey:lock"; if (!$redis->set($lockKey, 1, ["nx", "ex" => 5])) { return false; } try { self::saveUrlCache(); $result = call_user_func_array([$redis, $command], $params); if ($result !== false) { return $result; } //只从 db 拉取一次,仍然没有即标记不存在, 有效期 15 分钟 $redis->setex($notFoundFlagKey, 900, date("Y-m-d H:i:s")); } catch (\Throwable $throwable) { do_log($throwable->getMessage(), 'error'); } finally { $redis->del($lockKey); } do_log(sprintf("redis command %s with args %s no result", $command, json_encode($params)), 'error'); return false; } }