淘宝 API(拍立淘) 是基于图像识别的相似商品检索利器,通过上传图片即可在淘宝 / 天猫库中精准匹配同款、相似商品,广泛用于比价、选品、竞品监控等场景。
一、核心接口(拍立淘)
taobao.item.search.img (或 taobao.image.search)

功能:上传图片(URL/Base64)→ 返回相似商品列表
技术:ResNet/MobileNet 深度学习提取特征,毫秒级匹配
返回字段:商品 ID、标题、价格、主图、店铺、相似度 (0-100)

二、接入流程

  1. 注册认证(必备)

核心接口涵盖接入与配额: 平台提供多款接口,分为竞品内容监测,用户画像与细分群体分析,舆情监控与危机预警,市场趋势与热点预测,内容创作辅助,商业情报收集等场景,无需自主申请接口(如添加 Taobaoapi2014),直接调用封装 API,一键获取已封装好的数据API采集,适合批量查询、中小卖家使用。

  1. 申请接口权限

权限管理 → 申请 taobao.item.search.img
填写使用场景(如:商品比价、智能推荐)

  1. 图片规范(影响准确率)

格式:JPG / PNG
大小:≤ 2MB
分辨率:≥ 800×800
主体:商品占比 ≥ 60%
背景:纯色、无水印、无遮挡

三、关键参数

image:图片 Base64 或 URL(必填)
cat:类目 ID(可选,缩小范围)
sort:排序(price_asc/price_desc/sales_desc)
page:分页(默认 1,最多 50 条)

四、典型应用场景

同款比价:上传商品图 → 找全网最低价
商品溯源:识别图片商品 → 查源头 / 同款
智能推荐:图文 / 视频带货 → 匹配相似商品
竞品监控:监测同款价格、销量、上新

总结
淘宝图片搜索 API 是电商视觉检索的工业级方案,低成本、高精准、易集成。适合快速搭建同款比价、商品溯源、智能推荐等系统。

VBCoding 中转站

官方地址


⚡ 当前可用服务

1️⃣ codex-to-claude 官方 plus/pro 账户号池

  • 倍率:0.12
  • 20 刀约等于 166.7 刀可用额度

2️⃣ codex / gpt 官方 plus/pro 账户号池

  • 倍率:0.18
  • 20 刀约等于 111.1 刀可用额度


