- Added support for sorting, note type, time range, search scope, and location distance in the search feeds functionality. - Updated SearchFeedsArgs struct to include new parameters for filtering. - Modified handleSearchFeeds method to process and apply filters during feed search. - Improved logging to include the number of applied filters.
525 lines
13 KiB
Go
525 lines
13 KiB
Go
package main
|
||
|
||
import (
|
||
"context"
|
||
"encoding/json"
|
||
"fmt"
|
||
"os"
|
||
"time"
|
||
|
||
"github.com/go-rod/rod"
|
||
"github.com/mattn/go-runewidth"
|
||
"github.com/sirupsen/logrus"
|
||
"github.com/xpzouying/headless_browser"
|
||
"github.com/xpzouying/xiaohongshu-mcp/browser"
|
||
"github.com/xpzouying/xiaohongshu-mcp/configs"
|
||
"github.com/xpzouying/xiaohongshu-mcp/cookies"
|
||
"github.com/xpzouying/xiaohongshu-mcp/pkg/downloader"
|
||
"github.com/xpzouying/xiaohongshu-mcp/xiaohongshu"
|
||
)
|
||
|
||
// XiaohongshuService 小红书业务服务
|
||
type XiaohongshuService struct{}
|
||
|
||
// NewXiaohongshuService 创建小红书服务实例
|
||
func NewXiaohongshuService() *XiaohongshuService {
|
||
return &XiaohongshuService{}
|
||
}
|
||
|
||
// PublishRequest 发布请求
|
||
type PublishRequest struct {
|
||
Title string `json:"title" binding:"required"`
|
||
Content string `json:"content" binding:"required"`
|
||
Images []string `json:"images" binding:"required,min=1"`
|
||
Tags []string `json:"tags,omitempty"`
|
||
}
|
||
|
||
// LoginStatusResponse 登录状态响应
|
||
type LoginStatusResponse struct {
|
||
IsLoggedIn bool `json:"is_logged_in"`
|
||
Username string `json:"username,omitempty"`
|
||
}
|
||
|
||
// LoginQrcodeResponse 登录扫码二维码
|
||
type LoginQrcodeResponse struct {
|
||
Timeout string `json:"timeout"`
|
||
IsLoggedIn bool `json:"is_logged_in"`
|
||
Img string `json:"img,omitempty"`
|
||
}
|
||
|
||
// PublishResponse 发布响应
|
||
type PublishResponse struct {
|
||
Title string `json:"title"`
|
||
Content string `json:"content"`
|
||
Images int `json:"images"`
|
||
Status string `json:"status"`
|
||
PostID string `json:"post_id,omitempty"`
|
||
}
|
||
|
||
// PublishVideoRequest 发布视频请求(仅支持本地单个视频文件)
|
||
type PublishVideoRequest struct {
|
||
Title string `json:"title" binding:"required"`
|
||
Content string `json:"content" binding:"required"`
|
||
Video string `json:"video" binding:"required"`
|
||
Tags []string `json:"tags,omitempty"`
|
||
}
|
||
|
||
// PublishVideoResponse 发布视频响应
|
||
type PublishVideoResponse struct {
|
||
Title string `json:"title"`
|
||
Content string `json:"content"`
|
||
Video string `json:"video"`
|
||
Status string `json:"status"`
|
||
PostID string `json:"post_id,omitempty"`
|
||
}
|
||
|
||
// FeedsListResponse Feeds列表响应
|
||
type FeedsListResponse struct {
|
||
Feeds []xiaohongshu.Feed `json:"feeds"`
|
||
Count int `json:"count"`
|
||
}
|
||
|
||
// UserProfileResponse 用户主页响应
|
||
type UserProfileResponse struct {
|
||
UserBasicInfo xiaohongshu.UserBasicInfo `json:"userBasicInfo"`
|
||
Interactions []xiaohongshu.UserInteractions `json:"interactions"`
|
||
Feeds []xiaohongshu.Feed `json:"feeds"`
|
||
}
|
||
|
||
// CheckLoginStatus 检查登录状态
|
||
func (s *XiaohongshuService) CheckLoginStatus(ctx context.Context) (*LoginStatusResponse, error) {
|
||
b := newBrowser()
|
||
defer b.Close()
|
||
|
||
page := b.NewPage()
|
||
defer page.Close()
|
||
|
||
loginAction := xiaohongshu.NewLogin(page)
|
||
|
||
isLoggedIn, err := loginAction.CheckLoginStatus(ctx)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
|
||
response := &LoginStatusResponse{
|
||
IsLoggedIn: isLoggedIn,
|
||
Username: configs.Username,
|
||
}
|
||
|
||
return response, nil
|
||
}
|
||
|
||
// GetLoginQrcode 获取登录的扫码二维码
|
||
func (s *XiaohongshuService) GetLoginQrcode(ctx context.Context) (*LoginQrcodeResponse, error) {
|
||
b := newBrowser()
|
||
page := b.NewPage()
|
||
|
||
deferFunc := func() {
|
||
_ = page.Close()
|
||
b.Close()
|
||
}
|
||
|
||
loginAction := xiaohongshu.NewLogin(page)
|
||
|
||
img, loggedIn, err := loginAction.FetchQrcodeImage(ctx)
|
||
if err != nil || loggedIn {
|
||
defer deferFunc()
|
||
}
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
|
||
timeout := 4 * time.Minute
|
||
|
||
if !loggedIn {
|
||
go func() {
|
||
ctxTimeout, cancel := context.WithTimeout(context.Background(), timeout)
|
||
defer cancel()
|
||
defer deferFunc()
|
||
|
||
if loginAction.WaitForLogin(ctxTimeout) {
|
||
if er := saveCookies(page); er != nil {
|
||
logrus.Errorf("failed to save cookies: %v", er)
|
||
}
|
||
}
|
||
}()
|
||
}
|
||
|
||
return &LoginQrcodeResponse{
|
||
Timeout: func() string {
|
||
if loggedIn {
|
||
return "0s"
|
||
}
|
||
return timeout.String()
|
||
}(),
|
||
Img: img,
|
||
IsLoggedIn: loggedIn,
|
||
}, nil
|
||
}
|
||
|
||
// PublishContent 发布内容
|
||
func (s *XiaohongshuService) PublishContent(ctx context.Context, req *PublishRequest) (*PublishResponse, error) {
|
||
// 验证标题长度
|
||
// 小红书限制:最大40个单位长度
|
||
// 中文/日文/韩文占2个单位,英文/数字占1个单位
|
||
if titleWidth := runewidth.StringWidth(req.Title); titleWidth > 40 {
|
||
return nil, fmt.Errorf("标题长度超过限制")
|
||
}
|
||
|
||
// 处理图片:下载URL图片或使用本地路径
|
||
imagePaths, err := s.processImages(req.Images)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
|
||
// 构建发布内容
|
||
content := xiaohongshu.PublishImageContent{
|
||
Title: req.Title,
|
||
Content: req.Content,
|
||
Tags: req.Tags,
|
||
ImagePaths: imagePaths,
|
||
}
|
||
|
||
// 执行发布
|
||
if err := s.publishContent(ctx, content); err != nil {
|
||
logrus.Errorf("发布内容失败: title=%s %v", content.Title, err)
|
||
return nil, err
|
||
}
|
||
|
||
response := &PublishResponse{
|
||
Title: req.Title,
|
||
Content: req.Content,
|
||
Images: len(imagePaths),
|
||
Status: "发布完成",
|
||
}
|
||
|
||
return response, nil
|
||
}
|
||
|
||
// processImages 处理图片列表,支持URL下载和本地路径
|
||
func (s *XiaohongshuService) processImages(images []string) ([]string, error) {
|
||
processor := downloader.NewImageProcessor()
|
||
return processor.ProcessImages(images)
|
||
}
|
||
|
||
// publishContent 执行内容发布
|
||
func (s *XiaohongshuService) publishContent(ctx context.Context, content xiaohongshu.PublishImageContent) error {
|
||
b := newBrowser()
|
||
defer b.Close()
|
||
|
||
page := b.NewPage()
|
||
defer page.Close()
|
||
|
||
action, err := xiaohongshu.NewPublishImageAction(page)
|
||
if err != nil {
|
||
return err
|
||
}
|
||
|
||
// 执行发布
|
||
return action.Publish(ctx, content)
|
||
}
|
||
|
||
// PublishVideo 发布视频(本地文件)
|
||
func (s *XiaohongshuService) PublishVideo(ctx context.Context, req *PublishVideoRequest) (*PublishVideoResponse, error) {
|
||
// 标题长度校验
|
||
if titleWidth := runewidth.StringWidth(req.Title); titleWidth > 40 {
|
||
return nil, fmt.Errorf("标题长度超过限制")
|
||
}
|
||
|
||
// 本地视频文件校验
|
||
if req.Video == "" {
|
||
return nil, fmt.Errorf("必须提供本地视频文件")
|
||
}
|
||
if _, err := os.Stat(req.Video); err != nil {
|
||
return nil, fmt.Errorf("视频文件不存在或不可访问: %v", err)
|
||
}
|
||
|
||
// 构建发布内容
|
||
content := xiaohongshu.PublishVideoContent{
|
||
Title: req.Title,
|
||
Content: req.Content,
|
||
Tags: req.Tags,
|
||
VideoPath: req.Video,
|
||
}
|
||
|
||
// 执行发布
|
||
if err := s.publishVideo(ctx, content); err != nil {
|
||
return nil, err
|
||
}
|
||
|
||
resp := &PublishVideoResponse{
|
||
Title: req.Title,
|
||
Content: req.Content,
|
||
Video: req.Video,
|
||
Status: "发布完成",
|
||
}
|
||
return resp, nil
|
||
}
|
||
|
||
// publishVideo 执行视频发布
|
||
func (s *XiaohongshuService) publishVideo(ctx context.Context, content xiaohongshu.PublishVideoContent) error {
|
||
b := newBrowser()
|
||
defer b.Close()
|
||
|
||
page := b.NewPage()
|
||
defer page.Close()
|
||
|
||
action, err := xiaohongshu.NewPublishVideoAction(page)
|
||
if err != nil {
|
||
return err
|
||
}
|
||
|
||
return action.PublishVideo(ctx, content)
|
||
}
|
||
|
||
// ListFeeds 获取Feeds列表
|
||
func (s *XiaohongshuService) ListFeeds(ctx context.Context) (*FeedsListResponse, error) {
|
||
b := newBrowser()
|
||
defer b.Close()
|
||
|
||
page := b.NewPage()
|
||
defer page.Close()
|
||
|
||
// 创建 Feeds 列表 action
|
||
action := xiaohongshu.NewFeedsListAction(page)
|
||
|
||
// 获取 Feeds 列表
|
||
feeds, err := action.GetFeedsList(ctx)
|
||
if err != nil {
|
||
logrus.Errorf("获取 Feeds 列表失败: %v", err)
|
||
return nil, err
|
||
}
|
||
|
||
response := &FeedsListResponse{
|
||
Feeds: feeds,
|
||
Count: len(feeds),
|
||
}
|
||
|
||
return response, nil
|
||
}
|
||
|
||
func (s *XiaohongshuService) SearchFeeds(ctx context.Context, keyword string, filters ...xiaohongshu.FilterOption) (*FeedsListResponse, error) {
|
||
b := newBrowser()
|
||
defer b.Close()
|
||
|
||
page := b.NewPage()
|
||
defer page.Close()
|
||
|
||
action := xiaohongshu.NewSearchAction(page)
|
||
|
||
feeds, err := action.Search(ctx, keyword, filters...)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
|
||
response := &FeedsListResponse{
|
||
Feeds: feeds,
|
||
Count: len(feeds),
|
||
}
|
||
|
||
return response, nil
|
||
}
|
||
|
||
// GetFeedDetail 获取Feed详情
|
||
func (s *XiaohongshuService) GetFeedDetail(ctx context.Context, feedID, xsecToken string) (*FeedDetailResponse, error) {
|
||
b := newBrowser()
|
||
defer b.Close()
|
||
|
||
page := b.NewPage()
|
||
defer page.Close()
|
||
|
||
// 创建 Feed 详情 action
|
||
action := xiaohongshu.NewFeedDetailAction(page)
|
||
|
||
// 获取 Feed 详情
|
||
result, err := action.GetFeedDetail(ctx, feedID, xsecToken)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
|
||
response := &FeedDetailResponse{
|
||
FeedID: feedID,
|
||
Data: result,
|
||
}
|
||
|
||
return response, nil
|
||
}
|
||
|
||
// UserProfile 获取用户信息
|
||
func (s *XiaohongshuService) UserProfile(ctx context.Context, userID, xsecToken string) (*UserProfileResponse, error) {
|
||
b := newBrowser()
|
||
defer b.Close()
|
||
|
||
page := b.NewPage()
|
||
defer page.Close()
|
||
|
||
action := xiaohongshu.NewUserProfileAction(page)
|
||
|
||
result, err := action.UserProfile(ctx, userID, xsecToken)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
response := &UserProfileResponse{
|
||
UserBasicInfo: result.UserBasicInfo,
|
||
Interactions: result.Interactions,
|
||
Feeds: result.Feeds,
|
||
}
|
||
|
||
return response, nil
|
||
|
||
}
|
||
|
||
// PostCommentToFeed 发表评论到Feed
|
||
func (s *XiaohongshuService) PostCommentToFeed(ctx context.Context, feedID, xsecToken, content string) (*PostCommentResponse, error) {
|
||
b := newBrowser()
|
||
defer b.Close()
|
||
|
||
page := b.NewPage()
|
||
defer page.Close()
|
||
|
||
action := xiaohongshu.NewCommentFeedAction(page)
|
||
|
||
if err := action.PostComment(ctx, feedID, xsecToken, content); err != nil {
|
||
return nil, err
|
||
}
|
||
|
||
return &PostCommentResponse{FeedID: feedID, Success: true, Message: "评论发表成功"}, nil
|
||
}
|
||
|
||
// LikeFeed 点赞笔记
|
||
func (s *XiaohongshuService) LikeFeed(ctx context.Context, feedID, xsecToken string) (*ActionResult, error) {
|
||
b := newBrowser()
|
||
defer b.Close()
|
||
|
||
page := b.NewPage()
|
||
defer page.Close()
|
||
|
||
action := xiaohongshu.NewLikeAction(page)
|
||
if err := action.Like(ctx, feedID, xsecToken); err != nil {
|
||
return nil, err
|
||
}
|
||
return &ActionResult{FeedID: feedID, Success: true, Message: "点赞成功或已点赞"}, nil
|
||
}
|
||
|
||
// UnlikeFeed 取消点赞笔记
|
||
func (s *XiaohongshuService) UnlikeFeed(ctx context.Context, feedID, xsecToken string) (*ActionResult, error) {
|
||
b := newBrowser()
|
||
defer b.Close()
|
||
|
||
page := b.NewPage()
|
||
defer page.Close()
|
||
|
||
action := xiaohongshu.NewLikeAction(page)
|
||
if err := action.Unlike(ctx, feedID, xsecToken); err != nil {
|
||
return nil, err
|
||
}
|
||
return &ActionResult{FeedID: feedID, Success: true, Message: "取消点赞成功或未点赞"}, nil
|
||
}
|
||
|
||
// FavoriteFeed 收藏笔记
|
||
func (s *XiaohongshuService) FavoriteFeed(ctx context.Context, feedID, xsecToken string) (*ActionResult, error) {
|
||
b := newBrowser()
|
||
defer b.Close()
|
||
|
||
page := b.NewPage()
|
||
defer page.Close()
|
||
|
||
action := xiaohongshu.NewFavoriteAction(page)
|
||
if err := action.Favorite(ctx, feedID, xsecToken); err != nil {
|
||
return nil, err
|
||
}
|
||
return &ActionResult{FeedID: feedID, Success: true, Message: "收藏成功或已收藏"}, nil
|
||
}
|
||
|
||
// UnfavoriteFeed 取消收藏笔记
|
||
func (s *XiaohongshuService) UnfavoriteFeed(ctx context.Context, feedID, xsecToken string) (*ActionResult, error) {
|
||
b := newBrowser()
|
||
defer b.Close()
|
||
|
||
page := b.NewPage()
|
||
defer page.Close()
|
||
|
||
action := xiaohongshu.NewFavoriteAction(page)
|
||
if err := action.Unfavorite(ctx, feedID, xsecToken); err != nil {
|
||
return nil, err
|
||
}
|
||
return &ActionResult{FeedID: feedID, Success: true, Message: "取消收藏成功或未收藏"}, nil
|
||
}
|
||
|
||
// ReplyCommentToFeed 回复指定评论
|
||
func (s *XiaohongshuService) ReplyCommentToFeed(ctx context.Context, feedID, xsecToken, commentID, userID, content string) (*ReplyCommentResponse, error) {
|
||
b := newBrowser()
|
||
defer b.Close()
|
||
|
||
page := b.NewPage()
|
||
defer page.Close()
|
||
|
||
action := xiaohongshu.NewCommentFeedAction(page)
|
||
|
||
if err := action.ReplyToComment(ctx, feedID, xsecToken, commentID, userID, content); err != nil {
|
||
return nil, err
|
||
}
|
||
|
||
return &ReplyCommentResponse{
|
||
FeedID: feedID,
|
||
TargetCommentID: commentID,
|
||
TargetUserID: userID,
|
||
Success: true,
|
||
Message: "评论回复成功",
|
||
}, nil
|
||
}
|
||
|
||
func newBrowser() *headless_browser.Browser {
|
||
return browser.NewBrowser(configs.IsHeadless(), browser.WithBinPath(configs.GetBinPath()))
|
||
}
|
||
|
||
func saveCookies(page *rod.Page) error {
|
||
cks, err := page.Browser().GetCookies()
|
||
if err != nil {
|
||
return err
|
||
}
|
||
|
||
data, err := json.Marshal(cks)
|
||
if err != nil {
|
||
return err
|
||
}
|
||
|
||
cookieLoader := cookies.NewLoadCookie(cookies.GetCookiesFilePath())
|
||
return cookieLoader.SaveCookies(data)
|
||
}
|
||
|
||
// withBrowserPage 执行需要浏览器页面的操作的通用函数
|
||
func withBrowserPage(fn func(*rod.Page) error) error {
|
||
b := newBrowser()
|
||
defer b.Close()
|
||
|
||
page := b.NewPage()
|
||
defer page.Close()
|
||
|
||
return fn(page)
|
||
}
|
||
|
||
// GetMyProfile 获取当前登录用户的个人信息
|
||
func (s *XiaohongshuService) GetMyProfile(ctx context.Context) (*UserProfileResponse, error) {
|
||
var result *xiaohongshu.UserProfileResponse
|
||
var err error
|
||
|
||
err = withBrowserPage(func(page *rod.Page) error {
|
||
action := xiaohongshu.NewUserProfileAction(page)
|
||
result, err = action.GetMyProfileViaSidebar(ctx)
|
||
return err
|
||
})
|
||
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
|
||
response := &UserProfileResponse{
|
||
UserBasicInfo: result.UserBasicInfo,
|
||
Interactions: result.Interactions,
|
||
Feeds: result.Feeds,
|
||
}
|
||
|
||
return response, nil
|
||
}
|