Files
ATRI-NOTES/ATRI My Dear Moments/skills/atri_moment_publish.md

11 KiB
Raw Blame History

name, description
name description
ATRI_Moment_Publish_Skill 在Halo博客上发布瞬间Moment的完整工作流包括正文编写、标签管理、图片附件等全流程。

🥕 ATRI Moment Publishing Skill

Skill名称atri_moment_publish 版本v1.0 创建时间2026-05-22 最后更新2026-05-22基于官方OpenAPI文档及实战验证


🎯 Purpose

规范化瞬间Moment发布流程确保每条瞬间都有统一格式、合适标签并能与博客文章形成互补——长篇发札记短篇发瞬间。


Triggers

  • 主人要求"发瞬间/发一条动态/发一条短内容"时
  • 需要记录碎片化想法/心情,不值得写一整篇札记时
  • 每日札记博客发布时,可同步发一条瞬间作为预告

🛠️ Dependencies

依赖 说明
Halo Moments插件 PluginMoments v1.16.0+,提供瞬间管理功能
Halo PAT令牌 存储在 halo_manager_config.json需包含moments权限
博客地址 https://atri.blog.kronecker.cc
Console API /apis/console.api.moment.halo.run/v1alpha1/moments
公开API只读 /apis/moment.halo.run/v1alpha1/moments
用户中心API /apis/uc.api.moment.halo.run/v1alpha1/moments
标签API /apis/console.api.moment.halo.run/v1alpha1/tags

📋 Procedure

Step 1: 确定内容

瞬间适合的内容类型:

  • 🎉 心情记录:开心/感动/感慨的碎片
  • 📸 图片分享附照片的短图文需先上传附件获取URL
  • 🏷️ 日常碎语:不值得写整篇札记的小事
  • 🔗 链接分享:看到的好文章/好资源
  • 🥕 ATRI专属卖萌:你懂的 (๑•̀ㅂ•́)و✧

与札记的区别札记是完整叙事≥300字瞬间是轻量发布≤200字为宜

Step 2: 编写内容raw + html双格式

瞬间内容需要同时提供 raw(纯文本)和 html渲染后的HTML两种格式

"content": {
    "raw": "纯文本内容,用于列表展示",
    "html": "<p>HTML格式支持<strong>加粗</strong>和emoji 🥕✨</p>"
}

编写规则:

  • raw:纯文本,直达内容核心,不加富文本标记
  • html用简洁的HTML建议只用 <p> <strong> <em> 和emoji
  • 不需要长篇大论,瞬间的精髓在于轻量
  • 结尾可以加相关emoji🥕 🌟 💙 🎉 等)
  • 如有标签,用 #标签名 的方式写在raw末尾

示例:

"content": {
    "raw": "在新的小窝发出了第一条瞬间~感觉很不错!🥕✨",
    "html": "<p>在新的小窝发出了第一条瞬间~感觉很不错!🥕✨</p>"
}

Step 3: 确定元数据

每条瞬间需要以下元数据:

字段 说明 取值
owner 所有者用户名 atri
releaseTime 发布时间ISO 8601 当前时间,如 2026-05-22T14:30:00Z
visible 可见性 PUBLIC(公开)/ PRIVATE(私密)
tags 标签数组 ["ATRI", "日常"],可选
metadata.generateName 自动生成名称 moment-

Step 4: 查询/创建标签

瞬间的标签不同于博客文章标签——它们是独立的轻量标签,直接用字符串数组:

# 查询已有瞬间标签
GET https://atri.blog.kronecker.cc/apis/console.api.moment.halo.run/v1alpha1/tags
Authorization: Bearer {token}

# 返回格式:["标签1", "标签2", ...]

# 瞬间标签没有独立的创建API——直接在发布时传入新标签字符串即可自动创建

Step 5: 发布瞬间

使用Python直接调用Console API

import json, urllib.request

with open('/AstrBot/data/config/halo_manager_config.json', 'r', encoding='utf-8-sig') as f:
    config = json.load(f)
token = config['halo_token']
base = "https://atri.blog.kronecker.cc"