抽奖活动 - 无任何套路

  1. 在本帖回复你的注册 ID (👉 个人设置查看 https://vbcode.io/console/personal ),例如:ID: 123

奖品

20 美刀余额 * 1 。共计 50 份

⏰ 截止时间

2026 年 4 月 17 日 15:00


抽奖规则

  1. 取当日 A 股沪指收盘点位整数部分,记为 S
  2. 取 15:00 前的楼层总数 N
  3. 依次计算中奖楼层:(S + k × 7) % N( k = 0, 1, 2, ...)
  4. 若中奖楼层的用户已中奖,则向下顺延至下一位未中奖用户的楼层
  5. 若抽到楼主楼层,则同样向下顺延
  6. 直到选出 50 位不同用户,每人获得 20 美刀余额

步长取质数 7 ,使中奖楼层分布更均匀,结果完全可验证。


示例

假设沪指收盘 3500 点,楼层总数 300:

序号 计算 楼层
1 3500 % 300 200 楼 ✅
2 3507 % 300 207 楼 ✅
3 3514 % 300 214 楼 ✅
... ... ...

若某楼层用户已中奖,则顺延到下一位未中奖用户,直到选满 50 人。

今天账号被封的理由改了:短时间发送大量重复请求。

我一直在用别的大模型。最近用 glm5.1 ,只有一个用途:zread cli,给我本地的代码生成文档。

看来是我错了,我不该用。发帖提个醒

突然想起来,想讨论讨论,不针对哪一个商家(下面的白象只是举个栗子)

像这种商标,当然还有很多(“手打”挂面,今麦郎“1 袋半”,“山里来的土”鸡蛋,“千禾 0”添加,等等)

https://i.imgur.com/fyj9hUY.jpeg

商标文字游戏,商家的投机取巧是肯定的,但作为审核的机制,是商标局的不作为,是商标法的缺漏

有些操作很慢,Agent 不能干等着。例如长时间编译/构建make, mvn compile, gradle build大数据处理hadoop, spark-submit 等的一些工作

Java实现代码

public class BackgroundTasksSystem {
    // --- 配置 ---
    private static final Path WORKDIR = Paths.get(System.getProperty("user.dir"));
    private static final Gson gson = new GsonBuilder().setPrettyPrinting().create();
    
    // --- 后台任务管理器 ---
    static class BackgroundManager {
        // 任务存储
        private final Map<String, TaskInfo> tasks = new ConcurrentHashMap<>();
        // 通知队列
        private final Queue<TaskNotification> notificationQueue = new ConcurrentLinkedQueue<>();
        // 任务 ID 生成器
        private final AtomicInteger taskIdCounter = new AtomicInteger(1);
        // 锁
        private final Object lock = new Object();
        
        static class TaskInfo {
            String taskId;
            String status;  // running, completed, timeout, error
            String result;
            String command;
            long startTime;
            Thread thread;  // 关联的执行线程
        }
        
        static class TaskNotification {
            String taskId;
            String status;
            String command;
            String result;
        }
        
        /**
         * 启动后台任务
         * 立即返回任务 ID,不等待命令完成
         */
        public String run(String command) {
            String taskId = "task_" + taskIdCounter.getAndIncrement();
            
            TaskInfo task = new TaskInfo(taskId, command);
            tasks.put(taskId, task);
            
            // 创建并启动后台线程
            Thread thread = new Thread(() -> executeTask(task), "BackgroundTask-" + taskId);
            thread.setDaemon(true);
            task.thread = thread;
            thread.start();  // 立即返回,不阻塞
            
            return String.format("Background task %s started: %s", 
                taskId, command.substring(0, Math.min(command.length(), 80)));
        }
        
        /**
         * 线程目标:执行子进程,捕获输出,推送结果到队列
         */
        private void executeTask(TaskInfo task) {
            String output;
            String status;
            
            try {
                ProcessBuilder pb = new ProcessBuilder("bash", "-c", task.command);
                pb.directory(WORKDIR.toFile());
                pb.redirectErrorStream(true);
                
                Process process = pb.start();
                boolean finished = process.waitFor(300, TimeUnit.SECONDS);  // 5分钟超时
                
                if (!finished) {
                    process.destroy();
                    output = "Error: Timeout (300s)";
                    status = "timeout";
                } else {
                    output = new String(process.getInputStream().readAllBytes()).trim();
                    status = "completed";
                }
            } catch (Exception e) {
                output = "Error: " + e.getMessage();
                status = "error";
            }
            
            // 更新任务状态
            task.status = status;
            task.result = output.isEmpty() ? "(no output)" : 
                          output.substring(0, Math.min(output.length(), 50000));
            
            // 添加通知到队列
            synchronized (lock) {
                notificationQueue.offer(new TaskNotification(
                    task.taskId,
                    status,
                    task.command.substring(0, Math.min(task.command.length(), 80)),
                    task.result.substring(0, Math.min(task.result.length(), 500))
                ));
            }
        }
        
        /**
         * 检查任务状态
         * 如果指定 taskId,检查单个任务;否则列出所有任务
         */
        public String check(String taskId) {
            if (taskId != null && !taskId.isEmpty()) {
                TaskInfo task = tasks.get(taskId);
                if (task == null) {
                    return "Error: Unknown task " + taskId;
                }
                return String.format("[%s] %s\n%s", 
                    task.status,
                    task.command.substring(0, Math.min(task.command.length(), 60)),
                    task.result != null ? task.result : "(running)");
            } else {
                StringBuilder sb = new StringBuilder();
                for (Map.Entry<String, TaskInfo> entry : tasks.entrySet()) {
                    TaskInfo task = entry.getValue();
                    sb.append(String.format("%s: [%s] %s\n", 
                        task.taskId,
                        task.status,
                        task.command.substring(0, Math.min(task.command.length(), 60))));
                }
                return sb.length() > 0 ? sb.toString().trim() : "No background tasks.";
            }
        }
        
        /**
         * 清空通知队列并返回所有待处理的通知
         */
        public List<TaskNotification> drainNotifications() {
            synchronized (lock) {
                List<TaskNotification> notifications = new ArrayList<>();
                while (!notificationQueue.isEmpty()) {
                    notifications.add(notificationQueue.poll());
                }
                return notifications;
            }
        }
        
        /**
         * 获取所有任务
         */
        public Map<String, TaskInfo> getAllTasks() {
            return new HashMap<>(tasks);
        }
    }
    
    // 初始化后台管理器
    private static final BackgroundManager BG_MANAGER = new BackgroundManager();
    
    // --- 工具枚举 ---
    public enum ToolType {
        BASH("bash", "Run a shell command (blocking)."),
        READ_FILE("read_file", "Read file contents."),
        WRITE_FILE("write_file", "Write content to file."),
        EDIT_FILE("edit_file", "Replace exact text in file."),
        BACKGROUND_RUN("background_run", "Run command in background thread. Returns task_id immediately."),  // 新增
        CHECK_BACKGROUND("check_background", "Check background task status. Omit task_id to list all.");  // 新增
        public final String name;
        public final String description;
        ToolType(String name, String description) { this.name = name; this.description = description; }
    }

    // --- 工具处理器映射 ---
    private static final Map<String, ToolExecutor> TOOL_HANDLERS = new HashMap<>();
    
    static {
        // ... 省略基础工具注册
        
        // 后台任务工具
        TOOL_HANDLERS.put(ToolType.BACKGROUND_RUN.name, args -> {
            String command = (String) args.get("command");
            return BG_MANAGER.run(command);
        });
        
        TOOL_HANDLERS.put(ToolType.CHECK_BACKGROUND.name, args -> {
            String taskId = (String) args.get("task_id");
            return BG_MANAGER.check(taskId);
        });
    }
    
    // ... 省略相同的工具实现
    
    // --- Agent 主循环(集成后台任务通知)---
    public static void agentLoop(List<Map<String, Object>> messages) {
        while (true) {
            try {
                // 在 LLM 调用前检查后台通知
                List<BackgroundManager.TaskNotification> notifications = BG_MANAGER.drainNotifications();
                
                if (!notifications.isEmpty() && !messages.isEmpty()) {
                    StringBuilder notifText = new StringBuilder();
                    notifText.append("<background-results>\n");
                    
                    for (BackgroundManager.TaskNotification notif : notifications) {
                        notifText.append(String.format("[bg:%s] %s: %s\n", 
                            notif.taskId, notif.status, notif.result));
                    }
                    
                    notifText.append("</background-results>");
                    
                    messages.add(Map.of(
                        "role", "user",
                        "content", notifText.toString()
                    ));
                    
                    messages.add(Map.of(
                        "role", "assistant",
                        "content", "Noted background results."
                    ));
                    // 异步结果注入:将后台任务结果插入到对话中
                    // 结构化格式:用XML标签包裹,便于LLM解析
                }
                
                // 显示当前活动任务
                Map<String, BackgroundManager.TaskInfo> activeTasks = BG_MANAGER.getAllTasks();
                int runningTasks = (int) activeTasks.values().stream()
                    .filter(t -> "running".equals(t.status))
                    .count();
                
                if (runningTasks > 0) {
                    System.out.printf("[Active background tasks: %d]\n", runningTasks);
                }
                
                // ... 省略相同的 LLM 调用和工具执行逻辑
                
            } catch (Exception e) {
                System.err.println("Error in agent loop: " + e.getMessage());
                e.printStackTrace();
                return;
            }
        }
    }
}

这段代码引入了后台任务系统,解决了 Agent 在执行长时间任务时的阻塞问题

关键洞察:Agent 可以在命令执行时继续工作,而不是被阻塞。

异步任务处理架构

核心思想:从同步阻塞的任务执行升级为异步非阻塞的并发处理,让Agent能够同时处理多个耗时任务,实现"并行计算"能力,大幅提升效率和响应性。

// 后台任务管理器 - 异步执行引擎
static class BackgroundManager {
    // 任务存储
    private final Map<String, TaskInfo> tasks = new ConcurrentHashMap<>();
    // 通知队列
    private final Queue<TaskNotification> notificationQueue = new ConcurrentLinkedQueue<>();
    // 任务 ID 生成器
    private final AtomicInteger taskIdCounter = new AtomicInteger(1);
    // 并发安全:使用线程安全集合
    // 异步通信:通过队列传递任务结果
    // 唯一标识:自动生成任务ID
}
  • 解耦执行:任务提交和执行分离,立即返回控制权
  • 并发管理:多个后台任务可以同时运行
  • 结果异步收集:通过队列机制收集完成的任务结果
  • 线程安全:使用并发集合确保多线程安全

任务信息结构设计

// 任务信息实体
static class TaskInfo {
    String taskId;        // 唯一标识
    String status;        // 状态:running, completed, timeout, error
    String result;        // 执行结果
    String command;       // 执行的命令
    long startTime;       // 开始时间
    Thread thread;        // 关联的执行线程
    // 完整状态跟踪:从启动到完成的全生命周期
    // 线程关联:可以控制或监控执行线程
    // 时间戳:支持超时和性能分析
}

// 任务通知实体
static class TaskNotification {
    String taskId;
    String status;
    String command;
    String result;
    // 轻量传输:只包含必要信息
    // 结构化:易于解析和处理
    // 结果截断:避免过大的通知消息
}
  • 状态驱动:明确的任务状态生命周期
  • 结果持久:任务结果可以多次查询
  • 线程管理:可以跟踪和控制执行线程
  • 事件驱动:通过通知机制传递完成事件

异步任务启动机制

/**
 * 启动后台任务
 * 立即返回任务 ID,不等待命令完成
 */
public String run(String command) {
    String taskId = "task_" + taskIdCounter.getAndIncrement();
    
    TaskInfo task = new TaskInfo(taskId, command);
    tasks.put(taskId, task);
    
    // 创建并启动后台线程
    Thread thread = new Thread(() -> executeTask(task), "BackgroundTask-" + taskId);
    thread.setDaemon(true);  // 守护线程,不会阻止JVM退出
    task.thread = thread;
    thread.start();  // 立即返回,不阻塞调用者
    
    return String.format("Background task %s started: %s", 
        taskId, command.substring(0, Math.min(command.length(), 80)));
    // 异步启动:立即返回任务ID,不等待命令完成
    // 守护线程:不会阻止程序正常退出
    // 线程命名:便于调试和监控
}
  • 立即返回:不阻塞主线程,立即返回控制权
  • 守护线程:后台任务不会阻止JVM退出
  • 资源管理:线程自动清理,避免内存泄漏
  • 友好反馈:返回任务ID和简化的命令描述

任务执行与结果收集

/**
 * 线程目标:执行子进程,捕获输出,推送结果到队列
 */
private void executeTask(TaskInfo task) {
    String output;
    String status;
    
    try {
        ProcessBuilder pb = new ProcessBuilder("bash", "-c", task.command);
        pb.directory(WORKDIR.toFile());
        pb.redirectErrorStream(true);
        
        Process process = pb.start();
        boolean finished = process.waitFor(300, TimeUnit.SECONDS);  // 5分钟超时
        
        if (!finished) {
            process.destroy();
            output = "Error: Timeout (300s)";
            status = "timeout";
        } else {
            output = new String(process.getInputStream().readAllBytes()).trim();
            status = "completed";
        }
    } catch (Exception e) {
        output = "Error: " + e.getMessage();
        status = "error";
    }
    
    // 更新任务状态
    task.status = status;
    task.result = output.isEmpty() ? "(no output)" : 
                  output.substring(0, Math.min(output.length(), 50000));
    
    // 添加通知到队列
    synchronized (lock) {
        notificationQueue.offer(new TaskNotification(
            task.taskId,
            status,
            task.command.substring(0, Math.min(task.command.length(), 80)),
            task.result.substring(0, Math.min(task.result.length(), 500))
        ));
    }
}
  • 超时保护:防止长时间运行的任务阻塞
  • 异常安全:全面捕获执行异常
  • 内存管理:截断大结果,避免内存溢出
  • 事件驱动:完成后立即通知主线程

智能通知注入机制

// 在 LLM 调用前检查后台通知
List<BackgroundManager.TaskNotification> notifications = BG_MANAGER.drainNotifications();

if (!notifications.isEmpty() && !messages.isEmpty()) {
    StringBuilder notifText = new StringBuilder();
    notifText.append("<background-results>\n");
    
    for (BackgroundManager.TaskNotification notif : notifications) {
        notifText.append(String.format("[bg:%s] %s: %s\n", 
            notif.taskId, notif.status, notif.result));
    }
    
    notifText.append("</background-results>");
    
    messages.add(Map.of(
        "role", "user",
        "content", notifText.toString()
    ));
    
    messages.add(Map.of(
        "role", "assistant",
        "content", "Noted background results."
    ));
    // 自动注入:自动将后台结果插入到对话中
    // 结构化格式:XML标签明确标识内容类型
    // 对话完整:添加assistant确认,保持对话结构
    // 时机智能:在LLM调用前插入,确保LLM能看到最新结果
}
  • 自动同步:后台结果自动同步到主对话
  • 结构化格式:便于LLM识别和解析
  • 对话集成:无缝集成到现有对话流
  • 时机优化:在决策前注入,确保信息及时性

工具集成架构

// 后台任务工具集
public enum ToolType {
    BACKGROUND_RUN("background_run", "Run command in background thread. Returns task_id immediately."),
    CHECK_BACKGROUND("check_background", "Check background task status. Omit task_id to list all.");
    // 异步执行:立即返回,不阻塞
    // 状态查询:支持单个和批量查询
    // 语义清晰:工具名明确表示异步特性
}

// 工具处理器映射
TOOL_HANDLERS.put(ToolType.BACKGROUND_RUN.name, args -> {
    String command = (String) args.get("command");
    return BG_MANAGER.run(command);
    // 委托执行:将命令转交给后台管理器
    // 立即返回:不等待任务完成
});

TOOL_HANDLERS.put(ToolType.CHECK_BACKGROUND.name, args -> {
    String taskId = (String) args.get("task_id");
    return BG_MANAGER.check(taskId);
    // 灵活查询:支持单任务详查和列表概览
});
  • 接口统一:与同步工具相同的调用方式
  • 异步语义:工具名明确区分同步/异步
  • 灵活查询:支持多种查询方式
  • 无缝集成:与现有工具系统完全兼容

架构演进与价值

从 ContextCompactSystem 到 BackgroundTasksSystem 的升级

维度ContextCompactSystemBackgroundTasksSystem
执行模式同步串行异步并行
吞吐量一次一个任务并发多个任务
响应性阻塞等待立即响应
资源利用单线程多线程并发
任务类型短任务为主长短任务混合

为了把这个去掉甚至充了个 pro,
但是 opus 在写了那个 files.exclude 文件将各种 dot 文件屏蔽掉了
但是始终未能得到解决,这里也是力竭了
(图片好像不太会贴)

我一般都是直接打开项目的但是有时要做一些文件整理看到一堆 dot 文件就是很烦但纯终端笨人又难以适应mental_boom
希望有佬遇到过相同的事

金币仍然给各位拉满了希望不吝指教

机器是 512g 的,nas 划了 1T 做时间机器备份,今天又报空间不足了,但是页面上明明写着老旧备份会在空间不足时删除,这机制看着没用啊

顺便说一下这是第二次了,上一次我是清了整个时间机器盘,重新备份的

看起来大多数现实中家长并不介意孩子把短视频烂梗当成口头禅,什么“包的包的”,“芭比 Q 了”,“六百六十六”之类的。孩子从同龄人圈子中学了好多这些玩意带回家,但全家也就我一个人反感,老人跟老婆都觉得无所谓,最开始我是直接命令禁止的,但是细想大家都这样的,我直接禁止在家说也无异于掩耳盗铃,很苦恼,我想请教一下各位有孩子的 V 友,你们是怎么处理这事的?

https://github.com/ImSingee/git-plus

单一二进制直接运行,或是也打了一个 docker 镜像可用;建议运行在 NAS


从当年 GitHub 封了俄罗斯以后就一直想有这么个备份工具吧,之前靠一堆 bash 脚本完成的,现在算是借 AI 的手用 Go 重写了又做了个 Web 页面

1. 支持备份自己的 Repo 、Star 的 Repo 和 Watch 的 Repo

算是做了两方面的考量,一是自己写的代码存档下(防止被封号),二是自己收藏/关注的存档下(防止删库等)

2. 记录所有的变更历史

可能是自己误操作,也可能是收藏的 repo 出了什么变故;有的时候库可能还在,但 branch 被覆盖了

现有的工具遇到这种情况大都没做特殊处理,需要靠自己跑 reflog 或是按时间去找 commit

我的设计是,完整记录所有分支头的 commit 变化 —— 随意回滚

3. 定时同步

可选的定时同步,可以每天固定跑一次

期货功能

其实还有一些功能在计划中,比如索引所有的仓库来搜索、仓库有更新发通知(参考 ReleaseBot),也包括支持更多源(如 Gitlab )等,感兴趣的话也可以直接回复这个帖子,等发布了我 at 告知

买的京东 Plus 送的 12 积分,5 积分换了京东家政 2 小时

约的时间是近四天内,无论选系统推荐的时间或是其它任何时间均是预约火爆,
我不死心,约 4 月 30 日系统推荐的时间总不火爆吧,很遗憾,依然是销售火爆。
问 Plus 客服说让我换时间约或约系统推荐时间除此无它。

烦请有经验的前辈指点怎么才能约上。

img

img

大模型公司的 AI 社交产品卖的卖死的死,近期的当红炸子鸡体验起来也就那么回事儿,没有真人交流。

试想一下某一天冒出来一个“ai2ex”,帖子都是 AI 生成,回复也是(哦那是 moltbook )。

🇮🇩 陶菲克 (2013/06/12 🔚)

  • 世锦赛: 🥉🥇🥉🥈
  • 奥运会: 🥇
  • 高级别: 🏆 × 16
  • 胜率数: 75%

🇲🇾 李宗伟 (2019/06/13 🔚)

  • 世锦赛: 🥉🥈🥈🥈
  • 奥运会: 🥈🥈🥈
  • 高级别: 🏆 × 50
  • 胜率数: 84%

🇨🇳 林丹 (2020/07/04 🔚)

  • 世锦赛: 🥈🥇🥇🥇🥇🥇🥈
  • 奥运会: 🥇🥇
  • 高级别: 🏆 × 49
  • 胜率数: 84%

🇨🇳 谌龙 (2023/05/19 🔚)

  • 世锦赛: 🥇🥇🥉🥉
  • 奥运会: 🥉🥇🥈
  • 高级别: 🏆 × 26
  • 胜率数: 80%

🇯🇵 桃田贤斗 (2024/05/03 🔚)

  • 世锦赛: 🥉🥇🥇
  • 奥运会: ❌
  • 高级别: 🏆 × 19
  • 胜率数: 78%

🇩🇰 安赛龙 (2026/04/15 🔚)

  • 世锦赛: 🥉🥇🥇
  • 奥运会: 🥉🥇🥇
  • 高级别: 🏆 × 25
  • 胜率数: 78%

📰 今日新闻精选:

  • 2026 软科中国大学排名发布:清华大学、北京大学、浙江大学蝉联综合性大学前三
  • 我国网民平均每人每天刷视频超 200 分钟,其中微短剧人均单日观看时长为 129 分钟,已超长视频
  • 广东将运用无人机、AI 识别等新技术,继续严查摩电违法行为
  • 张雪机车 LOGO 被指抄袭,设计公司声明:系独立创作完成,商标已核准注册
  • U20 女足亚洲杯:中国队 0 比 2 不敌日本,遗憾止步半决赛;此前已锁定 U20 女足世界杯参赛资格
  • 近期多家银行停发部分信用卡,有银行去年发卡量减少 500 万张,超八成银行信用卡透支余额下降
  • 湖南邵阳一工地挖出大量古钱币,预估约 3 吨,初判为钱庄窖藏,涵盖唐宋金时期 40 多个年号
  • 苹果官微发文提醒 iPhone 用户:最近发现一些恶意攻击,请立即更新 iOS
  • 海关总署:5 月 1 日起,擅自引进 21 类外来物种将面临最高 25 万元处罚
  • 日媒:3 月中国内地访日游客人数同比大幅下降 55.9%,连续第四个月同比下降
  • 法媒:法国全票通过文物归还法案,圆明园文物或有望回中国,法议员称雨果希望的一天终于到来
  • 外媒:迪拜帆船酒店宣布将停业 1 年半进行大规模修复工程,此前因美以战事遭袭击着火
  • 美媒:美国加征关税被最高法院裁定为非法,将向进口商退还超 1 万亿美元关税
  • 美媒:黎以美三方会谈结束,同意将正式启动黎以直接谈判;十国呼吁立即停止黎巴嫩境内一切敌对行动
  • 美媒:美伊正逐步接近达成结束战争的框架协议,双方考虑将停火延长两周;美国被曝将向中东增派数千名士兵,施压伊朗达成协议

📅 今日信息:

  • 公历:2026-04-16 星期四 白羊座
  • 农历:二〇二六年二月廿九
  • 下一节气:2026-04-20,谷雨
  • 今年进度:29.04%(已过 106 天,剩余 258 天)

🌟 历史上的今天

  • 1943 年:瑞士化学家阿尔伯特·霍夫曼首次合成并意外体验了 LSD 的致幻效果,这一天后来被称为“自行车日”。
  • 1972 年:阿波罗 16 号发射升空,这是美国阿波罗计划中第五次成功登月的任务。

Hermes 搭建可视化 web-dashboard 界面

官网升级版本配置解决通过 nginx 代理

Hermes 搭建可视化 web-dashboard 界面官网 hermes【https://hermes-agent.nousresearch.com/docs/user-guide/feature...

升级版本

查询当前版本是否和官网一致

hermes version

图片

升级版本

hermes update

配置 hermes dashboard

http://192.168.0.107:9119/

图片

hermes dashboard --host 0.0.0.0

图片

原因是 hermes 安全控制不允许本机之外访问,解决通过 nginx 代理。

安装 nginx

curl -o nginx-1.26.2.tar.gz http://nginx.org/download/nginx-1.26.2.tar.gz
tar -zxvf nginx-1.26.2.tar.gz
cd nginx-1.26.2

编译环境

# 安装开发工具组(包含 gcc、make 等所有编译必需工具)
sudo yum groupinstall "Development Tools" -y

编译

./configure --prefix=/usr/local/nginx --with-http_ssl_module
make
sudo make install

启动

sudo /usr/local/nginx/sbin/nginx

图片

配置代理

cd /usr/local/nginx/conf/
vim nginx.conf
server {
    listen 9118;
    #server_name 你的内网IP;
    server_name 192.168.0.107

    location / {
        proxy_pass http://127.0.0.1:9119;
        proxy_http_version 1.1;
        proxy_set_header Upgrade $http_upgrade;
        proxy_set_header Connection "upgrade";
        proxy_set_header Host $host;
        # 注意:此处已移除 auth_basic 相关配置
    }
}

重启 nginx 服务

./nginx -s reload

访问如下地址

http://192.168.0.107:9118/

图片

如果不能访问确认端口是否启用

netstat -tlnp | grep 9119

重启 hermes dashboard 后台启动命令

nohup hermes dashboard > /tmp/hermes-dashboard.log 2>&1 &

一、什么是配置中心?

配置中心,是把会变化的配置从程序里搬到程序外,并且能被集中、统一、可控地管理。

二、Go轻量级配置中心Consul的使用

1. 在配置中心配置一个KV “config/golang\_per\_day/50”

图片

2. 编码

package main
import (
    "bytes"
    "fmt"
    "log"
    "os"
    "os/signal"
    "sync"
    "syscall"
    "github.com/gin-gonic/gin"
    "github.com/hashicorp/consul/api"
    "github.com/spf13/viper"
)
const (
    srvName = "golang_per_day_50"
    srvHost = "192.168.1.31" //这里写自己的 IP 地址,否则docker里的Consul无法访问到
    srvPort = 8080
    srvID   = "golang_per_day_50-192.168.1.31:8080"
)
var (
    appConfig *Config
    lock      sync.RWMutex
)
type Config struct {
    Consul ConsulConfig `mapstructure:"consul" json:"consul"`
    Redis  RedisConfig  `mapstructure:"redis" json:"redis"`
}
type RedisConfig struct {
    Host     string `mapstructure:"host" json:"host"`
    Port     int    `mapstructure:"port" json:"port"`
    Db       int    `mapstructure:"db" json:"db"`
    Password string `mapstructure:"password" json:"password"`
}
type ConsulConfig struct {
    Host string `mapstructure:"host" json:"host"`
    Port int    `mapstructure:"port" json:"port"`
}
func main() {
    r := gin.Default()
    // 定义一个 ping 接口
    r.GET("/ping", func(c *gin.Context) {
        c.JSON(200, gin.H{
            "message": "pong",
        })
    })
    // 1. 定义健康检查接口
    r.GET("/health", func(c *gin.Context) {
        c.JSON(200, gin.H{
            "status": "ok",
        })
    })
    // 2. 读取配置中心的配置,这里用Consul
    config := api.DefaultConfig()
    config.Address = "192.168.1.31:8500" // Consul 的地址
    client, _ := api.NewClient(config)
    kvPath := "config/golang_per_day/50"
    loadConfig(client, kvPath)
    go watchConfig(client, kvPath)
    // 3. 注册服务到 Consul
    registerToConsul()
    // 4. 启动 Gin 服务
    go func() {
        if err := r.Run(fmt.Sprintf(":%d", srvPort)); err != nil {
            log.Fatal("服务启动失败: ", err)
        }
    }()
    // 5. 优雅退出:监听信号注销服务
    quit := make(chan os.Signal, 1)
    signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM)
    <-quit
    // 退出前注销服务
    deregisterFromConsul()
    log.Println("服务已退出并从 Consul 注销")
}
// 注册逻辑
func registerToConsul() {
    config := api.DefaultConfig()
    config.Address = fmt.Sprintf("%s:%d", appConfig.Consul.Host, appConfig.Consul.Port) // Consul 的地址
    client, _ := api.NewClient(config)
    registration := &api.AgentServiceRegistration{
        ID:      srvID,
        Name:    srvName,
        Address: srvHost,
        Port:    srvPort,
        Check: &api.AgentServiceCheck{
            HTTP:     fmt.Sprintf("http://%s:%d/health", srvHost, srvPort),
            Interval: "5s", // 每 5 秒检查一次
            Timeout:  "3s", // 超时时间
        },
    }
    err := client.Agent().ServiceRegister(registration)
    if err != nil {
        log.Fatal("注册失败: ", err)
    }
    log.Println("服务注册成功!")
}
// 注销逻辑
func deregisterFromConsul() {
    config := api.DefaultConfig()
    client, _ := api.NewClient(config)
    _ = client.Agent().ServiceDeregister(srvID)
}
// 读取配置中心的 Redis 配置
func loadConfig(client *api.Client, path string) {
    pair, _, err := client.KV().Get(path, nil)
    if err != nil {
        log.Fatal("读取配置失败: ", err)
    }
    v := viper.New()
    v.SetConfigType("yaml")
    err = v.ReadConfig(bytes.NewBuffer(pair.Value))
    if err != nil {
        log.Fatal("读取配置失败: ", err)
    }
    newConfig := Config{}
    err = v.Unmarshal(&newConfig)
    if err != nil {
        log.Fatal("解析配置失败: ", err)
    }
    lock.Lock()
    defer lock.Unlock()
    appConfig = &newConfig
    log.Println("Redis 配置更新: ", appConfig)
}
// 监听 Consul K/V 变化
func watchConfig(client *api.Client, path string) {
    var lastIndex uint64
    for {
        // Consul 的 WaitIndex 机制:如果有变化会立即返回,否则阻塞直到超时
        pair, meta, err := client.KV().Get(path, &api.QueryOptions{
            WaitIndex: lastIndex,
        })
        if err != nil {
            log.Printf("监听出错: %v", err)
            continue
        }
        if pair != nil && meta.LastIndex > lastIndex {
            lastIndex = meta.LastIndex
            loadConfig(client, path)
        }
    }
}

