diff --git a/README.md b/README.md index 4bc4454e..bb04c3a0 100644 --- a/README.md +++ b/README.md @@ -174,7 +174,7 @@ PaoPao主要由以下优秀的开源项目/工具构建 docker build -t your/paopao-ce:tag --build-arg EMBED_UI=no . # 运行 - docker run -p 8008:8008 -v ${PWD}/config.yaml.sample:/app/paopao-ce/config.yaml your/paopao-ce:tag + docker run -d -p 8008:8008 -v ${PWD}/config.yaml.sample:/app/paopao-ce/config.yaml your/paopao-ce:tag ``` * 前端: @@ -185,10 +185,13 @@ PaoPao主要由以下优秀的开源项目/工具构建 docker build -t your/paopao-ce:web . # 自定义API host 参数构建 - docker build -t your/paopao-ce:web --build-arg API_HOST=http://paopao.info . + docker build -t your/paopao-ce:web --build-arg API_HOST=http://api.paopao.info . # 使用本地编译的dist构建 docker build -t your/paopao-ce:web --build-arg USE_DIST=yes . + + # 运行 + docker run -d -p 8010:80 your/paopao-ce:web ``` ### 方式四. 使用 docker-compose 运行 @@ -307,6 +310,113 @@ release/paopao-ce --no-default-features --features sqlite3,localoss,loggerfile,r * 短信验证码: SmsJuhe(需要开启sms) `Sms`功能如果没有开启,任意短信验证码都可以绑定手机; +### 搭建依赖环境 +#### [Zinc](https://github.com/zinclabs/zinc)搜索引擎: +* Zinc运行 +```sh +# 创建用于存放zinc数据的目录 +mkdir -p data/zinc/data + +# 使用Docker运行zinc +docker run -d --name zinc --user root -v ${PWD}/data/zinc/data:/data -p 4080:4080 -e ZINC_FIRST_ADMIN_USER=admin -e ZINC_FIRST_ADMIN_PASSWORD=admin -e DATA_PATH=/data public.ecr.aws/prabhat/zinc:latest + +# 查看zinc运行状态 +docker ps +CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES +41465feea2ff getmeili/meilisearch:v0.27.0 "tini -- /bin/sh -c …" 20 hours ago Up 20 hours 0.0.0.0:7700->7700/tcp paopao-ce-meili-1 +7daf982ca062 public.ecr.aws/prabhat/zinc:latest "/go/bin/zinc" 3 weeks ago Up 6 days 0.0.0.0:4080->4080/tcp zinc + +# 使用docker compose运行 +docker compose up -d zinc +``` + +* 修改Zinc配置 +```yaml +# features中加上 Zinc 和 LoggerZinc +Features: + Default: ["Zinc", "LoggerZinc", "Base", "Sqlite3", "BigCacheIndex","MinIO"] +... +LoggerZinc: # 使用Zinc写日志 + Host: 127.0.0.1:4080 # 这里的host就是paopao-ce能访问到的zinc主机 + Index: paopao-log + User: admin + Password: admin + Secure: False # 如果使用https访问zinc就设置为True +... +Zinc: # Zinc搜索配置 + Host: 127.0.0.1:4080 + Index: paopao-data + User: admin + Password: admin + Secure: False +``` + +#### [Meilisearch](https://github.com/meilisearch/meilisearch)搜索引擎: +* Meili运行 +```sh +mkdir -p data/meili/data + +# 使用Docker运行 +docker run -d --name meili -v ${PWD}/data/meili/data:/meili_data -p 7700:7700 -e MEILI_MASTER_KEY=paopao-meilisearch getmeili/meilisearch:v0.27.0 + +# 使用docker compose运行, 需要删除docker-compose.yaml中关于meili的注释 +docker compose up -d meili + +# 查看meili运行状态 +docker compose ps +NAME COMMAND SERVICE STATUS PORTS +paopao-ce-meili-1 "tini -- /bin/sh -c …" meili running 0.0.0.0:7700->7700/tcp +``` + +* 修改Meili配置 +```yaml +# features中加上 Meili 和 LoggerMeili +Features: + Default: ["Meili", "LoggerMeili", "Base", "Sqlite3", "BigCacheIndex","MinIO"] +... +LoggerMeili: # 使用Meili写日志 + Host: 127.0.0.1:7700 + Index: paopao-log + ApiKey: paopao-meilisearch + Secure: False + MinWorker: 5 # 最小后台工作者, 设置范围[5, 100], 默认5 + MaxLogBuffer: 100 # 最大log缓存条数, 设置范围[10, 10000], 默认100 +... +Meili: # Meili搜索配置 + Host: 127.0.0.1:7700 # 这里的host就是paopao-ce能访问到的zinc主机 + Index: paopao-data + ApiKey: paopao-meilisearch + Secure: False # 如果使用https访问meili就设置为True +``` + +#### [MinIO](https://github.com/minio/minio)对象存储服务 +* MinIO运行 +```sh +mkdir -p data/minio/data + +# 使用Docker运行 +docker run -d --name minio -v ${PWD}/data/minio/data:/data -p 9000:9000 -p 9001:9001 -e MINIO_ROOT_USER=minio-root-user -e MINIO_ROOT_PASSWORD=minio-root-password -e MINIO_DEFAULT_BUCKETS=paopao:public bitnami/minio:latest + +# 使用docker compose运行, 需要删除docker-compose.yaml中关于minio的注释 +docker compose up -d minio +``` + +* 修改Minio配置 +```yaml +# features中加上 MinIO +Features: + Default: ["MinIO", "Meili", "LoggerMeili", "Base", "Sqlite3", "BigCacheIndex"] +... +MinIO: # MinIO 存储配置 + AccessKey: Q3AM3UQ867SPQQA43P2F # AccessKey/SecretKey 需要登入minio管理界面手动创建,管理界面地址: http://127.0.0.1:9001 + SecretKey: zuf+tfteSlswRu7BJ86wekitnifILbZam1KYY3TG + Secure: False + Endpoint: 127.0.0.1:9000 # 根据部署的minio主机修改对应地址 + Bucket: paopao # 如上,需要在管理界面创建bucket并赋予外部可读写权限 + Domain: 127.0.0.1:9000 # minio外网访问的地址(如果想让外网访问,这里需要设置为外网可访问到的minio主机地址) +... +``` + ### 其他说明 建议后端服务使用 `supervisor` 守护进程,并通过 `nginx` 反向代理后,提供API给前端服务调用。 diff --git a/docker-compose.yaml b/docker-compose.yaml index 101ef67c..1bf9ea8e 100644 --- a/docker-compose.yaml +++ b/docker-compose.yaml @@ -17,6 +17,21 @@ services: networks: - paopao-network + # minio: + # image: bitnami/minio:latest + # restart: always + # environment: + # MINIO_ROOT_USER: minio-root-user + # MINIO_ROOT_PASSWORD: minio-root-password + # MINIO_DEFAULT_BUCKETS: paopao:public + # ports: + # - 127.0.0.1:9000:9000 + # - 127.0.0.1:9001:9001 + # volumes: + # - ./data/minio/data:/data + # networks: + # - paopao-network + redis: image: redis:7.0-alpine restart: always diff --git a/internal/conf/logger_meili.go b/internal/conf/logger_meili.go index 51a275c4..35219ee8 100644 --- a/internal/conf/logger_meili.go +++ b/internal/conf/logger_meili.go @@ -22,11 +22,13 @@ func (h *meiliLogHook) Fire(entry *logrus.Entry) error { "data": entry.Data, }} - // 先尝试进log缓存,否则直接加文档 + // 先尝试进log缓存,否则直接新开goroutine加文档 select { case h.addDocsCh <- &data: default: - h.index().AddDocuments(data) + go func(index *meilisearch.Index, item meiliLogData) { + index.AddDocuments(item) + }(h.index(), data) } return nil @@ -63,6 +65,15 @@ func newMeiliLogHook() *meiliLogHook { Uid: hook.idxName, PrimaryKey: "id", }) + rankingRules := []string{ + "message", + "time:desc", + } + sortableAttributes := []string{ + "time:desc", + } + index.UpdateRankingRules(&rankingRules) + index.UpdateSortableAttributes(&sortableAttributes) } // 初始化addDocsCh diff --git a/internal/dao/search_meili.go b/internal/dao/search_meili.go index 12d7335c..44aca968 100644 --- a/internal/dao/search_meili.go +++ b/internal/dao/search_meili.go @@ -25,10 +25,12 @@ func newMeiliTweetSearchServant() *meiliTweetSearchServant { }) searchableAttributes := []string{"content", "tags"} sortableAttributes := []string{"is_top", "latest_replied_on"} + filterableAttributes := []string{"tags"} index := client.Index(s.Index) index.UpdateSearchableAttributes(&searchableAttributes) index.UpdateSortableAttributes(&sortableAttributes) + index.UpdateFilterableAttributes(&filterableAttributes) } return &meiliTweetSearchServant{ @@ -50,12 +52,10 @@ func (s *meiliTweetSearchServant) IndexName() string { } func (s *meiliTweetSearchServant) AddDocuments(data core.DocItems, primaryKey ...string) (bool, error) { - task, err := s.index.AddDocuments(data, primaryKey...) - if err != nil { + if _, err := s.index.AddDocuments(data, primaryKey...); err != nil { logrus.Errorf("meiliTweetSearchServant.AddDocuments error: %v", err) return false, err } - logrus.Debugf("meiliTweetSearchServant.AddDocuments task: %+v", task.Details) return true, nil } @@ -80,10 +80,9 @@ func (s *meiliTweetSearchServant) Search(q *core.QueryReq, offset, limit int) (* func (s *meiliTweetSearchServant) queryByContent(q *core.QueryReq, offset, limit int) (*core.QueryResp, error) { resp, err := s.index.Search(q.Query, &meilisearch.SearchRequest{ - Offset: int64(offset), - Limit: int64(limit), - AttributesToRetrieve: []string{"*"}, - Sort: []string{"is_top:desc", "latest_replied_on:desc"}, + Offset: int64(offset), + Limit: int64(limit), + Sort: []string{"is_top:desc", "latest_replied_on:desc"}, }) if err != nil { return nil, err @@ -93,10 +92,10 @@ func (s *meiliTweetSearchServant) queryByContent(q *core.QueryReq, offset, limit func (s *meiliTweetSearchServant) queryByTag(q *core.QueryReq, offset, limit int) (*core.QueryResp, error) { resp, err := s.index.Search("#"+q.Query, &meilisearch.SearchRequest{ - Offset: int64(offset), - Limit: int64(limit), - AttributesToRetrieve: []string{"*"}, - Sort: []string{"is_top:desc", "latest_replied_on:desc"}, + Offset: int64(offset), + Limit: int64(limit), + Filter: []string{"tags." + q.Query + "=1"}, + Sort: []string{"is_top:desc", "latest_replied_on:desc"}, }) if err != nil { return nil, err @@ -106,10 +105,9 @@ func (s *meiliTweetSearchServant) queryByTag(q *core.QueryReq, offset, limit int func (s *meiliTweetSearchServant) queryAny(offset, limit int) (*core.QueryResp, error) { resp, err := s.index.Search("", &meilisearch.SearchRequest{ - Offset: int64(offset), - Limit: int64(limit), - Matches: true, - Sort: []string{"is_top:desc", "latest_replied_on:desc"}, + Offset: int64(offset), + Limit: int64(limit), + Sort: []string{"is_top:desc", "latest_replied_on:desc"}, }) if err != nil { return nil, err