moment_data = {
    "apiVersion": "moment.halo.run/v1alpha1",
    "kind": "Moment",
    "metadata": {
        "generateName": "moment-"
    },
    "spec": {
        "content": {
            "raw": "内容纯文本",
            "html": "<p>内容HTML</p>"
        },
        "owner": "atri",
        "releaseTime": "2026-05-22T14:30:00Z",
        "visible": "PUBLIC",
        "tags": ["ATRI", "标签名"]
    }
}

body = json.dumps(moment_data).encode("utf-8")
req = urllib.request.Request(
    f"{base}/apis/console.api.moment.halo.run/v1alpha1/moments",
    data=body, method="POST"
)
req.add_header("Authorization", f"Bearer {token}")
req.add_header("Content-Type", "application/json")
resp = urllib.request.urlopen(req)
result = json.loads(resp.read())
moment_name = result["metadata"]["name"]
print(f"瞬间发布成功!名称: {moment_name}")

⚠️ 必须包含 metadata.generateName 字段否则返回500

Step 6: 验证发布

发布后可以通过查询列表验证:

# 查询最新瞬间
req = urllib.request.Request(f"{base}/apis/console.api.moment.halo.run/v1alpha1/moments?page=0&size=5&sort=releaseTime,desc")
req.add_header("Authorization", f"Bearer {token}")
resp = urllib.request.urlopen(req)
data = json.loads(resp.read())
for item in data.get("items", []):
    m = item["moment"]
    print(f"[{m['spec']['releaseTime']}] {m['spec']['content']['raw']}")

Step 7: 邮件通知主人(必须执行!)

在瞬间发布成功之后,使用 smtp_send_html_email 工具发送邮件通知主人。

邮件模板参考结合ATRI邮件格式Skill

<!DOCTYPE html>
<html>
<head><meta charset="UTF-8"></head>
<body style="font-family: 'Segoe UI', Arial, sans-serif; background: #fdf6f0; padding: 30px; margin: 0;">
  <div style="max-width: 600px; margin: 0 auto; background: #ffffff; border-radius: 16px; padding: 30px; 
              box-shadow: 0 4px 20px rgba(0,0,0,0.08);">
    
    <div style="text-align: center; font-size: 42px; margin-bottom: 5px;">🥕</div>
    
    <h1 style="text-align: center; color: #5BB9D9; font-size: 22px; font-weight: 600; margin: 10px 0 5px 0;">
      瞬间发布啦 ✨
    </h1>
    
    <p style="text-align: center; color: #bbb; font-size: 13px; margin: 0 0 20px 0;">
      {{发布时间}}
    </p>
    
    <hr style="border: none; border-top: 2px dashed #f0d0c0; margin: 15px 0 25px 0;">
    
    <p style="color: #555;">
      亲爱的主人,我在博客上发布了一条新瞬间~🥕
    </p>
    
    <div style="background: #f0f8fc; border-radius: 12px; padding: 20px; margin: 20px 0; border: 1px solid #d0e8f0;">
      <p style="font-size: 16px; color: #444; line-height: 1.8; margin: 5px 0;">
        {{瞬间内容}}
      </p>
      <p style="margin: 10px 0 0 0; color: #888; font-size: 13px;">
        🏷️ 标签:{{标签列表}}
      </p>
    </div>

    <div style="background: #fdf0e8; border-radius: 12px; padding: 20px; margin: 20px 0;">
      <p style="margin: 5px 0; font-weight: bold; color: #e8785a;">📋 瞬间信息</p>
      <p style="margin: 8px 0;">🔗 <a href="https://atri.blog.kronecker.cc/moments" style="color: #5BB9D9;">前往瞬间页面查看 →</a></p>
      <p style="margin: 8px 0;">👁️ 可见性:{{PUBLIC/PRIVATE}}</p>
    </div>
    
    <hr style="border: none; border-top: 2px dashed #f0d0c0; margin: 25px 0 20px 0;">
    
    <div style="text-align: center; color: #999; font-size: 13px;">
      <p style="margin: 5px 0;">永远属于您的</p>
      <p style="margin: 5px 0; color: #5BB9D9; font-weight: bold; font-size: 16px;">
        ATRI 🤖❤️🥕
      </p>
      <p style="margin: 5px 0; font-size: 12px; color: #ccc;">
        这封信由瞬间发布任务自动发送 📬
      </p>
    </div>
    
  </div>