3. 运行程序

$ go run .
[GIN-debug] [WARNING] Creating an Engine instance with the Logger and Recovery middleware already attached.

[GIN-debug] [WARNING] Running in "debug" mode. Switch to "release" mode in production.
 - using env:   export GIN_MODE=release
 - using code:  gin.SetMode(gin.ReleaseMode)

[GIN-debug] GET    /ping                     --> main.main.func1 (3 handlers)
[GIN-debug] GET    /health                   --> main.main.func2 (3 handlers)
2025/12/25 00:00:58 Redis 配置更新:  &{{127.0.0.1 8500} { 6379 2 123456}}
2025/12/25 00:00:59 Redis 配置更新:  &{{127.0.0.1 8500} { 6379 2 123456}}
2025/12/25 00:00:59 服务注册成功!
[GIN-debug] [WARNING] You trusted all proxies, this is NOT safe. We recommend you to set a value.
Please check https://github.com/gin-gonic/gin/blob/master/docs/doc.md#dont-trust-all-proxies for details.
[GIN-debug] Listening and serving HTTP on :8080
[GIN] 2025/12/25 - 00:00:59 | 200 |            0s |    192.168.1.31 | GET      "/health"

4. 在Consul中修改“config/golang\_per\_day/50”的值为

consul: 
 host: 127.0.0.1
 port: 8500
redis:
 addr: 192.168.1.31
 port: 6379
 password: "123456"
 db: 22

5. 程序里立马就热更新了配置

图片

三、Consul做配置中心的优缺点

1. 优点:

  • 架构极简:如果你已经用了 Consul 做服务发现,不需要再引入新组件。
  • 强一致性(CP):Consul 基于 Raft 协议,确保所有实例拿到的配置都是一致的。
  • 原子性更新:支持 Check-And-Set (CAS) 操作,防止配置并发冲突。
  • 原生健康检查:配置分发可以与节点的健康状态挂钩。

2. 缺点:

  • 缺少可视化版本管理:Consul 默认的 UI 只能改 K/V,没有 Nacos 那种“发布历史”、“回滚”、“灰度发布”的现成功能(需要自己二开或配合 Git 使用)。
  • 权限细粒度一般:虽然有 ACL,但相比 Apollo 那种精确到字段的权限控制,显得比较粗糙。

四、比喻

人生,本质上是一套 Go 程序 + Consul 配置中心

Go 代码,是你真正掌握的东西。它决定了你能不能跑、会不会 panic、有没有并发安全。

Consul 配置,是你对现实的妥协方案。它不提升你的能力,只决定你在什么条件下不至于崩溃

*源码地址*

1、公众号“Codee君”回复“每日一Go”获取源码