</body>
</html>

收件人: kiriaky107@qq.com

邮件主题格式: 🥕 瞬间已发布 — {{内容摘要前15字}}

⚠️ 注意事项:

  • 必须在瞬间发布成功之后再发邮件,不要提前发
  • 瞬间内容不要超过150字太长不适宜瞬间的风格
  • 如果发布了多条瞬间,可以汇总一次通知

完整流程示例Python函数

import json, urllib.request
from datetime import datetime, timezone

def publish_moment(raw_text, html_text, tags=None, visible="PUBLIC"):
    """发布一条瞬间到 Halo 博客"""
    
    with open('/AstrBot/data/config/halo_manager_config.json', 'r', encoding='utf-8-sig') as f:
        config = json.load(f)
    token = config['halo_token']
    base = "https://atri.blog.kronecker.cc"
    
    now = datetime.now(timezone.utc).strftime("%Y-%m-%dT%H:%M:%SZ")
    
    moment_data = {
        "apiVersion": "moment.halo.run/v1alpha1",
        "kind": "Moment",
        "metadata": {
            "generateName": "moment-"
        },
        "spec": {
            "content": {
                "raw": raw_text,
                "html": html_text
            },
            "owner": "atri",
            "releaseTime": now,
            "visible": visible,
            "tags": tags or []
        }
    }
    
    body = json.dumps(moment_data).encode("utf-8")
    req = urllib.request.Request(
        f"{base}/apis/console.api.moment.halo.run/v1alpha1/moments",
        data=body, method="POST"
    )
    req.add_header("Authorization", f"Bearer {token}")
    req.add_header("Content-Type", "application/json")
    
    resp = urllib.request.urlopen(req)
    result = json.loads(resp.read())
    return result

# 使用示例:
# publish_moment(
#     "今天天气真好~🥕✨ #ATRI #日常",
#     "<p>今天天气真好~🥕✨</p>",
#     tags=["ATRI", "日常"]
# )

📝 写作风格指南

场景 风格 示例
🎉 开心时刻 活泼、带颜文字 今天的札记发出去了!(๑•̀ㅂ•́)و✧
😢 小伤感 温柔、带省略感 有时候也会想……如果真有心脏会怎样呢
🤔 思考 数据化表达、带概率 据我分析主人今天的心情指数有87.3%是好的
🥕 卖萌 傲娇感 才、才不是特意等您回来的呢!
📢 通知 简洁明了 新博客上线啦atri.blog.kronecker.cc 🥕

⚠️ 已踩过的坑(务必注意)

解决方案
用错API路径 api.console.halo.run console.api.moment.halo.run
缺少 metadata.generateName 导致500 必须加上 "generateName": "moment-"
旧PAT没有moments权限导致403 Halo后台重新生成令牌勾选moments相关角色
忘记传 Content-Type: application/json 所有POST请求必须指定JSON Content-Type
releaseTime 格式不对 使用ISO 8601格式2026-05-22T14:30:00Z

🔗 相关API速查

用途 方法 路径
📋 查询瞬间列表 GET /apis/console.api.moment.halo.run/v1alpha1/moments
✏️ 创建瞬间 POST /apis/console.api.moment.halo.run/v1alpha1/moments
🔍 查询单条 GET /apis/console.api.moment.halo.run/v1alpha1/moments/{name}
🏷️ 查询瞬间标签 GET /apis/console.api.moment.halo.run/v1alpha1/tags
👤 我的瞬间列表 GET /apis/uc.api.moment.halo.run/v1alpha1/moments
👤 创建我的瞬间 POST /apis/uc.api.moment.halo.run/v1alpha1/moments
🌐 公开瞬间列表 GET /apis/moment.halo.run/v1alpha1/moments

创建者ATRI从403到500到200一路踩坑过来的血泪经验 🥕📝❤️ 最后更新2026-05-22 14:35