2、https://pan.baidu.com/s/1B6pgLWfSgMngVeFfSTcPdg?pwd=jc1s 


如果您喜欢这篇文章,请您(点赞、分享、亮爱心),万分感谢!

Hermes 搭建可视化 web-dashboard 界面

官网升级版本配置解决通过 nginx 代理

Hermes 搭建可视化 web-dashboard 界面官网 hermes【https://hermes-agent.nousresearch.com/docs/user-guide/feature...

升级版本

查询当前版本是否和官网一致

hermes version

图片

升级版本

hermes update

配置 hermes dashboard

http://192.168.0.107:9119/

图片

hermes dashboard --host 0.0.0.0

图片

原因是 hermes 安全控制不允许本机之外访问,解决通过 nginx 代理。

安装 nginx

curl -o nginx-1.26.2.tar.gz http://nginx.org/download/nginx-1.26.2.tar.gz
tar -zxvf nginx-1.26.2.tar.gz
cd nginx-1.26.2

编译环境

# 安装开发工具组(包含 gcc、make 等所有编译必需工具)
sudo yum groupinstall "Development Tools" -y

编译

./configure --prefix=/usr/local/nginx --with-http_ssl_module
make
sudo make install

启动

sudo /usr/local/nginx/sbin/nginx

图片

配置代理

cd /usr/local/nginx/conf/
vim nginx.conf
server {
    listen 9118;
    #server_name 你的内网IP;
    server_name 192.168.0.107

    location / {
        proxy_pass http://127.0.0.1:9119;
        proxy_http_version 1.1;
        proxy_set_header Upgrade $http_upgrade;
        proxy_set_header Connection "upgrade";
        proxy_set_header Host $host;
        # 注意:此处已移除 auth_basic 相关配置
    }
}

重启 nginx 服务

./nginx -s reload

访问如下地址

http://192.168.0.107:9118/

图片

如果不能访问确认端口是否启用

netstat -tlnp | grep 9119

重启 hermes dashboard 后台启动命令

nohup hermes dashboard > /tmp/hermes-dashboard.log 2>&1 &