mirror of
				https://github.com/openimsdk/open-im-server.git
				synced 2025-10-27 05:52:29 +08:00 
			
		
		
		
	Merge remote-tracking branch 'origin/main' into js-server
This commit is contained in:
		
						commit
						3d47897ba3
					
				
							
								
								
									
										8
									
								
								.env
									
									
									
									
									
								
							
							
						
						
									
										8
									
								
								.env
									
									
									
									
									
								
							| @ -8,12 +8,12 @@ PROMETHEUS_IMAGE=prom/prometheus:v2.45.6 | ||||
| ALERTMANAGER_IMAGE=prom/alertmanager:v0.27.0 | ||||
| GRAFANA_IMAGE=grafana/grafana:11.0.1 | ||||
| 
 | ||||
| OPENIM_WEB_FRONT_IMAGE=openim/openim-web-front:release-v3.8.0 | ||||
| OPENIM_ADMIN_FRONT_IMAGE=openim/openim-admin-front:release-v1.8.0 | ||||
| OPENIM_WEB_FRONT_IMAGE=openim/openim-web-front:release-v3.8.1 | ||||
| OPENIM_ADMIN_FRONT_IMAGE=openim/openim-admin-front:release-v1.8.2 | ||||
| 
 | ||||
| #FRONT_IMAGE: use aliyun images | ||||
| #OPENIM_WEB_FRONT_IMAGE=registry.cn-hangzhou.aliyuncs.com/openimsdk/openim-web-front:release-v3.5.1 | ||||
| #OPENIM_ADMIN_FRONT_IMAGE=registry.cn-hangzhou.aliyuncs.com/openimsdk/openim-admin-front:release-v1.7 | ||||
| #OPENIM_WEB_FRONT_IMAGE=registry.cn-hangzhou.aliyuncs.com/openimsdk/openim-web-front:release-v3.8.1 | ||||
| #OPENIM_ADMIN_FRONT_IMAGE=registry.cn-hangzhou.aliyuncs.com/openimsdk/openim-admin-front:release-v1.8.2 | ||||
| 
 | ||||
| DATA_DIR=./ | ||||
| 
 | ||||
|  | ||||
							
								
								
									
										59
									
								
								.github/workflows/go-build-test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										59
									
								
								.github/workflows/go-build-test.yml
									
									
									
									
										vendored
									
									
								
							| @ -89,6 +89,65 @@ jobs: | ||||
|           mage start | ||||
|           mage check | ||||
| 
 | ||||
|   go-test: | ||||
|     name: Benchmark Test with go ${{ matrix.go_version }} on ${{ matrix.os }} | ||||
|     runs-on: ${{ matrix.os }} | ||||
|     permissions: | ||||
|       contents: write | ||||
|     env: | ||||
|       SDK_DIR: openim-sdk-core | ||||
|       CONFIG_PATH: config/notification.yml | ||||
|     #   pull-requests: write | ||||
|     strategy: | ||||
|       matrix: | ||||
|         os: [ ubuntu-latest ] | ||||
|         go_version: [ "1.22.x" ] | ||||
| 
 | ||||
|     steps: | ||||
|       - name: Checkout Server repository | ||||
|         uses: actions/checkout@v4 | ||||
| 
 | ||||
|       - name: Checkout SDK repository | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           repository: 'openimsdk/openim-sdk-core' | ||||
|           path: ${{ env.SDK_DIR }} | ||||
| 
 | ||||
|       - name: Set up Go ${{ matrix.go_version }} | ||||
|         uses: actions/setup-go@v5 | ||||
|         with: | ||||
|           go-version: ${{ matrix.go_version }} | ||||
| 
 | ||||
|       - name: Get Server dependencies | ||||
|         run: | | ||||
|           go install github.com/magefile/mage@latest | ||||
|           go mod download | ||||
| 
 | ||||
|       - name: Install yq | ||||
|         run: | | ||||
|           sudo wget https://github.com/mikefarah/yq/releases/download/v4.34.1/yq_linux_amd64 -O /usr/bin/yq | ||||
|           sudo chmod +x /usr/bin/yq | ||||
| 
 | ||||
|       - name: Modify Server Configuration | ||||
|         run: | | ||||
|           yq e '.groupCreated.unreadCount = true' -i ${{ env.CONFIG_PATH }} | ||||
|           yq e '.friendApplicationApproved.unreadCount = true' -i ${{ env.CONFIG_PATH }} | ||||
| 
 | ||||
|       - name: Start Server Services | ||||
|         run: | | ||||
|           docker compose up -d | ||||
|           mage build | ||||
|           mage start | ||||
|           mage check | ||||
| 
 | ||||
|       - name: Build test SDK core | ||||
|         run: | | ||||
|           cd ${{ env.SDK_DIR }} | ||||
|           go mod tidy | ||||
|           cd integration_test | ||||
|           mkdir data | ||||
|           go run main.go -lgr 0.8 -imf -crg -ckgn -ckcon -sem -ckmsn -u 20 -su 5 -lg 2 -cg 2 -cgm 3 -sm 10 -gm 10 -reg | ||||
| 
 | ||||
|   dockerfile-test: | ||||
|     name: Build and Test Dockerfile | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
							
								
								
									
										81
									
								
								.github/workflows/publish-docker-image.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										81
									
								
								.github/workflows/publish-docker-image.yml
									
									
									
									
										vendored
									
									
								
							| @ -4,6 +4,8 @@ on: | ||||
|   push: | ||||
|     branches: | ||||
|       - release-* | ||||
|     # tags: | ||||
|     #   - 'v*' | ||||
| 
 | ||||
|   release: | ||||
|     types: [published] | ||||
| @ -15,11 +17,8 @@ on: | ||||
|         required: true | ||||
|         default: "v3.8.0" | ||||
| 
 | ||||
| # env: | ||||
| #   GO_VERSION: "1.21" | ||||
| 
 | ||||
| jobs: | ||||
|   publish-docker-images: | ||||
|   build-and-test: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
| @ -32,12 +31,18 @@ jobs: | ||||
|       - name: Set up Docker Buildx | ||||
|         uses: docker/setup-buildx-action@v3 | ||||
| 
 | ||||
|       - name: Build and push Docker image | ||||
|       - name: Build Docker image | ||||
|         id: build | ||||
|         uses: docker/build-push-action@v5 | ||||
|         with: | ||||
|           context: ./main-repo | ||||
|           load: true | ||||
|           tags: "openim/openim-server:local" | ||||
|           cache-from: type=gha | ||||
|           cache-to: type=gha,mode=max | ||||
| 
 | ||||
|       - name: Save Docker image to file | ||||
|         run: docker save -o image.tar openim/openim-server:local | ||||
| 
 | ||||
|       - name: Checkout compose repository | ||||
|         uses: actions/checkout@v4 | ||||
| @ -63,38 +68,39 @@ jobs: | ||||
|           docker compose up -d | ||||
|           sleep 60 | ||||
| 
 | ||||
|       - name: Check openim-server health | ||||
|         run: | | ||||
|           timeout=300 | ||||
|           interval=30 | ||||
|           elapsed=0 | ||||
|           while [[ $elapsed -le $timeout ]]; do | ||||
|             if ! docker exec openim-server mage check; then | ||||
|               echo "openim-server is not ready, waiting..." | ||||
|               sleep $interval | ||||
|               elapsed=$(($elapsed + $interval)) | ||||
|             else | ||||
|               echo "Health check successful" | ||||
|               exit 0 | ||||
|             fi | ||||
|           done | ||||
|           echo "Health check failed after 5 minutes" | ||||
|           exit 1 | ||||
|       # - name: Check openim-server health | ||||
|       #   run: | | ||||
|       #     timeout=300 | ||||
|       #     interval=30 | ||||
|       #     elapsed=0 | ||||
|       #     while [[ $elapsed -le $timeout ]]; do | ||||
|       #       if ! docker exec openim-server mage check; then | ||||
|       #         echo "openim-server is not ready, waiting..." | ||||
|       #         sleep $interval | ||||
|       #         elapsed=$(($elapsed + $interval)) | ||||
|       #       else | ||||
|       #         echo "Health check successful" | ||||
|       #         exit 0 | ||||
|       #       fi | ||||
|       #     done | ||||
|       #     echo "Health check failed after 5 minutes" | ||||
|       #     exit 1 | ||||
| 
 | ||||
|       - name: Check openim-chat health | ||||
|         if: success() | ||||
|         run: | | ||||
|           if ! docker exec openim-chat mage check; then | ||||
|               echo "openim-chat check failed" | ||||
|               exit 1 | ||||
|             else | ||||
|               echo "Health check successful" | ||||
|               exit 0 | ||||
|             fi | ||||
|       # - name: Check openim-chat health | ||||
|       #   if: success() | ||||
|       #   run: | | ||||
|       #     if ! docker exec openim-chat mage check; then | ||||
|       #         echo "openim-chat check failed" | ||||
|       #         exit 1 | ||||
|       #       else | ||||
|       #         echo "Health check successful" | ||||
|       #         exit 0 | ||||
|       #       fi | ||||
| 
 | ||||
|       - name: Load Docker image from file | ||||
|         run: docker load -i image.tar | ||||
| 
 | ||||
|       - name: Extract metadata for Docker #  (tags, labels)  | ||||
|         if: success() | ||||
|       - name: Extract metadata for Docker (tags, labels) | ||||
|         id: meta | ||||
|         uses: docker/metadata-action@v5.5.1 | ||||
|         with: | ||||
| @ -102,18 +108,17 @@ jobs: | ||||
|             openim/openim-server | ||||
|             ghcr.io/openimsdk/openim-server | ||||
|             registry.cn-hangzhou.aliyuncs.com/openimsdk/openim-server | ||||
| 
 | ||||
|           # generate Docker tags based on the following events/attributes | ||||
|           tags: | | ||||
|             type=ref,event=tag | ||||
|             type=schedule | ||||
|             type=ref,event=branch | ||||
|             type=ref,event=pr | ||||
|             type=semver,pattern={{version}} | ||||
|             type=semver,pattern=v{{version}} | ||||
|             type=semver,pattern={{major}}.{{minor}} | ||||
|             type=semver,pattern={{major}} | ||||
|             type=semver,pattern=release-{{raw}} | ||||
|             type=sha | ||||
|             type=raw,value=${{ github.event.inputs.tag }} | ||||
| 
 | ||||
|       - name: Log in to Docker Hub | ||||
|         uses: docker/login-action@v2 | ||||
| @ -135,7 +140,7 @@ jobs: | ||||
|           username: ${{ secrets.ALIREGISTRY_USERNAME }} | ||||
|           password: ${{ secrets.ALIREGISTRY_TOKEN }} | ||||
| 
 | ||||
|       - name: Build and push Docker images | ||||
|       - name: Push Docker images | ||||
|         uses: docker/build-push-action@v5 | ||||
|         with: | ||||
|           context: ./main-repo | ||||
|  | ||||
| @ -1,5 +1,5 @@ | ||||
| # Use Go 1.21 Alpine as the base image for building the application | ||||
| FROM golang:1.21-alpine as builder | ||||
| FROM golang:1.21-alpine AS builder | ||||
| 
 | ||||
| # Define the base directory for the application as an environment variable | ||||
| ENV SERVER_DIR=/openim-server | ||||
|  | ||||
| @ -15,10 +15,9 @@ | ||||
| package main | ||||
| 
 | ||||
| import ( | ||||
| 	_ "net/http/pprof" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/cmd" | ||||
| 	"github.com/openimsdk/tools/system/program" | ||||
| 	_ "net/http/pprof" | ||||
| ) | ||||
| 
 | ||||
| func main() { | ||||
|  | ||||
| @ -5,9 +5,4 @@ etcd: | ||||
|   username: '' | ||||
|   password: '' | ||||
| 
 | ||||
| zookeeper: | ||||
|   schema: openim | ||||
|   address: [ localhost:12181 ] | ||||
|   username: '' | ||||
|   password: '' | ||||
| 
 | ||||
|  | ||||
| @ -54,7 +54,7 @@ | ||||
|   "liveNow": false, | ||||
|   "panels": [ | ||||
|     { | ||||
|       "collapsed": true, | ||||
|       "collapsed": false, | ||||
|       "gridPos": { | ||||
|         "h": 1, | ||||
|         "w": 24, | ||||
| @ -62,7 +62,10 @@ | ||||
|         "y": 0 | ||||
|       }, | ||||
|       "id": 35, | ||||
|       "panels": [ | ||||
|       "panels": [], | ||||
|       "title": "Server", | ||||
|       "type": "row" | ||||
|     }, | ||||
|     { | ||||
|       "datasource": { | ||||
|         "type": "prometheus", | ||||
| @ -115,7 +118,8 @@ | ||||
|             "mode": "absolute", | ||||
|             "steps": [ | ||||
|               { | ||||
|                     "color": "green" | ||||
|                 "color": "green", | ||||
|                 "value": null | ||||
|               }, | ||||
|               { | ||||
|                 "color": "red", | ||||
| @ -220,7 +224,8 @@ | ||||
|             "mode": "absolute", | ||||
|             "steps": [ | ||||
|               { | ||||
|                     "color": "green" | ||||
|                 "color": "green", | ||||
|                 "value": null | ||||
|               }, | ||||
|               { | ||||
|                 "color": "red", | ||||
| @ -355,7 +360,8 @@ | ||||
|             "mode": "absolute", | ||||
|             "steps": [ | ||||
|               { | ||||
|                     "color": "green" | ||||
|                 "color": "green", | ||||
|                 "value": null | ||||
|               }, | ||||
|               { | ||||
|                 "color": "red", | ||||
| @ -477,7 +483,8 @@ | ||||
|             "mode": "absolute", | ||||
|             "steps": [ | ||||
|               { | ||||
|                     "color": "green" | ||||
|                 "color": "green", | ||||
|                 "value": null | ||||
|               }, | ||||
|               { | ||||
|                 "color": "red", | ||||
| @ -595,7 +602,8 @@ | ||||
|             "mode": "absolute", | ||||
|             "steps": [ | ||||
|               { | ||||
|                     "color": "green" | ||||
|                 "color": "green", | ||||
|                 "value": null | ||||
|               }, | ||||
|               { | ||||
|                 "color": "red", | ||||
| @ -745,7 +753,8 @@ | ||||
|             "mode": "absolute", | ||||
|             "steps": [ | ||||
|               { | ||||
|                     "color": "green" | ||||
|                 "color": "green", | ||||
|                 "value": null | ||||
|               }, | ||||
|               { | ||||
|                 "color": "red", | ||||
| @ -776,8 +785,8 @@ | ||||
|       }, | ||||
|       "gridPos": { | ||||
|         "h": 11, | ||||
|             "w": 6, | ||||
|             "x": 4, | ||||
|         "w": 8, | ||||
|         "x": 0, | ||||
|         "y": 33 | ||||
|       }, | ||||
|       "id": 42, | ||||
| @ -807,7 +816,7 @@ | ||||
|           "hide": false, | ||||
|           "instant": false, | ||||
|           "interval": "", | ||||
|               "legendFormat": "failed msgs", | ||||
|           "legendFormat": "addr:{{instance}}", | ||||
|           "range": true, | ||||
|           "refId": "A" | ||||
|         } | ||||
| @ -867,7 +876,8 @@ | ||||
|             "mode": "absolute", | ||||
|             "steps": [ | ||||
|               { | ||||
|                     "color": "green" | ||||
|                 "color": "green", | ||||
|                 "value": null | ||||
|               }, | ||||
|               { | ||||
|                 "color": "red", | ||||
| @ -898,8 +908,8 @@ | ||||
|       }, | ||||
|       "gridPos": { | ||||
|         "h": 11, | ||||
|             "w": 6, | ||||
|             "x": 14, | ||||
|         "w": 8, | ||||
|         "x": 8, | ||||
|         "y": 33 | ||||
|       }, | ||||
|       "id": 43, | ||||
| @ -929,7 +939,7 @@ | ||||
|           "hide": false, | ||||
|           "instant": false, | ||||
|           "interval": "", | ||||
|               "legendFormat": "failed addr: {{instance}}", | ||||
|           "legendFormat": "addr: {{instance}}", | ||||
|           "range": true, | ||||
|           "refId": "A" | ||||
|         } | ||||
| @ -937,6 +947,129 @@ | ||||
|       "title": "Seq Set Failed Num", | ||||
|       "type": "timeseries" | ||||
|     }, | ||||
|     { | ||||
|       "datasource": { | ||||
|         "type": "prometheus", | ||||
|         "uid": "${DS_PROMETHEUS}" | ||||
|       }, | ||||
|       "description": "This metric represents the number of messages that take a long time to send.", | ||||
|       "fieldConfig": { | ||||
|         "defaults": { | ||||
|           "color": { | ||||
|             "mode": "palette-classic" | ||||
|           }, | ||||
|           "custom": { | ||||
|             "axisBorderShow": false, | ||||
|             "axisCenteredZero": false, | ||||
|             "axisColorMode": "text", | ||||
|             "axisLabel": "", | ||||
|             "axisPlacement": "auto", | ||||
|             "barAlignment": 0, | ||||
|             "drawStyle": "line", | ||||
|             "fillOpacity": 0, | ||||
|             "gradientMode": "none", | ||||
|             "hideFrom": { | ||||
|               "legend": false, | ||||
|               "tooltip": false, | ||||
|               "viz": false | ||||
|             }, | ||||
|             "insertNulls": false, | ||||
|             "lineInterpolation": "linear", | ||||
|             "lineStyle": { | ||||
|               "fill": "solid" | ||||
|             }, | ||||
|             "lineWidth": 1, | ||||
|             "pointSize": 5, | ||||
|             "scaleDistribution": { | ||||
|               "type": "linear" | ||||
|             }, | ||||
|             "showPoints": "auto", | ||||
|             "spanNulls": false, | ||||
|             "stacking": { | ||||
|               "group": "A", | ||||
|               "mode": "none" | ||||
|             }, | ||||
|             "thresholdsStyle": { | ||||
|               "mode": "off" | ||||
|             } | ||||
|           }, | ||||
|           "fieldMinMax": false, | ||||
|           "mappings": [], | ||||
|           "thresholds": { | ||||
|             "mode": "absolute", | ||||
|             "steps": [ | ||||
|               { | ||||
|                 "color": "green", | ||||
|                 "value": null | ||||
|               }, | ||||
|               { | ||||
|                 "color": "red", | ||||
|                 "value": 80 | ||||
|               } | ||||
|             ] | ||||
|           }, | ||||
|           "unit": "none" | ||||
|         }, | ||||
|         "overrides": [ | ||||
|           { | ||||
|             "matcher": { | ||||
|               "id": "byName", | ||||
|               "options": "failed msgs" | ||||
|             }, | ||||
|             "properties": [ | ||||
|               { | ||||
|                 "id": "color", | ||||
|                 "value": { | ||||
|                   "fixedColor": "dark-red", | ||||
|                   "mode": "fixed", | ||||
|                   "seriesBy": "last" | ||||
|                 } | ||||
|               } | ||||
|             ] | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|       "gridPos": { | ||||
|         "h": 11, | ||||
|         "w": 8, | ||||
|         "x": 16, | ||||
|         "y": 33 | ||||
|       }, | ||||
|       "id": 60, | ||||
|       "options": { | ||||
|         "legend": { | ||||
|           "calcs": [], | ||||
|           "displayMode": "list", | ||||
|           "placement": "bottom", | ||||
|           "showLegend": true | ||||
|         }, | ||||
|         "tooltip": { | ||||
|           "maxHeight": 600, | ||||
|           "mode": "single", | ||||
|           "sort": "none" | ||||
|         } | ||||
|       }, | ||||
|       "targets": [ | ||||
|         { | ||||
|           "datasource": { | ||||
|             "type": "prometheus", | ||||
|             "uid": "${DS_PROMETHEUS}" | ||||
|           }, | ||||
|           "editorMode": "code", | ||||
|           "exemplar": false, | ||||
|           "expr": "msg_long_time_push_total", | ||||
|           "format": "time_series", | ||||
|           "hide": false, | ||||
|           "instant": false, | ||||
|           "interval": "", | ||||
|           "legendFormat": "addr:{{instance}}", | ||||
|           "range": true, | ||||
|           "refId": "A" | ||||
|         } | ||||
|       ], | ||||
|       "title": "Long Time Send Msg Total", | ||||
|       "type": "timeseries" | ||||
|     }, | ||||
|     { | ||||
|       "datasource": { | ||||
|         "type": "prometheus", | ||||
| @ -989,7 +1122,8 @@ | ||||
|             "mode": "absolute", | ||||
|             "steps": [ | ||||
|               { | ||||
|                     "color": "green" | ||||
|                 "color": "green", | ||||
|                 "value": null | ||||
|               }, | ||||
|               { | ||||
|                 "color": "red", | ||||
| @ -1107,7 +1241,8 @@ | ||||
|             "mode": "absolute", | ||||
|             "steps": [ | ||||
|               { | ||||
|                     "color": "green" | ||||
|                 "color": "green", | ||||
|                 "value": null | ||||
|               }, | ||||
|               { | ||||
|                 "color": "red", | ||||
| @ -1172,10 +1307,6 @@ | ||||
|       ], | ||||
|       "title": "Msg Failed Insert Num", | ||||
|       "type": "timeseries" | ||||
|         } | ||||
|       ], | ||||
|       "title": "Server", | ||||
|       "type": "row" | ||||
|     }, | ||||
|     { | ||||
|       "collapsed": true, | ||||
| @ -1183,7 +1314,7 @@ | ||||
|         "h": 1, | ||||
|         "w": 24, | ||||
|         "x": 0, | ||||
|         "y": 1 | ||||
|         "y": 54 | ||||
|       }, | ||||
|       "id": 22, | ||||
|       "panels": [ | ||||
| @ -1973,7 +2104,7 @@ | ||||
|         "h": 1, | ||||
|         "w": 24, | ||||
|         "x": 0, | ||||
|         "y": 2 | ||||
|         "y": 55 | ||||
|       }, | ||||
|       "id": 28, | ||||
|       "panels": [ | ||||
| @ -2827,7 +2958,7 @@ | ||||
|         "h": 1, | ||||
|         "w": 24, | ||||
|         "x": 0, | ||||
|         "y": 3 | ||||
|         "y": 56 | ||||
|       }, | ||||
|       "id": 25, | ||||
|       "panels": [ | ||||
| @ -3377,18 +3508,15 @@ | ||||
|       "type": "row" | ||||
|     }, | ||||
|     { | ||||
|       "collapsed": false, | ||||
|       "collapsed": true, | ||||
|       "gridPos": { | ||||
|         "h": 1, | ||||
|         "w": 24, | ||||
|         "x": 0, | ||||
|         "y": 4 | ||||
|         "y": 57 | ||||
|       }, | ||||
|       "id": 6, | ||||
|       "panels": [], | ||||
|       "title": "Process", | ||||
|       "type": "row" | ||||
|     }, | ||||
|       "panels": [ | ||||
|         { | ||||
|           "datasource": { | ||||
|             "type": "prometheus", | ||||
| @ -4062,8 +4190,7 @@ | ||||
|                 "mode": "absolute", | ||||
|                 "steps": [ | ||||
|                   { | ||||
|                 "color": "green", | ||||
|                 "value": null | ||||
|                     "color": "green" | ||||
|                   }, | ||||
|                   { | ||||
|                     "color": "red", | ||||
| @ -4166,8 +4293,7 @@ | ||||
|                 "mode": "absolute", | ||||
|                 "steps": [ | ||||
|                   { | ||||
|                 "color": "green", | ||||
|                 "value": null | ||||
|                     "color": "green" | ||||
|                   }, | ||||
|                   { | ||||
|                     "color": "red", | ||||
| @ -4220,6 +4346,10 @@ | ||||
|           ], | ||||
|           "title": "Resident Memory bytes", | ||||
|           "type": "timeseries" | ||||
|         } | ||||
|       ], | ||||
|       "title": "Process", | ||||
|       "type": "row" | ||||
|     }, | ||||
|     { | ||||
|       "collapsed": true, | ||||
| @ -4227,7 +4357,7 @@ | ||||
|         "h": 1, | ||||
|         "w": 24, | ||||
|         "x": 0, | ||||
|         "y": 49 | ||||
|         "y": 58 | ||||
|       }, | ||||
|       "id": 3, | ||||
|       "panels": [ | ||||
| @ -5441,6 +5571,6 @@ | ||||
|   "timezone": "", | ||||
|   "title": "Demo", | ||||
|   "uid": "a506d250-b606-4702-86a7-ac6aa1d069a1", | ||||
|   "version": 23, | ||||
|   "version": 2, | ||||
|   "weekStart": "" | ||||
| } | ||||
| @ -14,12 +14,16 @@ toRedisTopic: toRedis | ||||
| toMongoTopic: toMongo | ||||
| # Kafka topic for push notifications | ||||
| toPushTopic: toPush | ||||
| # Kafka topic for offline push notifications | ||||
| toOfflinePushTopic: toOfflinePush | ||||
| # Consumer group ID for Redis topic | ||||
| toRedisGroupID: redis | ||||
| # Consumer group ID for MongoDB topic | ||||
| toMongoGroupID: mongo | ||||
| # Consumer group ID for push notifications topic | ||||
| toPushGroupID: push | ||||
| # Consumer group ID for offline push notifications topic | ||||
| toOfflinePushGroupID: offlinePush | ||||
| # TLS (Transport Layer Security) configuration | ||||
| tls: | ||||
|   # Enable or disable TLS | ||||
|  | ||||
| @ -3,11 +3,14 @@ api: | ||||
|   listenIP: 0.0.0.0 | ||||
|   # Listening ports; if multiple are configured, multiple instances will be launched, must be consistent with the number of prometheus.ports | ||||
|   ports: [ 10002 ] | ||||
|   # API compression level; 0: default compression, 1: best compression, 2: best speed, -1: no compression | ||||
|   compressionLevel: 0 | ||||
| 
 | ||||
| 
 | ||||
| prometheus: | ||||
|   # Whether to enable prometheus | ||||
|   enable: true | ||||
|   # Prometheus listening ports, must match the number of api.ports | ||||
|   ports: [ 20502 ] | ||||
|   ports: [ 12002 ] | ||||
|   # This address can be accessed via a browser | ||||
|   grafanaURL: http://127.0.0.1:13000/ | ||||
|  | ||||
| @ -2,13 +2,13 @@ rpc: | ||||
|   # The IP address where this RPC service registers itself; if left blank, it defaults to the internal network IP | ||||
|   registerIP:  | ||||
|   # List of ports that the RPC service listens on; configuring multiple ports will launch multiple instances. These must match the number of configured prometheus ports | ||||
|   ports: [ 10140 ] | ||||
|   ports: [ 10140, 10141, 10142, 10143, 10144, 10145, 10146, 10147, 10148, 10149, 10150, 10151, 10152, 10153, 10154, 10155 ] | ||||
| 
 | ||||
| prometheus: | ||||
|   # Enable or disable Prometheus monitoring | ||||
|   enable: true | ||||
|   # List of ports that Prometheus listens on; these must match the number of rpc.ports to ensure correct monitoring setup | ||||
|   ports: [ 20640 ] | ||||
|   ports: [ 12140, 12141, 12142, 12143, 12144, 12145, 12146, 12147, 12148, 12149, 12150, 12151, 12152, 12153, 12154, 12155 ] | ||||
| 
 | ||||
| # IP address that the RPC/WebSocket service listens on; setting to 0.0.0.0 listens on both internal and external IPs. If left blank, it automatically uses the internal network IP | ||||
| listenIP: 0.0.0.0 | ||||
| @ -23,8 +23,4 @@ longConnSvr: | ||||
|   # WebSocket connection handshake timeout in seconds | ||||
|   websocketTimeout: 10 | ||||
| 
 | ||||
| # 1: For Android, iOS, Windows, Mac, and web platforms, only one instance can be online at a time | ||||
| multiLoginPolicy: 1 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
|  | ||||
| @ -3,4 +3,4 @@ prometheus: | ||||
|   enable: true | ||||
|   # List of ports that Prometheus listens on; each port corresponds to an instance of monitoring. Ensure these are managed accordingly | ||||
|   # Because four instances have been launched, four ports need to be specified | ||||
|   ports: [ 20600, 20601, 20602, 20603 ] | ||||
|   ports: [ 12020, 12021, 12022, 12023, 12024, 12025, 12026, 12027, 12028, 12029, 12030, 12031, 12032, 12033, 12034, 12035 ] | ||||
|  | ||||
| @ -4,13 +4,13 @@ rpc: | ||||
|   # IP address that the RPC service listens on; setting to 0.0.0.0 listens on both internal and external IPs. If left blank, it automatically uses the internal network IP | ||||
|   listenIP: 0.0.0.0 | ||||
|   # List of ports that the RPC service listens on; configuring multiple ports will launch multiple instances. These must match the number of configured prometheus ports | ||||
|   ports: [ 10170, 10171, 10172, 10173 ] | ||||
|   ports: [ 10170, 10171, 10172, 10173, 10174, 10175, 10176, 10177, 10178, 10179, 10180, 10181, 10182, 10183, 10184, 10185 ] | ||||
| 
 | ||||
| prometheus: | ||||
|   # Enable or disable Prometheus monitoring | ||||
|   enable: true | ||||
|   # List of ports that Prometheus listens on; these must match the number of rpc.ports to ensure correct monitoring setup | ||||
|   ports: [ 20670, 20671, 20672, 20673 ] | ||||
|   ports: [ 12170, 12171, 12172, 12173, 12174, 12175, 12176, 12177, 12178, 12179, 12180, 12182, 12183, 12184, 12185, 12186 ] | ||||
| 
 | ||||
| maxConcurrentWorkers: 3 | ||||
| #Use geTui for offline push notifications, or choose fcm or jpns; corresponding configuration settings must be specified. | ||||
| @ -38,9 +38,4 @@ iosPush: | ||||
|       badgeCount: true | ||||
|       production: false | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| fullUserCache: true | ||||
|  | ||||
| @ -4,15 +4,14 @@ rpc: | ||||
|   # IP address that the RPC service listens on; setting to 0.0.0.0 listens on both internal and external IPs. If left blank, it automatically uses the internal network IP | ||||
|   listenIP: 0.0.0.0 | ||||
|   # List of ports that the RPC service listens on; configuring multiple ports will launch multiple instances. These must match the number of configured prometheus ports | ||||
|   ports: [ 10160 ] | ||||
|   ports: [ 10200 ] | ||||
| 
 | ||||
| prometheus: | ||||
|   # Enable or disable Prometheus monitoring | ||||
|   enable: true | ||||
|   # List of ports that Prometheus listens on; these must match the number of rpc.ports to ensure correct monitoring setup | ||||
|   ports: [ 20660 ] | ||||
|   ports: [ 12200 ] | ||||
| 
 | ||||
| tokenPolicy: | ||||
|   # Token validity period, in days | ||||
|   expire: 90 | ||||
| 
 | ||||
|  | ||||
| @ -4,10 +4,10 @@ rpc: | ||||
|   # IP address that the RPC service listens on; setting to 0.0.0.0 listens on both internal and external IPs. If left blank, it automatically uses the internal network IP | ||||
|   listenIP: 0.0.0.0 | ||||
|   # List of ports that the RPC service listens on; configuring multiple ports will launch multiple instances. These must match the number of configured prometheus ports | ||||
|   ports: [ 10180 ] | ||||
|   ports: [ 10220 ] | ||||
| 
 | ||||
| prometheus: | ||||
|   # Enable or disable Prometheus monitoring | ||||
|   enable: true | ||||
|   # List of ports that Prometheus listens on; these must match the number of rpc.ports to ensure correct monitoring setup | ||||
|   ports: [ 20680 ] | ||||
|   ports: [ 12220 ] | ||||
|  | ||||
| @ -4,10 +4,10 @@ rpc: | ||||
|   # IP address that the RPC service listens on; setting to 0.0.0.0 listens on both internal and external IPs. If left blank, it automatically uses the internal network IP | ||||
|   listenIP: 0.0.0.0 | ||||
|   # List of ports that the RPC service listens on; configuring multiple ports will launch multiple instances. These must match the number of configured prometheus ports | ||||
|   ports: [ 10120 ] | ||||
|   ports: [ 10240 ] | ||||
| 
 | ||||
| prometheus: | ||||
|   # Enable or disable Prometheus monitoring | ||||
|   enable: true | ||||
|   # List of ports that Prometheus listens on; these must match the number of rpc.ports to ensure correct monitoring setup | ||||
|   ports: [ 20620 ] | ||||
|   ports: [ 12240 ] | ||||
|  | ||||
| @ -4,13 +4,13 @@ rpc: | ||||
|   # IP address that the RPC service listens on; setting to 0.0.0.0 listens on both internal and external IPs. If left blank, it automatically uses the internal network IP | ||||
|   listenIP: 0.0.0.0 | ||||
|   # List of ports that the RPC service listens on; configuring multiple ports will launch multiple instances. These must match the number of configured prometheus ports | ||||
|   ports: [ 10150 ] | ||||
|   ports: [ 10260 ] | ||||
| 
 | ||||
| prometheus: | ||||
|   # Enable or disable Prometheus monitoring | ||||
|   enable: true | ||||
|   # List of ports that Prometheus listens on; these must match the number of rpc.ports to ensure correct monitoring setup | ||||
|   ports: [ 20650 ] | ||||
|   ports: [ 12260 ] | ||||
| 
 | ||||
| 
 | ||||
| enableHistoryForNewMembers: true | ||||
| @ -4,17 +4,14 @@ rpc: | ||||
|   # IP address that the RPC service listens on; setting to 0.0.0.0 listens on both internal and external IPs. If left blank, it automatically uses the internal network IP | ||||
|   listenIP: 0.0.0.0 | ||||
|   # List of ports that the RPC service listens on; configuring multiple ports will launch multiple instances. These must match the number of configured prometheus ports | ||||
|   ports: [ 10130 ] | ||||
|   ports: [ 10280 ] | ||||
| 
 | ||||
| prometheus: | ||||
|   # Enable or disable Prometheus monitoring | ||||
|   enable: true | ||||
|   # List of ports that Prometheus listens on; these must match the number of rpc.ports to ensure correct monitoring setup | ||||
|   ports: [ 20630 ] | ||||
|   ports: [ 12280 ] | ||||
| 
 | ||||
| 
 | ||||
| # Does sending messages require friend verification | ||||
| friendVerify: false | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
|  | ||||
| @ -4,13 +4,13 @@ rpc: | ||||
|   # IP address that the RPC service listens on; setting to 0.0.0.0 listens on both internal and external IPs. If left blank, it automatically uses the internal network IP | ||||
|   listenIP: 0.0.0.0 | ||||
|   # List of ports that the RPC service listens on; configuring multiple ports will launch multiple instances. These must match the number of configured prometheus ports | ||||
|   ports: [ 10190 ] | ||||
|   ports: [ 10300 ] | ||||
| 
 | ||||
| prometheus: | ||||
|   # Enable or disable Prometheus monitoring | ||||
|   enable: true | ||||
|   # List of ports that Prometheus listens on; these must match the number of rpc.ports to ensure correct monitoring setup | ||||
|   ports: [ 20690 ] | ||||
|   ports: [ 12300 ] | ||||
| 
 | ||||
| 
 | ||||
| object: | ||||
|  | ||||
| @ -4,14 +4,10 @@ rpc: | ||||
|   # Listening IP; 0.0.0.0 means both internal and external IPs are listened to, if blank, the internal network IP is automatically obtained by default | ||||
|   listenIP: 0.0.0.0 | ||||
|   # Listening ports; if multiple are configured, multiple instances will be launched, and must be consistent with the number of prometheus.ports | ||||
|   ports: [ 10110 ] | ||||
|   ports: [ 10320 ] | ||||
| 
 | ||||
| prometheus: | ||||
|   # Whether to enable prometheus | ||||
|   enable: true | ||||
|   # Prometheus listening ports, must be consistent with the number of rpc.ports | ||||
|   ports: [ 20610 ] | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
|   ports: [ 12320 ] | ||||
|  | ||||
| @ -28,56 +28,59 @@ scrape_configs: | ||||
|       - targets: [ internal_ip:20500 ] | ||||
|   - job_name: openimserver-openim-api | ||||
|     static_configs: | ||||
|       - targets: [ internal_ip:20502 ] | ||||
|       - targets: [ internal_ip:12002 ] | ||||
|         labels: | ||||
|           namespace: default | ||||
|   - job_name: openimserver-openim-msggateway | ||||
|     static_configs: | ||||
|       - targets: [ internal_ip:20640 ] | ||||
|       - targets: [ internal_ip:12140 ] | ||||
| #      - targets: [ internal_ip:12140, internal_ip:12141, internal_ip:12142, internal_ip:12143, internal_ip:12144, internal_ip:12145, internal_ip:12146, internal_ip:12147, internal_ip:12148, internal_ip:12149, internal_ip:12150, internal_ip:12151, internal_ip:12152, internal_ip:12153, internal_ip:12154, internal_ip:12155 ] | ||||
|         labels: | ||||
|           namespace: default | ||||
|   - job_name: openimserver-openim-msgtransfer | ||||
|     static_configs: | ||||
|       - targets: [ internal_ip:20600, internal_ip:20601, internal_ip:20602, internal_ip:20603 ] | ||||
|       - targets: [ internal_ip:12020, internal_ip:12021, internal_ip:12022, internal_ip:12023, internal_ip:12024, internal_ip:12025, internal_ip:12026, internal_ip:12027 ] | ||||
| #      - targets: [ internal_ip:12020, internal_ip:12021, internal_ip:12022, internal_ip:12023, internal_ip:12024, internal_ip:12025, internal_ip:12026, internal_ip:12027, internal_ip:12028, internal_ip:12029, internal_ip:12030, internal_ip:12031, internal_ip:12032, internal_ip:12033, internal_ip:12034, internal_ip:12035 ] | ||||
|         labels: | ||||
|           namespace: default | ||||
|   - job_name: openimserver-openim-push | ||||
|     static_configs: | ||||
|       - targets: [ internal_ip:20670, internal_ip:20671, internal_ip:20672, internal_ip:20673] | ||||
|       - targets: [ internal_ip:12170, internal_ip:12171, internal_ip:12172, internal_ip:12173, internal_ip:12174, internal_ip:12175, internal_ip:12176, internal_ip:12177 ] | ||||
| #      - targets: [ internal_ip:12170, internal_ip:12171, internal_ip:12172, internal_ip:12173, internal_ip:12174, internal_ip:12175, internal_ip:12176, internal_ip:12177, internal_ip:12178, internal_ip:12179, internal_ip:12180,  internal_ip:12182, internal_ip:12183, internal_ip:12184, internal_ip:12185, internal_ip:12186 ] | ||||
|         labels: | ||||
|           namespace: default | ||||
|   - job_name: openimserver-openim-rpc-auth | ||||
|     static_configs: | ||||
|       - targets: [ internal_ip:20600 ] | ||||
|       - targets: [ internal_ip:12200 ] | ||||
|         labels: | ||||
|           namespace: default | ||||
|   - job_name: openimserver-openim-rpc-conversation | ||||
|     static_configs: | ||||
|       - targets: [ internal_ip:20680 ] | ||||
|       - targets: [ internal_ip:12220 ] | ||||
|         labels: | ||||
|           namespace: default | ||||
|   - job_name: openimserver-openim-rpc-friend | ||||
|     static_configs: | ||||
|       - targets: [ internal_ip:20620 ] | ||||
|       - targets: [ internal_ip:12240 ] | ||||
|         labels: | ||||
|           namespace: default | ||||
|   - job_name: openimserver-openim-rpc-group | ||||
|     static_configs: | ||||
|       - targets: [ internal_ip:20650 ] | ||||
|       - targets: [ internal_ip:12260 ] | ||||
|         labels: | ||||
|           namespace: default | ||||
|   - job_name: openimserver-openim-rpc-msg | ||||
|     static_configs: | ||||
|       - targets: [ internal_ip:20630 ] | ||||
|       - targets: [ internal_ip:12280 ] | ||||
|         labels: | ||||
|           namespace: default | ||||
|   - job_name: openimserver-openim-rpc-third | ||||
|     static_configs: | ||||
|       - targets: [ internal_ip:20690 ] | ||||
|       - targets: [ internal_ip:12300 ] | ||||
|         labels: | ||||
|           namespace: default | ||||
|   - job_name: openimserver-openim-rpc-user | ||||
|     static_configs: | ||||
|       - targets: [ internal_ip:20610 ] | ||||
|       - targets: [ internal_ip:12320 ] | ||||
|         labels: | ||||
|           namespace: default | ||||
| @ -4,3 +4,4 @@ password: openIM123 | ||||
| clusterMode: false | ||||
| db: 0 | ||||
| maxRetry: 10 | ||||
| poolSize: 100 | ||||
|  | ||||
| @ -12,3 +12,18 @@ rpcRegisterName: | ||||
| 
 | ||||
| imAdminUserID: [ imAdmin ] | ||||
| 
 | ||||
| # 1: For Android, iOS, Windows, Mac, and web platforms, only one instance can be online at a time | ||||
| multiLogin: | ||||
|   policy: 1 | ||||
|   maxNumOneEnd: 30 | ||||
|   customizeLoginNum: | ||||
|     ios: 1 | ||||
|     android: 1 | ||||
|     windows: 1 | ||||
|     osx: 1 | ||||
|     web: 1 | ||||
|     miniWeb: 1 | ||||
|     linux: 1 | ||||
|     aPad: 1 | ||||
|     iPad: 1 | ||||
|     admin: 1 | ||||
|  | ||||
| @ -1,8 +1,18 @@ | ||||
| url: webhook://127.0.0.1:10008/callbackExample | ||||
| url: http://127.0.0.1:10006/callbackExample | ||||
| beforeSendSingleMsg: | ||||
|   enable: false | ||||
|   timeout: 5 | ||||
|   failedContinue: true | ||||
|   # Only the contentType in allowedTypes will send the callback. | ||||
|   # Supports two formats: a single type or a range. The range is defined by the lower and upper bounds connected with a hyphen ("-"). | ||||
|   # e.g. allowedTypes: [1, 100, 200-500, 600-700] means that only contentType within the range | ||||
|   # {1, 100} ∪ [200, 500] ∪ [600, 700] will be allowed through the filter. | ||||
|   # If not set, all contentType messages will through this filter. | ||||
|   allowedTypes: [] | ||||
|   # Only the contentType not in deniedTypes will send the callback. | ||||
|   # Supports two formats, same as allowedTypes. | ||||
|   # If not set, all contentType messages will through this filter. | ||||
|   deniedTypes: [] | ||||
| beforeUpdateUserInfoEx: | ||||
|   enable:  false | ||||
|   timeout: 5 | ||||
| @ -16,17 +26,29 @@ afterSendSingleMsg: | ||||
|   # Only the senID/recvID specified in attentionIds will send the callback | ||||
|   # if not set, all user messages will be callback | ||||
|   attentionIds: [] | ||||
|   # See beforeSendSingleMsg comment. | ||||
|   allowedTypes: [] | ||||
|   deniedTypes: [] | ||||
| beforeSendGroupMsg: | ||||
|   enable: false | ||||
|   timeout: 5 | ||||
|   failedContinue: true | ||||
|   # See beforeSendSingleMsg comment. | ||||
|   allowedTypes: [] | ||||
|   deniedTypes: [] | ||||
| beforeMsgModify: | ||||
|   enable: false | ||||
|   timeout: 5 | ||||
|   failedContinue: true | ||||
|   # See beforeSendSingleMsg comment. | ||||
|   allowedTypes: [] | ||||
|   deniedTypes: [] | ||||
| afterSendGroupMsg: | ||||
|   enable: false | ||||
|   timeout: 5 | ||||
|   # See beforeSendSingleMsg comment. | ||||
|   allowedTypes: [] | ||||
|   deniedTypes: [] | ||||
| afterUserOnline: | ||||
|   enable: false | ||||
|   timeout: 5 | ||||
| @ -130,10 +152,10 @@ beforeSetGroupInfo: | ||||
|   enable: false | ||||
|   timeout: 5 | ||||
|   failedContinue: true | ||||
| afterSetGroupInfoEX: | ||||
| afterSetGroupInfoEx: | ||||
|   enable: false | ||||
|   timeout: 5 | ||||
| beforeSetGroupInfoEX: | ||||
| beforeSetGroupInfoEx: | ||||
|   enable: false | ||||
|   timeout: 5 | ||||
|   failedContinue: true | ||||
| @ -151,6 +173,9 @@ beforeAddFriendAgree: | ||||
|   enable: false | ||||
|   timeout: 5 | ||||
|   failedContinue: true | ||||
| afterAddFriendAgree: | ||||
|   enable: false | ||||
|   timeout: 5 | ||||
| afterDeleteFriend: | ||||
|   enable: false | ||||
|   timeout: 5 | ||||
|  | ||||
| @ -1,5 +1,3 @@ | ||||
| version: '3' | ||||
| 
 | ||||
| networks: | ||||
|   openim: | ||||
|     driver: bridge | ||||
| @ -45,19 +43,6 @@ services: | ||||
|     networks: | ||||
|       - openim | ||||
| 
 | ||||
|   zookeeper: | ||||
|     image: "${ZOOKEEPER_IMAGE}" | ||||
|     container_name: zookeeper | ||||
|     ports: | ||||
|       - "12181:2181" | ||||
|     environment: | ||||
|       #JVMFLAGS: "-Xms32m -Xmx128m" | ||||
|       TZ: "Asia/Shanghai" | ||||
|       ALLOW_ANONYMOUS_LOGIN: "yes" | ||||
|     restart: always | ||||
|     networks: | ||||
|       - openim | ||||
| 
 | ||||
|   etcd: | ||||
|     image: "${ETCD_IMAGE}" | ||||
|     container_name: etcd | ||||
| @ -144,6 +129,7 @@ services: | ||||
| #    image: ${PROMETHEUS_IMAGE} | ||||
| #    container_name: prometheus | ||||
| #    restart: always | ||||
| #    user: root | ||||
| #    volumes: | ||||
| #      - ./config/prometheus.yml:/etc/prometheus/prometheus.yml | ||||
| #      - ./config/instance-down-rules.yml:/etc/prometheus/instance-down-rules.yml | ||||
| @ -186,4 +172,3 @@ services: | ||||
| #    networks: | ||||
| #      - openim | ||||
| 
 | ||||
| 
 | ||||
|  | ||||
							
								
								
									
										79
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										79
									
								
								go.mod
									
									
									
									
									
								
							| @ -3,24 +3,24 @@ module github.com/openimsdk/open-im-server/v3 | ||||
| go 1.21.2 | ||||
| 
 | ||||
| require ( | ||||
| 	firebase.google.com/go v3.13.0+incompatible | ||||
| 	firebase.google.com/go/v4 v4.14.1 | ||||
| 	github.com/dtm-labs/rockscache v0.1.1 | ||||
| 	github.com/gin-gonic/gin v1.9.1 | ||||
| 	github.com/go-playground/validator/v10 v10.18.0 | ||||
| 	github.com/go-playground/validator/v10 v10.20.0 | ||||
| 	github.com/gogo/protobuf v1.3.2 // indirect | ||||
| 	github.com/golang-jwt/jwt/v4 v4.5.0 | ||||
| 	github.com/gorilla/websocket v1.5.1 | ||||
| 	github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0 | ||||
| 	github.com/mitchellh/mapstructure v1.5.0 | ||||
| 	github.com/openimsdk/protocol v0.0.72-alpha.9 | ||||
| 	github.com/openimsdk/tools v0.0.49-alpha.55 | ||||
| 	github.com/openimsdk/protocol v0.0.72-alpha.46 | ||||
| 	github.com/openimsdk/tools v0.0.50-alpha.16 | ||||
| 	github.com/pkg/errors v0.9.1 // indirect | ||||
| 	github.com/prometheus/client_golang v1.18.0 | ||||
| 	github.com/stretchr/testify v1.9.0 | ||||
| 	go.mongodb.org/mongo-driver v1.14.0 | ||||
| 	google.golang.org/api v0.165.0 | ||||
| 	google.golang.org/grpc v1.62.1 | ||||
| 	google.golang.org/protobuf v1.33.0 | ||||
| 	google.golang.org/api v0.170.0 | ||||
| 	google.golang.org/grpc v1.66.2 | ||||
| 	google.golang.org/protobuf v1.34.2 | ||||
| 	gopkg.in/yaml.v3 v3.0.1 | ||||
| ) | ||||
| 
 | ||||
| @ -29,6 +29,7 @@ require github.com/google/uuid v1.6.0 | ||||
| require ( | ||||
| 	github.com/IBM/sarama v1.43.0 | ||||
| 	github.com/fatih/color v1.14.1 | ||||
| 	github.com/gin-contrib/gzip v1.0.1 | ||||
| 	github.com/go-redis/redis v6.15.9+incompatible | ||||
| 	github.com/go-redis/redismock/v9 v9.2.0 | ||||
| 	github.com/hashicorp/golang-lru/v2 v2.0.7 | ||||
| @ -42,17 +43,17 @@ require ( | ||||
| 	github.com/stathat/consistent v1.0.0 | ||||
| 	go.uber.org/automaxprocs v1.5.3 | ||||
| 	golang.org/x/exp v0.0.0-20230905200255-921286631fa9 | ||||
| 	golang.org/x/sync v0.6.0 | ||||
| 	golang.org/x/sync v0.8.0 | ||||
| ) | ||||
| 
 | ||||
| require ( | ||||
| 	cloud.google.com/go v0.112.0 // indirect | ||||
| 	cloud.google.com/go/compute v1.23.3 // indirect | ||||
| 	cloud.google.com/go/compute/metadata v0.2.3 // indirect | ||||
| 	cloud.google.com/go/firestore v1.14.0 // indirect | ||||
| 	cloud.google.com/go/iam v1.1.5 // indirect | ||||
| 	cloud.google.com/go/longrunning v0.5.4 // indirect | ||||
| 	cloud.google.com/go/storage v1.36.0 // indirect | ||||
| 	cloud.google.com/go v0.112.1 // indirect | ||||
| 	cloud.google.com/go/compute/metadata v0.3.0 // indirect | ||||
| 	cloud.google.com/go/firestore v1.15.0 // indirect | ||||
| 	cloud.google.com/go/iam v1.1.7 // indirect | ||||
| 	cloud.google.com/go/longrunning v0.5.5 // indirect | ||||
| 	cloud.google.com/go/storage v1.40.0 // indirect | ||||
| 	github.com/MicahParks/keyfunc v1.9.0 // indirect | ||||
| 	github.com/aliyun/aliyun-oss-go-sdk v3.0.2+incompatible // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2 v1.23.1 // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2/aws/protocol/eventstream v1.5.1 // indirect | ||||
| @ -73,10 +74,12 @@ require ( | ||||
| 	github.com/aws/aws-sdk-go-v2/service/sts v1.25.4 // indirect | ||||
| 	github.com/aws/smithy-go v1.17.0 // indirect | ||||
| 	github.com/beorn7/perks v1.0.1 // indirect | ||||
| 	github.com/bytedance/sonic v1.9.1 // indirect | ||||
| 	github.com/cespare/xxhash/v2 v2.2.0 // indirect | ||||
| 	github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 // indirect | ||||
| 	github.com/bytedance/sonic v1.11.6 // indirect | ||||
| 	github.com/bytedance/sonic/loader v0.1.1 // indirect | ||||
| 	github.com/cespare/xxhash/v2 v2.3.0 // indirect | ||||
| 	github.com/clbanning/mxj v1.8.4 // indirect | ||||
| 	github.com/cloudwego/base64x v0.1.4 // indirect | ||||
| 	github.com/cloudwego/iasm v0.2.0 // indirect | ||||
| 	github.com/coreos/go-semver v0.3.0 // indirect | ||||
| 	github.com/coreos/go-systemd/v22 v22.3.2 // indirect | ||||
| 	github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc // indirect | ||||
| @ -100,7 +103,7 @@ require ( | ||||
| 	github.com/google/go-querystring v1.1.0 // indirect | ||||
| 	github.com/google/s2a-go v0.1.7 // indirect | ||||
| 	github.com/googleapis/enterprise-certificate-proxy v0.3.2 // indirect | ||||
| 	github.com/googleapis/gax-go/v2 v2.12.0 // indirect | ||||
| 	github.com/googleapis/gax-go/v2 v2.12.3 // indirect | ||||
| 	github.com/hashicorp/errwrap v1.1.0 // indirect | ||||
| 	github.com/hashicorp/go-multierror v1.1.1 // indirect | ||||
| 	github.com/hashicorp/go-uuid v1.0.3 // indirect | ||||
| @ -117,7 +120,7 @@ require ( | ||||
| 	github.com/json-iterator/go v1.1.12 // indirect | ||||
| 	github.com/kelindar/simd v1.1.2 // indirect | ||||
| 	github.com/klauspost/compress v1.17.7 // indirect | ||||
| 	github.com/klauspost/cpuid/v2 v2.2.6 // indirect | ||||
| 	github.com/klauspost/cpuid/v2 v2.2.7 // indirect | ||||
| 	github.com/leodido/go-urn v1.4.0 // indirect | ||||
| 	github.com/lestrrat-go/strftime v1.0.6 // indirect | ||||
| 	github.com/lithammer/shortuuid v3.0.0+incompatible // indirect | ||||
| @ -132,7 +135,7 @@ require ( | ||||
| 	github.com/modern-go/reflect2 v1.0.2 // indirect | ||||
| 	github.com/montanaflynn/stats v0.0.0-20171201202039-1bf9dbcd8cbe // indirect | ||||
| 	github.com/mozillazg/go-httpheader v0.4.0 // indirect | ||||
| 	github.com/pelletier/go-toml/v2 v2.1.0 // indirect | ||||
| 	github.com/pelletier/go-toml/v2 v2.2.2 // indirect | ||||
| 	github.com/pierrec/lz4/v4 v4.1.21 // indirect | ||||
| 	github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 // indirect | ||||
| 	github.com/prometheus/client_model v0.5.0 // indirect | ||||
| @ -162,24 +165,24 @@ require ( | ||||
| 	go.etcd.io/etcd/client/pkg/v3 v3.5.13 // indirect | ||||
| 	go.etcd.io/etcd/client/v3 v3.5.13 // indirect | ||||
| 	go.opencensus.io v0.24.0 // indirect | ||||
| 	go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.47.0 // indirect | ||||
| 	go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.47.0 // indirect | ||||
| 	go.opentelemetry.io/otel v1.23.0 // indirect | ||||
| 	go.opentelemetry.io/otel/metric v1.23.0 // indirect | ||||
| 	go.opentelemetry.io/otel/trace v1.23.0 // indirect | ||||
| 	go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.49.0 // indirect | ||||
| 	go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.49.0 // indirect | ||||
| 	go.opentelemetry.io/otel v1.24.0 // indirect | ||||
| 	go.opentelemetry.io/otel/metric v1.24.0 // indirect | ||||
| 	go.opentelemetry.io/otel/trace v1.24.0 // indirect | ||||
| 	go.uber.org/atomic v1.9.0 // indirect | ||||
| 	go.uber.org/multierr v1.11.0 // indirect | ||||
| 	golang.org/x/arch v0.3.0 // indirect | ||||
| 	golang.org/x/arch v0.7.0 // indirect | ||||
| 	golang.org/x/image v0.15.0 // indirect | ||||
| 	golang.org/x/net v0.22.0 // indirect | ||||
| 	golang.org/x/oauth2 v0.17.0 // indirect | ||||
| 	golang.org/x/sys v0.19.0 // indirect | ||||
| 	golang.org/x/text v0.14.0 // indirect | ||||
| 	golang.org/x/net v0.29.0 // indirect | ||||
| 	golang.org/x/oauth2 v0.21.0 // indirect | ||||
| 	golang.org/x/sys v0.25.0 // indirect | ||||
| 	golang.org/x/text v0.18.0 // indirect | ||||
| 	golang.org/x/time v0.5.0 // indirect | ||||
| 	google.golang.org/appengine v1.6.8 // indirect | ||||
| 	google.golang.org/genproto v0.0.0-20240125205218-1f4bbc51befe // indirect | ||||
| 	google.golang.org/genproto/googleapis/api v0.0.0-20240123012728-ef4313101c80 // indirect | ||||
| 	google.golang.org/genproto/googleapis/rpc v0.0.0-20240318140521-94a12d6c2237 // indirect | ||||
| 	google.golang.org/appengine/v2 v2.0.2 // indirect | ||||
| 	google.golang.org/genproto v0.0.0-20240213162025-012b6fc9bca9 // indirect | ||||
| 	google.golang.org/genproto/googleapis/api v0.0.0-20240604185151-ef581f913117 // indirect | ||||
| 	google.golang.org/genproto/googleapis/rpc v0.0.0-20240903143218-8af14fe29dc1 // indirect | ||||
| 	gorm.io/gorm v1.25.8 // indirect | ||||
| 	stathat.com/c/consistent v1.0.0 // indirect | ||||
| ) | ||||
| @ -187,10 +190,10 @@ require ( | ||||
| require ( | ||||
| 	github.com/go-playground/locales v0.14.1 // indirect | ||||
| 	github.com/goccy/go-json v0.10.2 // indirect | ||||
| 	github.com/mattn/go-isatty v0.0.19 // indirect | ||||
| 	github.com/mattn/go-isatty v0.0.20 // indirect | ||||
| 	github.com/spf13/cobra v1.8.0 | ||||
| 	github.com/ugorji/go/codec v1.2.11 // indirect | ||||
| 	github.com/ugorji/go/codec v1.2.12 // indirect | ||||
| 	go.uber.org/zap v1.24.0 // indirect | ||||
| 	golang.org/x/crypto v0.21.0 // indirect | ||||
| 	golang.org/x/crypto v0.27.0 // indirect | ||||
| 	gopkg.in/ini.v1 v1.67.0 // indirect | ||||
| ) | ||||
|  | ||||
							
								
								
									
										182
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										182
									
								
								go.sum
									
									
									
									
									
								
							| @ -1,23 +1,23 @@ | ||||
| cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= | ||||
| cloud.google.com/go v0.112.0 h1:tpFCD7hpHFlQ8yPwT3x+QeXqc2T6+n6T+hmABHfDUSM= | ||||
| cloud.google.com/go v0.112.0/go.mod h1:3jEEVwZ/MHU4djK5t5RHuKOA/GbLddgTdVubX1qnPD4= | ||||
| cloud.google.com/go/compute v1.23.3 h1:6sVlXXBmbd7jNX0Ipq0trII3e4n1/MsADLK6a+aiVlk= | ||||
| cloud.google.com/go/compute v1.23.3/go.mod h1:VCgBUoMnIVIR0CscqQiPJLAG25E3ZRZMzcFZeQ+h8CI= | ||||
| cloud.google.com/go/compute/metadata v0.2.3 h1:mg4jlk7mCAj6xXp9UJ4fjI9VUI5rubuGBW5aJ7UnBMY= | ||||
| cloud.google.com/go/compute/metadata v0.2.3/go.mod h1:VAV5nSsACxMJvgaAuX6Pk2AawlZn8kiOGuCv6gTkwuA= | ||||
| cloud.google.com/go/firestore v1.14.0 h1:8aLcKnMPoldYU3YHgu4t2exrKhLQkqaXAGqT0ljrFVw= | ||||
| cloud.google.com/go/firestore v1.14.0/go.mod h1:96MVaHLsEhbvkBEdZgfN+AS/GIkco1LRpH9Xp9YZfzQ= | ||||
| cloud.google.com/go/iam v1.1.5 h1:1jTsCu4bcsNsE4iiqNT5SHwrDRCfRmIaaaVFhRveTJI= | ||||
| cloud.google.com/go/iam v1.1.5/go.mod h1:rB6P/Ic3mykPbFio+vo7403drjlgvoWfYpJhMXEbzv8= | ||||
| cloud.google.com/go/longrunning v0.5.4 h1:w8xEcbZodnA2BbW6sVirkkoC+1gP8wS57EUUgGS0GVg= | ||||
| cloud.google.com/go/longrunning v0.5.4/go.mod h1:zqNVncI0BOP8ST6XQD1+VcvuShMmq7+xFSzOL++V0dI= | ||||
| cloud.google.com/go/storage v1.36.0 h1:P0mOkAcaJxhCTvAkMhxMfrTKiNcub4YmmPBtlhAyTr8= | ||||
| cloud.google.com/go/storage v1.36.0/go.mod h1:M6M/3V/D3KpzMTJyPOR/HU6n2Si5QdaXYEsng2xgOs8= | ||||
| firebase.google.com/go v3.13.0+incompatible h1:3TdYC3DDi6aHn20qoRkxwGqNgdjtblwVAyRLQwGn/+4= | ||||
| firebase.google.com/go v3.13.0+incompatible/go.mod h1:xlah6XbEyW6tbfSklcfe5FHJIwjt8toICdV5Wh9ptHs= | ||||
| cloud.google.com/go v0.112.1 h1:uJSeirPke5UNZHIb4SxfZklVSiWWVqW4oXlETwZziwM= | ||||
| cloud.google.com/go v0.112.1/go.mod h1:+Vbu+Y1UU+I1rjmzeMOb/8RfkKJK2Gyxi1X6jJCZLo4= | ||||
| cloud.google.com/go/compute/metadata v0.3.0 h1:Tz+eQXMEqDIKRsmY3cHTL6FVaynIjX2QxYC4trgAKZc= | ||||
| cloud.google.com/go/compute/metadata v0.3.0/go.mod h1:zFmK7XCadkQkj6TtorcaGlCW1hT1fIilQDwofLpJ20k= | ||||
| cloud.google.com/go/firestore v1.15.0 h1:/k8ppuWOtNuDHt2tsRV42yI21uaGnKDEQnRFeBpbFF8= | ||||
| cloud.google.com/go/firestore v1.15.0/go.mod h1:GWOxFXcv8GZUtYpWHw/w6IuYNux/BtmeVTMmjrm4yhk= | ||||
| cloud.google.com/go/iam v1.1.7 h1:z4VHOhwKLF/+UYXAJDFwGtNF0b6gjsW1Pk9Ml0U/IoM= | ||||
| cloud.google.com/go/iam v1.1.7/go.mod h1:J4PMPg8TtyurAUvSmPj8FF3EDgY1SPRZxcUGrn7WXGA= | ||||
| cloud.google.com/go/longrunning v0.5.5 h1:GOE6pZFdSrTb4KAiKnXsJBtlE6mEyaW44oKyMILWnOg= | ||||
| cloud.google.com/go/longrunning v0.5.5/go.mod h1:WV2LAxD8/rg5Z1cNW6FJ/ZpX4E4VnDnoTk0yawPBB7s= | ||||
| cloud.google.com/go/storage v1.40.0 h1:VEpDQV5CJxFmJ6ueWNsKxcr1QAYOXEgxDa+sBbJahPw= | ||||
| cloud.google.com/go/storage v1.40.0/go.mod h1:Rrj7/hKlG87BLqDJYtwR0fbPld8uJPbQ2ucUMY7Ir0g= | ||||
| firebase.google.com/go/v4 v4.14.1 h1:4qiUETaFRWoFGE1XP5VbcEdtPX93Qs+8B/7KvP2825g= | ||||
| firebase.google.com/go/v4 v4.14.1/go.mod h1:fgk2XshgNDEKaioKco+AouiegSI9oTWVqRaBdTTGBoM= | ||||
| github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= | ||||
| github.com/IBM/sarama v1.43.0 h1:YFFDn8mMI2QL0wOrG0J2sFoVIAFl7hS9JQi2YZsXtJc= | ||||
| github.com/IBM/sarama v1.43.0/go.mod h1:zlE6HEbC/SMQ9mhEYaF7nNLYOUyrs0obySKCckWP9BM= | ||||
| github.com/MicahParks/keyfunc v1.9.0 h1:lhKd5xrFHLNOWrDc4Tyb/Q1AJ4LCzQ48GVJyVIID3+o= | ||||
| github.com/MicahParks/keyfunc v1.9.0/go.mod h1:IdnCilugA0O/99dW+/MkvlyrsX8+L8+x95xuVNtM5jw= | ||||
| github.com/QcloudApi/qcloud_sign_golang v0.0.0-20141224014652-e4130a326409/go.mod h1:1pk82RBxDY/JZnPQrtqHlUFfCctgdorsd9M06fMynOM= | ||||
| github.com/aliyun/aliyun-oss-go-sdk v3.0.2+incompatible h1:8psS8a+wKfiLt1iVDX79F7Y6wUM49Lcha2FMXt4UM8g= | ||||
| github.com/aliyun/aliyun-oss-go-sdk v3.0.2+incompatible/go.mod h1:T/Aws4fEfogEE9v+HPhhw+CntffsBHJ8nXQCwKr0/g8= | ||||
| @ -65,21 +65,21 @@ github.com/bsm/ginkgo/v2 v2.12.0 h1:Ny8MWAHyOepLGlLKYmXG4IEkioBysk6GpaRTLC8zwWs= | ||||
| github.com/bsm/ginkgo/v2 v2.12.0/go.mod h1:SwYbGRRDovPVboqFv0tPTcG1sN61LM1Z4ARdbAV9g4c= | ||||
| github.com/bsm/gomega v1.27.10 h1:yeMWxP2pV2fG3FgAODIY8EiRE3dy0aeFYt4l7wh6yKA= | ||||
| github.com/bsm/gomega v1.27.10/go.mod h1:JyEr/xRbxbtgWNi8tIEVPUYZ5Dzef52k01W3YH0H+O0= | ||||
| github.com/bytedance/sonic v1.5.0/go.mod h1:ED5hyg4y6t3/9Ku1R6dU/4KyJ48DZ4jPhfY1O2AihPM= | ||||
| github.com/bytedance/sonic v1.9.1 h1:6iJ6NqdoxCDr6mbY8h18oSO+cShGSMRGCEo7F2h0x8s= | ||||
| github.com/bytedance/sonic v1.9.1/go.mod h1:i736AoUSYt75HyZLoJW9ERYxcy6eaN6h4BZXU064P/U= | ||||
| github.com/bytedance/sonic v1.11.6 h1:oUp34TzMlL+OY1OUWxHqsdkgC/Zfc85zGqw9siXjrc0= | ||||
| github.com/bytedance/sonic v1.11.6/go.mod h1:LysEHSvpvDySVdC2f87zGWf6CIKJcAvqab1ZaiQtds4= | ||||
| github.com/bytedance/sonic/loader v0.1.1 h1:c+e5Pt1k/cy5wMveRDyk2X4B9hF4g7an8N3zCYjJFNM= | ||||
| github.com/bytedance/sonic/loader v0.1.1/go.mod h1:ncP89zfokxS5LZrJxl5z0UJcsk4M4yY2JpfqGeCtNLU= | ||||
| github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= | ||||
| github.com/cespare/xxhash/v2 v2.2.0 h1:DC2CZ1Ep5Y4k3ZQ899DldepgrayRUGE6BBZ/cd9Cj44= | ||||
| github.com/cespare/xxhash/v2 v2.2.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||
| github.com/chenzhuoyu/base64x v0.0.0-20211019084208-fb5309c8db06/go.mod h1:DH46F32mSOjUmXrMHnKwZdA8wcEefY7UVqBKYGjpdQY= | ||||
| github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 h1:qSGYFH7+jGhDF8vLC+iwCD4WpbV1EBDSzWkJODFLams= | ||||
| github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311/go.mod h1:b583jCggY9gE99b6G5LEC39OIiVsWj+R97kbl5odCEk= | ||||
| github.com/cespare/xxhash/v2 v2.3.0 h1:UL815xU9SqsFlibzuggzjXhog7bL6oX9BbNZnL2UFvs= | ||||
| github.com/cespare/xxhash/v2 v2.3.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||
| github.com/clbanning/mxj v1.8.4 h1:HuhwZtbyvyOw+3Z1AowPkU87JkJUSv751ELWaiTpj8I= | ||||
| github.com/clbanning/mxj v1.8.4/go.mod h1:BVjHeAH+rl9rs6f+QIpeRl0tfu10SXn1pUSa5PVGJng= | ||||
| github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw= | ||||
| github.com/cloudwego/base64x v0.1.4 h1:jwCgWpFanWmN8xoIUHa2rtzmkd5J2plF/dnLS6Xd/0Y= | ||||
| github.com/cloudwego/base64x v0.1.4/go.mod h1:0zlkT4Wn5C6NdauXdJRhSKRlJvmclQ1hhJgA0rcu/8w= | ||||
| github.com/cloudwego/iasm v0.2.0 h1:1KNIy1I1H9hNNFEEH3DVnI4UujN+1zjpuk6gwHLTssg= | ||||
| github.com/cloudwego/iasm v0.2.0/go.mod h1:8rXZaNYT2n95jn+zTI1sDr+IgcD2GVs0nlbbQPiEFhY= | ||||
| github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc= | ||||
| github.com/cncf/xds/go v0.0.0-20231128003011-0fa0005c9caa h1:jQCWAUqqlij9Pgj2i/PB79y4KOPYVyFYdROxgaCwdTQ= | ||||
| github.com/cncf/xds/go v0.0.0-20231128003011-0fa0005c9caa/go.mod h1:x/1Gn8zydmfq8dk6e9PdstVsDgu9RuyIIJqAaF//0IM= | ||||
| github.com/coreos/go-semver v0.3.0 h1:wkHLiw0WNATZnSG7epLsujiMCgPAc9xhjJ4tgnAxmfM= | ||||
| github.com/coreos/go-semver v0.3.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk= | ||||
| github.com/coreos/go-systemd/v22 v22.3.2 h1:D9/bQk5vlXQFZ6Kwuu6zaiXJ9oTPe68++AzAJc1DzSI= | ||||
| @ -107,8 +107,6 @@ github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymF | ||||
| github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= | ||||
| github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98= | ||||
| github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= | ||||
| github.com/envoyproxy/protoc-gen-validate v1.0.4 h1:gVPz/FMfvh57HdSJQyvBtF00j8JU4zdyUgIUNhlgg0A= | ||||
| github.com/envoyproxy/protoc-gen-validate v1.0.4/go.mod h1:qys6tmnRsYrQqIhm2bvKZH4Blx/1gTIZ2UKVY1M+Yew= | ||||
| github.com/fatih/color v1.14.1 h1:qfhVLaG5s+nCROl1zJsZRxFeYrHLqWroPOQ8BWiNb4w= | ||||
| github.com/fatih/color v1.14.1/go.mod h1:2oHN61fhTpgcxD3TSWCgKDiH1+x4OiDVVGH8WlgGZGg= | ||||
| github.com/felixge/httpsnoop v1.0.4 h1:NFTV2Zj1bL4mc9sqWACXbQFVBBg2W3GPvqp8/ESS2Wg= | ||||
| @ -121,6 +119,8 @@ github.com/fsnotify/fsnotify v1.7.0 h1:8JEhPFa5W2WU7YfeZzPNqzMP6Lwt7L2715Ggo0nos | ||||
| github.com/fsnotify/fsnotify v1.7.0/go.mod h1:40Bi/Hjc2AVfZrqy+aj+yEI+/bRxZnMJyTJwOpGvigM= | ||||
| github.com/gabriel-vasile/mimetype v1.4.3 h1:in2uUcidCuFcDKtdcBxlR0rJ1+fsokWf+uqxgUFjbI0= | ||||
| github.com/gabriel-vasile/mimetype v1.4.3/go.mod h1:d8uq/6HKRL6CGdk+aubisF/M5GcPfT7nKyLpA0lbSSk= | ||||
| github.com/gin-contrib/gzip v1.0.1 h1:HQ8ENHODeLY7a4g1Au/46Z92bdGFl74OhxcZble9WJE= | ||||
| github.com/gin-contrib/gzip v1.0.1/go.mod h1:njt428fdUNRvjuJf16tZMYZ2Yl+WQB53X5wmhDwXvC4= | ||||
| github.com/gin-contrib/sse v0.1.0 h1:Y/yl/+YNO8GZSjAhjMsSuLt29uWRFHdHYUb5lYOV9qE= | ||||
| github.com/gin-contrib/sse v0.1.0/go.mod h1:RHrZQHXnP2xjPF+u1gW/2HnVO7nvIa9PG3Gm+fLHvGI= | ||||
| github.com/gin-gonic/gin v1.9.1 h1:4idEAncQnU5cB7BeOkPtxjfCSye0AAm1R0RVIqJ+Jmg= | ||||
| @ -144,8 +144,8 @@ github.com/go-playground/universal-translator v0.18.0/go.mod h1:UvRDBj+xPUEGrFYl | ||||
| github.com/go-playground/universal-translator v0.18.1 h1:Bcnm0ZwsGyWbCzImXv+pAJnYK9S473LQFuzCbDbfSFY= | ||||
| github.com/go-playground/universal-translator v0.18.1/go.mod h1:xekY+UJKNuX9WP91TpwSH2VMlDf28Uj24BCp08ZFTUY= | ||||
| github.com/go-playground/validator/v10 v10.8.0/go.mod h1:9JhgTzTaE31GZDpH/HSvHiRJrJ3iKAgqqH0Bl/Ocjdk= | ||||
| github.com/go-playground/validator/v10 v10.18.0 h1:BvolUXjp4zuvkZ5YN5t7ebzbhlUtPsPm2S9NAZ5nl9U= | ||||
| github.com/go-playground/validator/v10 v10.18.0/go.mod h1:dbuPbCMFw/DrkbEynArYaCwl3amGuJotoKCe95atGMM= | ||||
| github.com/go-playground/validator/v10 v10.20.0 h1:K9ISHbSaI0lyB2eWMPJo+kOS/FBExVwjEviJTixqxL8= | ||||
| github.com/go-playground/validator/v10 v10.20.0/go.mod h1:dbuPbCMFw/DrkbEynArYaCwl3amGuJotoKCe95atGMM= | ||||
| github.com/go-redis/redis v6.15.9+incompatible h1:K0pv1D7EQUjfyoMql+r/jZqCLizCGKFlFgcHWWmHQjg= | ||||
| github.com/go-redis/redis v6.15.9+incompatible/go.mod h1:NAIEuMOZ/fxfXJIrKDQDz8wamY7mA7PouImQ2Jvg6kA= | ||||
| github.com/go-redis/redismock/v9 v9.2.0 h1:ZrMYQeKPECZPjOj5u9eyOjg8Nnb0BS9lkVIZ6IpsKLw= | ||||
| @ -157,6 +157,7 @@ github.com/goccy/go-json v0.10.2/go.mod h1:6MelG93GURQebXPDq3khkgXZkazVtN9CRI+MG | ||||
| github.com/godbus/dbus/v5 v5.0.4/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA= | ||||
| github.com/gogo/protobuf v1.3.2 h1:Ov1cvc58UF3b5XjBnZv7+opcTcQFZebYjWzi34vdm4Q= | ||||
| github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q= | ||||
| github.com/golang-jwt/jwt/v4 v4.4.2/go.mod h1:m21LjoU+eqJr34lmDMbreY2eSTRJ1cv77w39/MY0Ch0= | ||||
| github.com/golang-jwt/jwt/v4 v4.5.0 h1:7cYmW1XlMY7h7ii7UhUyChSgS5wUJEnm9uZVTGqOWzg= | ||||
| github.com/golang-jwt/jwt/v4 v4.5.0/go.mod h1:m21LjoU+eqJr34lmDMbreY2eSTRJ1cv77w39/MY0Ch0= | ||||
| github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= | ||||
| @ -165,6 +166,7 @@ github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da h1:oI5xCqsCo564l | ||||
| github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= | ||||
| github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= | ||||
| github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | ||||
| github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | ||||
| github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | ||||
| github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8= | ||||
| github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA= | ||||
| @ -173,8 +175,6 @@ github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:W | ||||
| github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0= | ||||
| github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8= | ||||
| github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= | ||||
| github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= | ||||
| github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= | ||||
| github.com/golang/protobuf v1.5.4 h1:i7eJL8qZTpSEXOPTxNKhASYpMn+8e5Q6AdndVa1dWek= | ||||
| github.com/golang/protobuf v1.5.4/go.mod h1:lnTiLA8Wa4RWRcIUkrtSVa5nRhsEGBg48fD6rSs7xps= | ||||
| github.com/golang/snappy v0.0.4 h1:yAGX7huGHXlcLOEtBnF4w7FQwA26wojNCwOYAEhLjQM= | ||||
| @ -186,7 +186,6 @@ github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/ | ||||
| github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.2/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.3/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= | ||||
| github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= | ||||
| github.com/google/go-querystring v1.0.0/go.mod h1:odCYkC5MyYFN7vkCjXpyrEuKhc/BUO6wN/zVPAxq5ck= | ||||
| @ -203,8 +202,8 @@ github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= | ||||
| github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/googleapis/enterprise-certificate-proxy v0.3.2 h1:Vie5ybvEvT75RniqhfFxPRy3Bf7vr3h0cechB90XaQs= | ||||
| github.com/googleapis/enterprise-certificate-proxy v0.3.2/go.mod h1:VLSiSSBs/ksPL8kq3OBOQ6WRI2QnaFynd1DCjZ62+V0= | ||||
| github.com/googleapis/gax-go/v2 v2.12.0 h1:A+gCJKdRfqXkr+BIRGtZLibNXf0m1f9E4HG56etFpas= | ||||
| github.com/googleapis/gax-go/v2 v2.12.0/go.mod h1:y+aIqrI5eb1YGMVJfuV3185Ts/D7qKpsEkdD5+I6QGU= | ||||
| github.com/googleapis/gax-go/v2 v2.12.3 h1:5/zPPDvw8Q1SuXjrqrZslrqT7dL/uJT2CQii/cLCKqA= | ||||
| github.com/googleapis/gax-go/v2 v2.12.3/go.mod h1:AKloxT6GtNbaLm8QTNSidHUVsHYcBHwWRvkNFJUQcS4= | ||||
| github.com/gorilla/securecookie v1.1.1/go.mod h1:ra0sb63/xPlUeL+yeDciTfxMRAA+MP+HVt/4epWDjd4= | ||||
| github.com/gorilla/sessions v1.2.1/go.mod h1:dk2InVEVJ0sfLlnXv9EAgkf6ecYs/i80K/zI+bUmuGM= | ||||
| github.com/gorilla/websocket v1.5.1 h1:gmztn0JnHVt9JZquRuzLw3g4wouNVzKL15iLr/zn/QY= | ||||
| @ -257,8 +256,9 @@ github.com/klauspost/compress v1.17.7 h1:ehO88t2UGzQK66LMdE8tibEd1ErmzZjNEqWkjLA | ||||
| github.com/klauspost/compress v1.17.7/go.mod h1:Di0epgTjJY877eYKx5yC51cX2A2Vl2ibi7bDH9ttBbw= | ||||
| github.com/klauspost/cpuid/v2 v2.0.1/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg= | ||||
| github.com/klauspost/cpuid/v2 v2.0.9/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg= | ||||
| github.com/klauspost/cpuid/v2 v2.2.6 h1:ndNyv040zDGIDh8thGkXYjnFtiN02M1PVVF+JE/48xc= | ||||
| github.com/klauspost/cpuid/v2 v2.2.6/go.mod h1:Lcz8mBdAVJIBVzewtcLocK12l3Y+JytZYpaMropDUws= | ||||
| github.com/klauspost/cpuid/v2 v2.2.7 h1:ZWSB3igEs+d0qvnxR/ZBzXVmxkgt8DdzP6m9pfuVLDM= | ||||
| github.com/klauspost/cpuid/v2 v2.2.7/go.mod h1:Lcz8mBdAVJIBVzewtcLocK12l3Y+JytZYpaMropDUws= | ||||
| github.com/knz/go-libedit v1.10.1/go.mod h1:MZTVkCWyz0oBc7JOWP3wNAzd002ZbM/5hgShxwh4x8M= | ||||
| github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= | ||||
| github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= | ||||
| github.com/kr/pretty v0.3.0/go.mod h1:640gp4NfQd8pI5XOwp5fnNeVWj67G7CFk/SaSQn7NBk= | ||||
| @ -288,8 +288,8 @@ github.com/magiconair/properties v1.8.7/go.mod h1:Dhd985XPs7jluiymwWYZ0G4Z61jb3v | ||||
| github.com/mattn/go-colorable v0.1.13 h1:fFA4WZxdEF4tXPZVKMLwD8oUnCTTo08duU7wxecdEvA= | ||||
| github.com/mattn/go-colorable v0.1.13/go.mod h1:7S9/ev0klgBDR4GtXTXX8a3vIGJpMovkB8vQcUbaXHg= | ||||
| github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM= | ||||
| github.com/mattn/go-isatty v0.0.19 h1:JITubQf0MOLdlGRuRq+jtsDlekdYPia9ZFsB8h/APPA= | ||||
| github.com/mattn/go-isatty v0.0.19/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y= | ||||
| github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY= | ||||
| github.com/mattn/go-isatty v0.0.20/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y= | ||||
| github.com/matttproud/golang_protobuf_extensions/v2 v2.0.0 h1:jWpvCLoY8Z/e3VKvlsiIGKtc+UG6U5vzxaoagmhXfyg= | ||||
| github.com/matttproud/golang_protobuf_extensions/v2 v2.0.0/go.mod h1:QUyp042oQthUoa9bqDv0ER0wrtXnBruoNd7aNjkbP+k= | ||||
| github.com/minio/md5-simd v1.1.2 h1:Gdi1DZK69+ZVMoNHRXJyNcxrMA4dSxoYHZSQbirFg34= | ||||
| @ -319,12 +319,12 @@ github.com/onsi/gomega v1.25.0 h1:Vw7br2PCDYijJHSfBOWhov+8cAnUf8MfMaIOV323l6Y= | ||||
| github.com/onsi/gomega v1.25.0/go.mod h1:r+zV744Re+DiYCIPRlYOTxn0YkOLcAnW8k1xXdMPGhM= | ||||
| github.com/openimsdk/gomake v0.0.14-alpha.5 h1:VY9c5x515lTfmdhhPjMvR3BBRrRquAUCFsz7t7vbv7Y= | ||||
| github.com/openimsdk/gomake v0.0.14-alpha.5/go.mod h1:PndCozNc2IsQIciyn9mvEblYWZwJmAI+06z94EY+csI= | ||||
| github.com/openimsdk/protocol v0.0.72-alpha.9 h1:Dyx4vs88IU4rJ2YcP/TdYp4ww8JjsMkV89hB/Eazx+A= | ||||
| github.com/openimsdk/protocol v0.0.72-alpha.9/go.mod h1:OZQA9FR55lseYoN2Ql1XAHYKHJGu7OMNkUbuekrKCM8= | ||||
| github.com/openimsdk/tools v0.0.49-alpha.55 h1:KPgC53oqiwZYssLKljhtXbWXifMlTj2SSQEusj4Uf4k= | ||||
| github.com/openimsdk/tools v0.0.49-alpha.55/go.mod h1:h1cYmfyaVtgFbKmb1Cfsl8XwUOMTt8ubVUQrdGtsUh4= | ||||
| github.com/pelletier/go-toml/v2 v2.1.0 h1:FnwAJ4oYMvbT/34k9zzHuZNrhlz48GB3/s6at6/MHO4= | ||||
| github.com/pelletier/go-toml/v2 v2.1.0/go.mod h1:tJU2Z3ZkXwnxa4DPO899bsyIoywizdUvyaeZurnPPDc= | ||||
| github.com/openimsdk/protocol v0.0.72-alpha.46 h1:1LZlfEHLzw1F4afFmqBczmXKJWm5rUQ+yr8rJ4oyEAc= | ||||
| github.com/openimsdk/protocol v0.0.72-alpha.46/go.mod h1:OZQA9FR55lseYoN2Ql1XAHYKHJGu7OMNkUbuekrKCM8= | ||||
| github.com/openimsdk/tools v0.0.50-alpha.16 h1:bC1AQvJMuOHtZm8LZRvN8L5mH1Ws2VYdL+TLTs1iGSc= | ||||
| github.com/openimsdk/tools v0.0.50-alpha.16/go.mod h1:h1cYmfyaVtgFbKmb1Cfsl8XwUOMTt8ubVUQrdGtsUh4= | ||||
| github.com/pelletier/go-toml/v2 v2.2.2 h1:aYUidT7k73Pcl9nb2gScu7NSrKCSHIDE89b3+6Wq+LM= | ||||
| github.com/pelletier/go-toml/v2 v2.2.2/go.mod h1:1t835xjRzz80PqgE6HHgN2JOsmgYu/h4qDAS4n929Rs= | ||||
| github.com/pierrec/lz4/v4 v4.1.21 h1:yOVMLb6qSIDP67pl/5F7RepeKYu/VmTyEXvuMI5d9mQ= | ||||
| github.com/pierrec/lz4/v4 v4.1.21/go.mod h1:gZWDp/Ze/IJXGXf23ltt2EXimqmTUXEy0GFuRQyBid4= | ||||
| github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsKMEsNbx1VGcRFpLqf3715MtcvvzbA= | ||||
| @ -408,8 +408,8 @@ github.com/tklauser/numcpus v0.7.0 h1:yjuerZP127QG9m5Zh/mSO4wqurYil27tHrqwRoRjpr | ||||
| github.com/tklauser/numcpus v0.7.0/go.mod h1:bb6dMVcj8A42tSE7i32fsIUCbQNllK5iDguyOZRUzAY= | ||||
| github.com/twitchyliquid64/golang-asm v0.15.1 h1:SU5vSMR7hnwNxj24w34ZyCi/FmDZTkS4MhqMhdFk5YI= | ||||
| github.com/twitchyliquid64/golang-asm v0.15.1/go.mod h1:a1lVb/DtPvCB8fslRZhAngC2+aY1QWCk3Cedj/Gdt08= | ||||
| github.com/ugorji/go/codec v1.2.11 h1:BMaWp1Bb6fHwEtbplGBGJ498wD+LKlNSl25MjdZY4dU= | ||||
| github.com/ugorji/go/codec v1.2.11/go.mod h1:UNopzCgEMSXjBc6AOMqYvWC1ktqTAfzJZUZgYf6w6lg= | ||||
| github.com/ugorji/go/codec v1.2.12 h1:9LC83zGrHhuUA9l16C9AHXAqEV/2wBQ4nkvumAE65EE= | ||||
| github.com/ugorji/go/codec v1.2.12/go.mod h1:UNopzCgEMSXjBc6AOMqYvWC1ktqTAfzJZUZgYf6w6lg= | ||||
| github.com/xdg-go/pbkdf2 v1.0.0 h1:Su7DPu48wXMwC3bs7MCNG+z4FhcyEuz5dlvchbq0B0c= | ||||
| github.com/xdg-go/pbkdf2 v1.0.0/go.mod h1:jrpuAogTd400dnrH08LKmI/xc1MbPOebTwRqcT5RDeI= | ||||
| github.com/xdg-go/scram v1.1.2 h1:FHX5I5B4i4hKRVRBCFRxq1iQRej7WO3hhBuJf+UUySY= | ||||
| @ -433,18 +433,18 @@ go.mongodb.org/mongo-driver v1.14.0 h1:P98w8egYRjYe3XDjxhYJagTokP/H6HzlsnojRgZRd | ||||
| go.mongodb.org/mongo-driver v1.14.0/go.mod h1:Vzb0Mk/pa7e6cWw85R4F/endUC3u0U9jGcNU603k65c= | ||||
| go.opencensus.io v0.24.0 h1:y73uSU6J157QMP2kn2r30vwW1A2W2WFwSCGnAVxeaD0= | ||||
| go.opencensus.io v0.24.0/go.mod h1:vNK8G9p7aAivkbmorf4v+7Hgx+Zs0yY+0fOtgBfjQKo= | ||||
| go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.47.0 h1:UNQQKPfTDe1J81ViolILjTKPr9WetKW6uei2hFgJmFs= | ||||
| go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.47.0/go.mod h1:r9vWsPS/3AQItv3OSlEJ/E4mbrhUbbw18meOjArPtKQ= | ||||
| go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.47.0 h1:sv9kVfal0MK0wBMCOGr+HeJm9v803BkJxGrk2au7j08= | ||||
| go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.47.0/go.mod h1:SK2UL73Zy1quvRPonmOmRDiWk1KBV3LyIeeIxcEApWw= | ||||
| go.opentelemetry.io/otel v1.23.0 h1:Df0pqjqExIywbMCMTxkAwzjLZtRf+bBKLbUcpxO2C9E= | ||||
| go.opentelemetry.io/otel v1.23.0/go.mod h1:YCycw9ZeKhcJFrb34iVSkyT0iczq/zYDtZYFufObyB0= | ||||
| go.opentelemetry.io/otel/metric v1.23.0 h1:pazkx7ss4LFVVYSxYew7L5I6qvLXHA0Ap2pwV+9Cnpo= | ||||
| go.opentelemetry.io/otel/metric v1.23.0/go.mod h1:MqUW2X2a6Q8RN96E2/nqNoT+z9BSms20Jb7Bbp+HiTo= | ||||
| go.opentelemetry.io/otel/sdk v1.21.0 h1:FTt8qirL1EysG6sTQRZ5TokkU8d0ugCj8htOgThZXQ8= | ||||
| go.opentelemetry.io/otel/sdk v1.21.0/go.mod h1:Nna6Yv7PWTdgJHVRD9hIYywQBRx7pbox6nwBnZIxl/E= | ||||
| go.opentelemetry.io/otel/trace v1.23.0 h1:37Ik5Ib7xfYVb4V1UtnT97T1jI+AoIYkJyPkuL4iJgI= | ||||
| go.opentelemetry.io/otel/trace v1.23.0/go.mod h1:GSGTbIClEsuZrGIzoEHqsVfxgn5UkggkflQwDScNUsk= | ||||
| go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.49.0 h1:4Pp6oUg3+e/6M4C0A/3kJ2VYa++dsWVTtGgLVj5xtHg= | ||||
| go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.49.0/go.mod h1:Mjt1i1INqiaoZOMGR1RIUJN+i3ChKoFRqzrRQhlkbs0= | ||||
| go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.49.0 h1:jq9TW8u3so/bN+JPT166wjOI6/vQPF6Xe7nMNIltagk= | ||||
| go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.49.0/go.mod h1:p8pYQP+m5XfbZm9fxtSKAbM6oIllS7s2AfxrChvc7iw= | ||||
| go.opentelemetry.io/otel v1.24.0 h1:0LAOdjNmQeSTzGBzduGe/rU4tZhMwL5rWgtp9Ku5Jfo= | ||||
| go.opentelemetry.io/otel v1.24.0/go.mod h1:W7b9Ozg4nkF5tWI5zsXkaKKDjdVjpD4oAt9Qi/MArHo= | ||||
| go.opentelemetry.io/otel/metric v1.24.0 h1:6EhoGWWK28x1fbpA4tYTOWBkPefTDQnb8WSGXlc88kI= | ||||
| go.opentelemetry.io/otel/metric v1.24.0/go.mod h1:VYhLe1rFfxuTXLgj4CBiyz+9WYBA8pNGJgDcSFRKBco= | ||||
| go.opentelemetry.io/otel/sdk v1.22.0 h1:6coWHw9xw7EfClIC/+O31R8IY3/+EiRFHevmHafB2Gw= | ||||
| go.opentelemetry.io/otel/sdk v1.22.0/go.mod h1:iu7luyVGYovrRpe2fmj3CVKouQNdTOkxtLzPvPz1DOc= | ||||
| go.opentelemetry.io/otel/trace v1.24.0 h1:CsKnnL4dUAr/0llH9FKuc698G04IrpWV0MQA/Y1YELI= | ||||
| go.opentelemetry.io/otel/trace v1.24.0/go.mod h1:HPc3Xr/cOApsBI154IU0OI0HJexz+aw5uPdbs3UCjNU= | ||||
| go.uber.org/atomic v1.9.0 h1:ECmE8Bn/WFTYwEW/bpKD3M8VtR/zQVbavAoalC1PYyE= | ||||
| go.uber.org/atomic v1.9.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc= | ||||
| go.uber.org/automaxprocs v1.5.3 h1:kWazyxZUrS3Gs4qUpbwo5kEIMGe/DAvi5Z4tl2NW4j8= | ||||
| @ -456,8 +456,8 @@ go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN8 | ||||
| go.uber.org/zap v1.24.0 h1:FiJd5l1UOLj0wCgbSE0rwwXHzEdAZS6hiiSnxJN/D60= | ||||
| go.uber.org/zap v1.24.0/go.mod h1:2kMP+WWQ8aoFoedH3T2sq6iJ2yDWpHbP0f6MQbS9Gkg= | ||||
| golang.org/x/arch v0.0.0-20210923205945-b76863e36670/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8= | ||||
| golang.org/x/arch v0.3.0 h1:02VY4/ZcO/gBOH6PUaoiptASxtXU10jazRCP865E97k= | ||||
| golang.org/x/arch v0.3.0/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8= | ||||
| golang.org/x/arch v0.7.0 h1:pskyeJh/3AmoQ8CPE95vxHLqp1G1GfGNXTmcl9NEKTc= | ||||
| golang.org/x/arch v0.7.0/go.mod h1:FEVrYAQjsQXMVJ1nsMoVVXPZg6p2JE2mx8psSWTDQys= | ||||
| golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= | ||||
| golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||
| golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | ||||
| @ -465,8 +465,8 @@ golang.org/x/crypto v0.0.0-20210711020723-a769d52b0f97/go.mod h1:GvvjBRRGRdwPK5y | ||||
| golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= | ||||
| golang.org/x/crypto v0.1.0/go.mod h1:RecgLatLF4+eUMCP1PoPZQb+cVrJcOPbHkTkbkB9sbw= | ||||
| golang.org/x/crypto v0.6.0/go.mod h1:OFC/31mSvZgRz0V1QTNCzfAI1aIRzbiufJtkMIlEp58= | ||||
| golang.org/x/crypto v0.21.0 h1:X31++rzVUdKhX5sWmSOFZxx8UW/ldWx55cbf08iNAMA= | ||||
| golang.org/x/crypto v0.21.0/go.mod h1:0BP7YvVV9gBbVKyeTG0Gyn+gZm94bibOW5BjDEYAOMs= | ||||
| golang.org/x/crypto v0.27.0 h1:GXm2NjJrPaiv/h1tb2UH8QfgC/hOf/+z0p6PT8o1w7A= | ||||
| golang.org/x/crypto v0.27.0/go.mod h1:1Xngt8kV6Dvbssa53Ziq6Eqn0HqbZi5Z6R0ZpwQzt70= | ||||
| golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= | ||||
| golang.org/x/exp v0.0.0-20230905200255-921286631fa9 h1:GoHiUyI/Tp2nVkLI2mCxVkOjsbSXD66ic0XW0js0R9g= | ||||
| golang.org/x/exp v0.0.0-20230905200255-921286631fa9/go.mod h1:S2oDrQGGwySpoQPVqRShND87VCbxmc6bL1Yd2oYrm6k= | ||||
| @ -489,23 +489,24 @@ golang.org/x/net v0.0.0-20200226121028-0de0cce0169b/go.mod h1:z5CRVTTTmAJ677TzLL | ||||
| golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20201110031124-69a78807bb2b/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | ||||
| golang.org/x/net v0.0.0-20220708220712-1185a9018129/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c= | ||||
| golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c= | ||||
| golang.org/x/net v0.1.0/go.mod h1:Cx3nUiGt4eDBEyega/BKRp+/AlGL8hYe7U9odMt2Cco= | ||||
| golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= | ||||
| golang.org/x/net v0.7.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= | ||||
| golang.org/x/net v0.22.0 h1:9sGLhx7iRIHEiX0oAJ3MRZMUCElJgy7Br1nO+AMN3Tc= | ||||
| golang.org/x/net v0.22.0/go.mod h1:JKghWKKOSdJwpW2GEx0Ja7fmaKnMsbu+MWVZTokSYmg= | ||||
| golang.org/x/net v0.29.0 h1:5ORfpBpCs4HzDYoodCDBbwHzdR5UrLBZ3sOnUJmFoHo= | ||||
| golang.org/x/net v0.29.0/go.mod h1:gLkgy8jTGERgjzMic6DS9+SP0ajcu6Xu3Orq/SpETg0= | ||||
| golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= | ||||
| golang.org/x/oauth2 v0.17.0 h1:6m3ZPmLEFdVxKKWnKq4VqZ60gutO35zm+zrAHVmHyDQ= | ||||
| golang.org/x/oauth2 v0.17.0/go.mod h1:OzPDGQiuQMguemayvdylqddI7qcD9lnSDb+1FiwQ5HA= | ||||
| golang.org/x/oauth2 v0.21.0 h1:tsimM75w1tF/uws5rbeHzIWxEqElMehnc+iW793zsZs= | ||||
| golang.org/x/oauth2 v0.21.0/go.mod h1:XYTD2NtWslqkgxebSiOHnXEap4TF09sJSc7H1sXbhtI= | ||||
| golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.6.0 h1:5BMeUDZ7vkXGfEr1x9B4bRcTH4lpkTkpdh0T/J+qjbQ= | ||||
| golang.org/x/sync v0.6.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= | ||||
| golang.org/x/sync v0.8.0 h1:3NFvSEYkUoMifnESzZl15y791HH1qU2xm6eCJU5ZPXQ= | ||||
| golang.org/x/sync v0.8.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= | ||||
| golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| @ -520,8 +521,8 @@ golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBc | ||||
| golang.org/x/sys v0.1.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.19.0 h1:q5f1RH2jigJ1MoAWp2KTp3gm5zAGFUTarQZ5U386+4o= | ||||
| golang.org/x/sys v0.19.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | ||||
| golang.org/x/sys v0.25.0 h1:r+8e+loiHxRqhXVl6ML1nO3l1+oFoWbnlu2Ehimmi34= | ||||
| golang.org/x/sys v0.25.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | ||||
| golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||
| golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= | ||||
| golang.org/x/term v0.1.0/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= | ||||
| @ -534,8 +535,8 @@ golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= | ||||
| golang.org/x/text v0.3.8/go.mod h1:E6s5w1FMmriuDzIBO73fBruAKo1PCIq6d2Q6DHfQ8WQ= | ||||
| golang.org/x/text v0.4.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= | ||||
| golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= | ||||
| golang.org/x/text v0.14.0 h1:ScX5w1eTa3QqT8oi6+ziP7dTV1S2+ALU0bI+0zXKWiQ= | ||||
| golang.org/x/text v0.14.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU= | ||||
| golang.org/x/text v0.18.0 h1:XvMDiNzPAl0jr17s6W9lcaIhGUfUORdGCNsuLmPG224= | ||||
| golang.org/x/text v0.18.0/go.mod h1:BuEKDfySbSR4drPmRPG/7iBdf8hvFMuRexcpahXilzY= | ||||
| golang.org/x/time v0.5.0 h1:o7cqy6amK/52YcAKIPlM3a+Fpj35zvRj2TP+e1xFSfk= | ||||
| golang.org/x/time v0.5.0/go.mod h1:3BpzKBy/shNhVucY/MWOyx10tF3SFh9QdLuxbVysPQM= | ||||
| golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| @ -551,30 +552,30 @@ golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8T | ||||
| golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20220907171357-04be3eba64a2 h1:H2TDz8ibqkAF6YGhCdN3jS9O0/s90v0rJh3X/OLHEUk= | ||||
| golang.org/x/xerrors v0.0.0-20220907171357-04be3eba64a2/go.mod h1:K8+ghG5WaK9qNqU5K3HdILfMLy1f3aNYFI/wnl100a8= | ||||
| google.golang.org/api v0.165.0 h1:zd5d4JIIIaYYsfVy1HzoXYZ9rWCSBxxAglbczzo7Bgc= | ||||
| google.golang.org/api v0.165.0/go.mod h1:2OatzO7ZDQsoS7IFf3rvsE17/TldiU3F/zxFHeqUB5o= | ||||
| golang.org/x/xerrors v0.0.0-20231012003039-104605ab7028 h1:+cNy6SZtPcJQH3LJVLOSmiC7MMxXNOb3PU/VUEz+EhU= | ||||
| golang.org/x/xerrors v0.0.0-20231012003039-104605ab7028/go.mod h1:NDW/Ps6MPRej6fsCIbMTohpP40sJ/P/vI1MoTEGwX90= | ||||
| google.golang.org/api v0.170.0 h1:zMaruDePM88zxZBG+NG8+reALO2rfLhe/JShitLyT48= | ||||
| google.golang.org/api v0.170.0/go.mod h1:/xql9M2btF85xac/VAm4PsLMTLVGUOpq4BE9R8jyNy8= | ||||
| google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= | ||||
| google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= | ||||
| google.golang.org/appengine v1.6.8 h1:IhEN5q69dyKagZPYMSdIjS2HqprW324FRQZJcGqPAsM= | ||||
| google.golang.org/appengine v1.6.8/go.mod h1:1jJ3jBArFh5pcgW8gCtRJnepW8FzD1V44FJffLiz/Ds= | ||||
| google.golang.org/appengine/v2 v2.0.2 h1:MSqyWy2shDLwG7chbwBJ5uMyw6SNqJzhJHNDwYB0Akk= | ||||
| google.golang.org/appengine/v2 v2.0.2/go.mod h1:PkgRUWz4o1XOvbqtWTkBtCitEJ5Tp4HoVEdMMYQR/8E= | ||||
| google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= | ||||
| google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= | ||||
| google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo= | ||||
| google.golang.org/genproto v0.0.0-20240125205218-1f4bbc51befe h1:USL2DhxfgRchafRvt/wYyyQNzwgL7ZiURcozOE/Pkvo= | ||||
| google.golang.org/genproto v0.0.0-20240125205218-1f4bbc51befe/go.mod h1:cc8bqMqtv9gMOr0zHg2Vzff5ULhhL2IXP4sbcn32Dro= | ||||
| google.golang.org/genproto/googleapis/api v0.0.0-20240123012728-ef4313101c80 h1:Lj5rbfG876hIAYFjqiJnPHfhXbv+nzTWfm04Fg/XSVU= | ||||
| google.golang.org/genproto/googleapis/api v0.0.0-20240123012728-ef4313101c80/go.mod h1:4jWUdICTdgc3Ibxmr8nAJiiLHwQBY0UI0XZcEMaFKaA= | ||||
| google.golang.org/genproto/googleapis/rpc v0.0.0-20240318140521-94a12d6c2237 h1:NnYq6UN9ReLM9/Y01KWNOWyI5xQ9kbIms5GGJVwS/Yc= | ||||
| google.golang.org/genproto/googleapis/rpc v0.0.0-20240318140521-94a12d6c2237/go.mod h1:WtryC6hu0hhx87FDGxWCDptyssuo68sk10vYjF+T9fY= | ||||
| google.golang.org/genproto v0.0.0-20240213162025-012b6fc9bca9 h1:9+tzLLstTlPTRyJTh+ah5wIMsBW5c4tQwGTN3thOW9Y= | ||||
| google.golang.org/genproto v0.0.0-20240213162025-012b6fc9bca9/go.mod h1:mqHbVIp48Muh7Ywss/AD6I5kNVKZMmAa/QEW58Gxp2s= | ||||
| google.golang.org/genproto/googleapis/api v0.0.0-20240604185151-ef581f913117 h1:+rdxYoE3E5htTEWIe15GlN6IfvbURM//Jt0mmkmm6ZU= | ||||
| google.golang.org/genproto/googleapis/api v0.0.0-20240604185151-ef581f913117/go.mod h1:OimBR/bc1wPO9iV4NC2bpyjy3VnAwZh5EBPQdtaE5oo= | ||||
| google.golang.org/genproto/googleapis/rpc v0.0.0-20240903143218-8af14fe29dc1 h1:pPJltXNxVzT4pK9yD8vR9X75DaWYYmLGMsEvBfFQZzQ= | ||||
| google.golang.org/genproto/googleapis/rpc v0.0.0-20240903143218-8af14fe29dc1/go.mod h1:UqMtugtsSgubUsoxbuAoiCXvqvErP7Gf0so0mK9tHxU= | ||||
| google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c= | ||||
| google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= | ||||
| google.golang.org/grpc v1.25.1/go.mod h1:c3i+UQWmh7LiEpx4sFZnkU36qjEYZ0imhYfXVyQciAY= | ||||
| google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= | ||||
| google.golang.org/grpc v1.33.2/go.mod h1:JMHMWHQWaTccqQQlmk3MJZS+GWXOdAesneDmEnv2fbc= | ||||
| google.golang.org/grpc v1.62.1 h1:B4n+nfKzOICUXMgyrNd19h/I9oH0L1pizfk1d4zSgTk= | ||||
| google.golang.org/grpc v1.62.1/go.mod h1:IWTG0VlJLCh1SkC58F7np9ka9mx/WNkjl4PGJaiq+QE= | ||||
| google.golang.org/grpc v1.66.2 h1:3QdXkuq3Bkh7w+ywLdLvM56cmGvQHUMZpiCzt6Rqaoo= | ||||
| google.golang.org/grpc v1.66.2/go.mod h1:s3/l6xSSCURdVfAnL+TqCNMyTDAGN6+lZeVxnZR128Y= | ||||
| google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8= | ||||
| google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= | ||||
| google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM= | ||||
| @ -584,10 +585,8 @@ google.golang.org/protobuf v1.22.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2 | ||||
| google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= | ||||
| google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= | ||||
| google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c= | ||||
| google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= | ||||
| google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= | ||||
| google.golang.org/protobuf v1.33.0 h1:uNO2rsAINq/JlFpSdYEKIZ0uKD/R9cpdv0T+yoGwGmI= | ||||
| google.golang.org/protobuf v1.33.0/go.mod h1:c6P6GXX6sHbq/GpV6MGZEdwhWPcYBgnhAHhKbcUYpos= | ||||
| google.golang.org/protobuf v1.34.2 h1:6xV6lTsCfpGD21XK49h7MhtcApnLqkfYgPcdHftf6hg= | ||||
| google.golang.org/protobuf v1.34.2/go.mod h1:qYOHts0dSfpeUzUFpOMr/WGzszTmLH+DiWniOlNbLDw= | ||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= | ||||
| @ -608,6 +607,7 @@ gorm.io/gorm v1.25.8 h1:WAGEZ/aEcznN4D03laj8DKnehe1e9gYQAjW8xyPRdeo= | ||||
| gorm.io/gorm v1.25.8/go.mod h1:hbnx/Oo0ChWMn1BIhpy1oYozzpM15i4YPuHDmfYtwg8= | ||||
| honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= | ||||
| honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= | ||||
| nullprogram.com/x/optparse v1.0.0/go.mod h1:KdyPE+Igbe0jQUrVfMqDMeJQIJZEuyV7pjYmp6pbG50= | ||||
| rsc.io/pdf v0.1.1/go.mod h1:n8OzWcQ6Sp37PL01nO98y4iUCRdTGarVfzxY20ICaU4= | ||||
| stathat.com/c/consistent v1.0.0 h1:ezyc51EGcRPJUxfHGSgJjWzJdj3NiMU9pNfLNGiXV0c= | ||||
| stathat.com/c/consistent v1.0.0/go.mod h1:QkzMWzcbB+yQBL2AttO6sgsQS/JSTapcDISJalmCDS0= | ||||
|  | ||||
| @ -27,8 +27,8 @@ func NewAuthApi(client rpcclient.Auth) AuthApi { | ||||
| 	return AuthApi(client) | ||||
| } | ||||
| 
 | ||||
| func (o *AuthApi) UserToken(c *gin.Context) { | ||||
| 	a2r.Call(auth.AuthClient.UserToken, o.Client, c) | ||||
| func (o *AuthApi) GetAdminToken(c *gin.Context) { | ||||
| 	a2r.Call(auth.AuthClient.GetAdminToken, o.Client, c) | ||||
| } | ||||
| 
 | ||||
| func (o *AuthApi) GetUserToken(c *gin.Context) { | ||||
|  | ||||
| @ -62,3 +62,11 @@ func (o *ConversationApi) GetIncrementalConversation(c *gin.Context) { | ||||
| func (o *ConversationApi) GetOwnerConversation(c *gin.Context) { | ||||
| 	a2r.Call(conversation.ConversationClient.GetOwnerConversation, o.Client, c) | ||||
| } | ||||
| 
 | ||||
| func (o *ConversationApi) GetNotNotifyConversationIDs(c *gin.Context) { | ||||
| 	a2r.Call(conversation.ConversationClient.GetNotNotifyConversationIDs, o.Client, c) | ||||
| } | ||||
| 
 | ||||
| func (o *ConversationApi) GetPinnedConversationIDs(c *gin.Context) { | ||||
| 	a2r.Call(conversation.ConversationClient.GetPinnedConversationIDs, o.Client, c) | ||||
| } | ||||
|  | ||||
| @ -72,6 +72,10 @@ func (o *FriendApi) GetPaginationBlacks(c *gin.Context) { | ||||
| 	a2r.Call(relation.FriendClient.GetPaginationBlacks, o.Client, c) | ||||
| } | ||||
| 
 | ||||
| func (o *FriendApi) GetSpecifiedBlacks(c *gin.Context) { | ||||
| 	a2r.Call(relation.FriendClient.GetSpecifiedBlacks, o.Client, c) | ||||
| } | ||||
| 
 | ||||
| func (o *FriendApi) RemoveBlack(c *gin.Context) { | ||||
| 	a2r.Call(relation.FriendClient.RemoveBlack, o.Client, c) | ||||
| } | ||||
|  | ||||
| @ -35,8 +35,8 @@ func (o *GroupApi) SetGroupInfo(c *gin.Context) { | ||||
| 	a2r.Call(group.GroupClient.SetGroupInfo, o.Client, c) | ||||
| } | ||||
| 
 | ||||
| func (o *GroupApi) SetGroupInfoEX(c *gin.Context) { | ||||
| 	a2r.Call(group.GroupClient.SetGroupInfoEX, o.Client, c) | ||||
| func (o *GroupApi) SetGroupInfoEx(c *gin.Context) { | ||||
| 	a2r.Call(group.GroupClient.SetGroupInfoEx, o.Client, c) | ||||
| } | ||||
| 
 | ||||
| func (o *GroupApi) JoinGroup(c *gin.Context) { | ||||
| @ -67,6 +67,10 @@ func (o *GroupApi) GetGroupUsersReqApplicationList(c *gin.Context) { | ||||
| 	a2r.Call(group.GroupClient.GetGroupUsersReqApplicationList, o.Client, c) | ||||
| } | ||||
| 
 | ||||
| func (o *GroupApi) GetSpecifiedUserGroupRequestInfo(c *gin.Context) { | ||||
| 	a2r.Call(group.GroupClient.GetSpecifiedUserGroupRequestInfo, o.Client, c) | ||||
| } | ||||
| 
 | ||||
| func (o *GroupApi) GetGroupsInfo(c *gin.Context) { | ||||
| 	a2r.Call(group.GroupClient.GetGroupsInfo, o.Client, c) | ||||
| 	//a2r.Call(group.GroupClient.GetGroupsInfo, o.Client, c, a2r.NewNilReplaceOption(group.GroupClient.GetGroupsInfo)) | ||||
|  | ||||
							
								
								
									
										262
									
								
								internal/api/jssdk/jssdk.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										262
									
								
								internal/api/jssdk/jssdk.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,262 @@ | ||||
| package jssdk | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"github.com/openimsdk/protocol/conversation" | ||||
| 	"github.com/openimsdk/protocol/group" | ||||
| 	"github.com/openimsdk/protocol/jssdk" | ||||
| 	"github.com/openimsdk/protocol/msg" | ||||
| 	"github.com/openimsdk/protocol/relation" | ||||
| 	"github.com/openimsdk/protocol/sdkws" | ||||
| 	"github.com/openimsdk/protocol/user" | ||||
| 	"github.com/openimsdk/tools/mcontext" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| 	"sort" | ||||
| ) | ||||
| 
 | ||||
| const ( | ||||
| 	maxGetActiveConversation     = 500 | ||||
| 	defaultGetActiveConversation = 100 | ||||
| ) | ||||
| 
 | ||||
| func NewJSSdkApi(user user.UserClient, friend relation.FriendClient, group group.GroupClient, msg msg.MsgClient, conv conversation.ConversationClient) *JSSdk { | ||||
| 	return &JSSdk{ | ||||
| 		user:   user, | ||||
| 		friend: friend, | ||||
| 		group:  group, | ||||
| 		msg:    msg, | ||||
| 		conv:   conv, | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| type JSSdk struct { | ||||
| 	user   user.UserClient | ||||
| 	friend relation.FriendClient | ||||
| 	group  group.GroupClient | ||||
| 	msg    msg.MsgClient | ||||
| 	conv   conversation.ConversationClient | ||||
| } | ||||
| 
 | ||||
| func (x *JSSdk) GetActiveConversations(c *gin.Context) { | ||||
| 	call(c, x.getActiveConversations) | ||||
| } | ||||
| 
 | ||||
| func (x *JSSdk) GetConversations(c *gin.Context) { | ||||
| 	call(c, x.getConversations) | ||||
| } | ||||
| 
 | ||||
| func (x *JSSdk) fillConversations(ctx context.Context, conversations []*jssdk.ConversationMsg) error { | ||||
| 	if len(conversations) == 0 { | ||||
| 		return nil | ||||
| 	} | ||||
| 	var ( | ||||
| 		userIDs  []string | ||||
| 		groupIDs []string | ||||
| 	) | ||||
| 	for _, c := range conversations { | ||||
| 		if c.Conversation.GroupID == "" { | ||||
| 			userIDs = append(userIDs, c.Conversation.UserID) | ||||
| 		} else { | ||||
| 			groupIDs = append(groupIDs, c.Conversation.GroupID) | ||||
| 		} | ||||
| 	} | ||||
| 	var ( | ||||
| 		userMap   map[string]*sdkws.UserInfo | ||||
| 		friendMap map[string]*relation.FriendInfoOnly | ||||
| 		groupMap  map[string]*sdkws.GroupInfo | ||||
| 	) | ||||
| 	if len(userIDs) > 0 { | ||||
| 		users, err := field(ctx, x.user.GetDesignateUsers, &user.GetDesignateUsersReq{UserIDs: userIDs}, (*user.GetDesignateUsersResp).GetUsersInfo) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		friends, err := field(ctx, x.friend.GetFriendInfo, &relation.GetFriendInfoReq{OwnerUserID: conversations[0].Conversation.OwnerUserID, FriendUserIDs: userIDs}, (*relation.GetFriendInfoResp).GetFriendInfos) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		userMap = datautil.SliceToMap(users, (*sdkws.UserInfo).GetUserID) | ||||
| 		friendMap = datautil.SliceToMap(friends, (*relation.FriendInfoOnly).GetFriendUserID) | ||||
| 	} | ||||
| 	if len(groupIDs) > 0 { | ||||
| 		resp, err := x.group.GetGroupsInfo(ctx, &group.GetGroupsInfoReq{GroupIDs: groupIDs}) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		groupMap = datautil.SliceToMap(resp.GroupInfos, (*sdkws.GroupInfo).GetGroupID) | ||||
| 	} | ||||
| 	for _, c := range conversations { | ||||
| 		if c.Conversation.GroupID == "" { | ||||
| 			c.User = userMap[c.Conversation.UserID] | ||||
| 			c.Friend = friendMap[c.Conversation.UserID] | ||||
| 		} else { | ||||
| 			c.Group = groupMap[c.Conversation.GroupID] | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func (x *JSSdk) getActiveConversations(ctx context.Context, req *jssdk.GetActiveConversationsReq) (*jssdk.GetActiveConversationsResp, error) { | ||||
| 	if req.Count <= 0 || req.Count > maxGetActiveConversation { | ||||
| 		req.Count = defaultGetActiveConversation | ||||
| 	} | ||||
| 	req.OwnerUserID = mcontext.GetOpUserID(ctx) | ||||
| 	conversationIDs, err := field(ctx, x.conv.GetConversationIDs, | ||||
| 		&conversation.GetConversationIDsReq{UserID: req.OwnerUserID}, (*conversation.GetConversationIDsResp).GetConversationIDs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if len(conversationIDs) == 0 { | ||||
| 		return &jssdk.GetActiveConversationsResp{}, nil | ||||
| 	} | ||||
| 	readSeq, err := field(ctx, x.msg.GetHasReadSeqs, | ||||
| 		&msg.GetHasReadSeqsReq{UserID: req.OwnerUserID, ConversationIDs: conversationIDs}, (*msg.SeqsInfoResp).GetMaxSeqs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	activeConversation, err := field(ctx, x.msg.GetActiveConversation, | ||||
| 		&msg.GetActiveConversationReq{ConversationIDs: conversationIDs}, (*msg.GetActiveConversationResp).GetConversations) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if len(activeConversation) == 0 { | ||||
| 		return &jssdk.GetActiveConversationsResp{}, nil | ||||
| 	} | ||||
| 	sortConversations := sortActiveConversations{ | ||||
| 		Conversation: activeConversation, | ||||
| 	} | ||||
| 	if len(activeConversation) > 1 { | ||||
| 		pinnedConversationIDs, err := field(ctx, x.conv.GetPinnedConversationIDs, | ||||
| 			&conversation.GetPinnedConversationIDsReq{UserID: req.OwnerUserID}, (*conversation.GetPinnedConversationIDsResp).GetConversationIDs) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		sortConversations.PinnedConversationIDs = datautil.SliceSet(pinnedConversationIDs) | ||||
| 	} | ||||
| 	sort.Sort(&sortConversations) | ||||
| 	sortList := sortConversations.Top(int(req.Count)) | ||||
| 	conversations, err := field(ctx, x.conv.GetConversations, | ||||
| 		&conversation.GetConversationsReq{ | ||||
| 			OwnerUserID: req.OwnerUserID, | ||||
| 			ConversationIDs: datautil.Slice(sortList, func(c *msg.ActiveConversation) string { | ||||
| 				return c.ConversationID | ||||
| 			})}, (*conversation.GetConversationsResp).GetConversations) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	msgs, err := field(ctx, x.msg.GetSeqMessage, | ||||
| 		&msg.GetSeqMessageReq{ | ||||
| 			UserID: req.OwnerUserID, | ||||
| 			Conversations: datautil.Slice(sortList, func(c *msg.ActiveConversation) *msg.ConversationSeqs { | ||||
| 				return &msg.ConversationSeqs{ | ||||
| 					ConversationID: c.ConversationID, | ||||
| 					Seqs:           []int64{c.MaxSeq}, | ||||
| 				} | ||||
| 			}), | ||||
| 		}, (*msg.GetSeqMessageResp).GetMsgs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	conversationMap := datautil.SliceToMap(conversations, func(c *conversation.Conversation) string { | ||||
| 		return c.ConversationID | ||||
| 	}) | ||||
| 	resp := make([]*jssdk.ConversationMsg, 0, len(sortList)) | ||||
| 	for _, c := range sortList { | ||||
| 		conv, ok := conversationMap[c.ConversationID] | ||||
| 		if !ok { | ||||
| 			continue | ||||
| 		} | ||||
| 		var lastMsg *sdkws.MsgData | ||||
| 		if msgList, ok := msgs[c.ConversationID]; ok && len(msgList.Msgs) > 0 { | ||||
| 			lastMsg = msgList.Msgs[0] | ||||
| 		} | ||||
| 		resp = append(resp, &jssdk.ConversationMsg{ | ||||
| 			Conversation: conv, | ||||
| 			LastMsg:      lastMsg, | ||||
| 			MaxSeq:       c.MaxSeq, | ||||
| 			ReadSeq:      readSeq[c.ConversationID], | ||||
| 		}) | ||||
| 	} | ||||
| 	if err := x.fillConversations(ctx, resp); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	var unreadCount int64 | ||||
| 	for _, c := range activeConversation { | ||||
| 		count := c.MaxSeq - readSeq[c.ConversationID] | ||||
| 		if count > 0 { | ||||
| 			unreadCount += count | ||||
| 		} | ||||
| 	} | ||||
| 	return &jssdk.GetActiveConversationsResp{ | ||||
| 		Conversations: resp, | ||||
| 		UnreadCount:   unreadCount, | ||||
| 	}, nil | ||||
| } | ||||
| 
 | ||||
| func (x *JSSdk) getConversations(ctx context.Context, req *jssdk.GetConversationsReq) (*jssdk.GetConversationsResp, error) { | ||||
| 	req.OwnerUserID = mcontext.GetOpUserID(ctx) | ||||
| 	conversations, err := field(ctx, x.conv.GetConversations, &conversation.GetConversationsReq{OwnerUserID: req.OwnerUserID, ConversationIDs: req.ConversationIDs}, (*conversation.GetConversationsResp).GetConversations) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if len(conversations) == 0 { | ||||
| 		return &jssdk.GetConversationsResp{}, nil | ||||
| 	} | ||||
| 	req.ConversationIDs = datautil.Slice(conversations, func(c *conversation.Conversation) string { | ||||
| 		return c.ConversationID | ||||
| 	}) | ||||
| 	maxSeqs, err := field(ctx, x.msg.GetMaxSeqs, | ||||
| 		&msg.GetMaxSeqsReq{ConversationIDs: req.ConversationIDs}, (*msg.SeqsInfoResp).GetMaxSeqs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	readSeqs, err := field(ctx, x.msg.GetHasReadSeqs, | ||||
| 		&msg.GetHasReadSeqsReq{UserID: req.OwnerUserID, ConversationIDs: req.ConversationIDs}, (*msg.SeqsInfoResp).GetMaxSeqs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	conversationSeqs := make([]*msg.ConversationSeqs, 0, len(conversations)) | ||||
| 	for _, c := range conversations { | ||||
| 		if seq := maxSeqs[c.ConversationID]; seq > 0 { | ||||
| 			conversationSeqs = append(conversationSeqs, &msg.ConversationSeqs{ | ||||
| 				ConversationID: c.ConversationID, | ||||
| 				Seqs:           []int64{seq}, | ||||
| 			}) | ||||
| 		} | ||||
| 	} | ||||
| 	var msgs map[string]*sdkws.PullMsgs | ||||
| 	if len(conversationSeqs) > 0 { | ||||
| 		msgs, err = field(ctx, x.msg.GetSeqMessage, | ||||
| 			&msg.GetSeqMessageReq{UserID: req.OwnerUserID, Conversations: conversationSeqs}, (*msg.GetSeqMessageResp).GetMsgs) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	} | ||||
| 	resp := make([]*jssdk.ConversationMsg, 0, len(conversations)) | ||||
| 	for _, c := range conversations { | ||||
| 		var lastMsg *sdkws.MsgData | ||||
| 		if msgList, ok := msgs[c.ConversationID]; ok && len(msgList.Msgs) > 0 { | ||||
| 			lastMsg = msgList.Msgs[0] | ||||
| 		} | ||||
| 		resp = append(resp, &jssdk.ConversationMsg{ | ||||
| 			Conversation: c, | ||||
| 			LastMsg:      lastMsg, | ||||
| 			MaxSeq:       maxSeqs[c.ConversationID], | ||||
| 			ReadSeq:      readSeqs[c.ConversationID], | ||||
| 		}) | ||||
| 	} | ||||
| 	if err := x.fillConversations(ctx, resp); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	var unreadCount int64 | ||||
| 	for conversationID, maxSeq := range maxSeqs { | ||||
| 		count := maxSeq - readSeqs[conversationID] | ||||
| 		if count > 0 { | ||||
| 			unreadCount += count | ||||
| 		} | ||||
| 	} | ||||
| 	return &jssdk.GetConversationsResp{ | ||||
| 		Conversations: resp, | ||||
| 		UnreadCount:   unreadCount, | ||||
| 	}, nil | ||||
| } | ||||
							
								
								
									
										33
									
								
								internal/api/jssdk/sort.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								internal/api/jssdk/sort.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,33 @@ | ||||
| package jssdk | ||||
| 
 | ||||
| import "github.com/openimsdk/protocol/msg" | ||||
| 
 | ||||
| type sortActiveConversations struct { | ||||
| 	Conversation          []*msg.ActiveConversation | ||||
| 	PinnedConversationIDs map[string]struct{} | ||||
| } | ||||
| 
 | ||||
| func (s sortActiveConversations) Top(limit int) []*msg.ActiveConversation { | ||||
| 	if limit > 0 && len(s.Conversation) > limit { | ||||
| 		return s.Conversation[:limit] | ||||
| 	} | ||||
| 	return s.Conversation | ||||
| } | ||||
| 
 | ||||
| func (s sortActiveConversations) Len() int { | ||||
| 	return len(s.Conversation) | ||||
| } | ||||
| 
 | ||||
| func (s sortActiveConversations) Less(i, j int) bool { | ||||
| 	iv, jv := s.Conversation[i], s.Conversation[j] | ||||
| 	_, ip := s.PinnedConversationIDs[iv.ConversationID] | ||||
| 	_, jp := s.PinnedConversationIDs[jv.ConversationID] | ||||
| 	if ip != jp { | ||||
| 		return ip | ||||
| 	} | ||||
| 	return iv.LastTime > jv.LastTime | ||||
| } | ||||
| 
 | ||||
| func (s sortActiveConversations) Swap(i, j int) { | ||||
| 	s.Conversation[i], s.Conversation[j] = s.Conversation[j], s.Conversation[i] | ||||
| } | ||||
							
								
								
									
										77
									
								
								internal/api/jssdk/tools.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										77
									
								
								internal/api/jssdk/tools.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,77 @@ | ||||
| package jssdk | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"github.com/openimsdk/tools/a2r" | ||||
| 	"github.com/openimsdk/tools/apiresp" | ||||
| 	"github.com/openimsdk/tools/checker" | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| 	"google.golang.org/grpc" | ||||
| 	"google.golang.org/protobuf/proto" | ||||
| 	"io" | ||||
| 	"strings" | ||||
| ) | ||||
| 
 | ||||
| func field[A, B, C any](ctx context.Context, fn func(ctx context.Context, req *A, opts ...grpc.CallOption) (*B, error), req *A, get func(*B) C) (C, error) { | ||||
| 	resp, err := fn(ctx, req) | ||||
| 	if err != nil { | ||||
| 		var c C | ||||
| 		return c, err | ||||
| 	} | ||||
| 	return get(resp), nil | ||||
| } | ||||
| 
 | ||||
| func call[A, B any](c *gin.Context, fn func(ctx context.Context, req *A) (*B, error)) { | ||||
| 	var isJSON bool | ||||
| 	switch contentType := c.GetHeader("Content-Type"); { | ||||
| 	case contentType == "": | ||||
| 		isJSON = true | ||||
| 	case strings.Contains(contentType, "application/json"): | ||||
| 		isJSON = true | ||||
| 	case strings.Contains(contentType, "application/protobuf"): | ||||
| 	case strings.Contains(contentType, "application/x-protobuf"): | ||||
| 	default: | ||||
| 		apiresp.GinError(c, errs.ErrArgs.WrapMsg("unsupported content type")) | ||||
| 		return | ||||
| 	} | ||||
| 	var req *A | ||||
| 	if isJSON { | ||||
| 		var err error | ||||
| 		req, err = a2r.ParseRequest[A](c) | ||||
| 		if err != nil { | ||||
| 			apiresp.GinError(c, err) | ||||
| 			return | ||||
| 		} | ||||
| 	} else { | ||||
| 		body, err := io.ReadAll(c.Request.Body) | ||||
| 		if err != nil { | ||||
| 			apiresp.GinError(c, err) | ||||
| 			return | ||||
| 		} | ||||
| 		req = new(A) | ||||
| 		if err := proto.Unmarshal(body, any(req).(proto.Message)); err != nil { | ||||
| 			apiresp.GinError(c, err) | ||||
| 			return | ||||
| 		} | ||||
| 		if err := checker.Validate(&req); err != nil { | ||||
| 			apiresp.GinError(c, err) | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
| 	resp, err := fn(c, req) | ||||
| 	if err != nil { | ||||
| 		apiresp.GinError(c, err) | ||||
| 		return | ||||
| 	} | ||||
| 	if isJSON { | ||||
| 		apiresp.GinSuccess(c, resp) | ||||
| 		return | ||||
| 	} | ||||
| 	body, err := proto.Marshal(any(resp).(proto.Message)) | ||||
| 	if err != nil { | ||||
| 		apiresp.GinError(c, err) | ||||
| 		return | ||||
| 	} | ||||
| 	apiresp.GinSuccess(c, body) | ||||
| } | ||||
| @ -3,6 +3,10 @@ package api | ||||
| import ( | ||||
| 	"fmt" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/internal/api/jssdk" | ||||
| 
 | ||||
| 	"github.com/gin-contrib/gzip" | ||||
| 
 | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"github.com/gin-gonic/gin/binding" | ||||
| 	"github.com/go-playground/validator/v10" | ||||
| @ -22,6 +26,13 @@ import ( | ||||
| 	"github.com/openimsdk/tools/mw" | ||||
| ) | ||||
| 
 | ||||
| const ( | ||||
| 	NoCompression      = -1 | ||||
| 	DefaultCompression = 0 | ||||
| 	BestCompression    = 1 | ||||
| 	BestSpeed          = 2 | ||||
| ) | ||||
| 
 | ||||
| func prommetricsGin() gin.HandlerFunc { | ||||
| 	return func(c *gin.Context) { | ||||
| 		c.Next() | ||||
| @ -54,10 +65,19 @@ func newGinRouter(disCov discovery.SvcDiscoveryRegistry, config *Config) *gin.En | ||||
| 	conversationRpc := rpcclient.NewConversation(disCov, config.Share.RpcRegisterName.Conversation) | ||||
| 	authRpc := rpcclient.NewAuth(disCov, config.Share.RpcRegisterName.Auth) | ||||
| 	thirdRpc := rpcclient.NewThird(disCov, config.Share.RpcRegisterName.Third, config.API.Prometheus.GrafanaURL) | ||||
| 
 | ||||
| 	switch config.API.Api.CompressionLevel { | ||||
| 	case NoCompression: | ||||
| 	case DefaultCompression: | ||||
| 		r.Use(gzip.Gzip(gzip.DefaultCompression)) | ||||
| 	case BestCompression: | ||||
| 		r.Use(gzip.Gzip(gzip.BestCompression)) | ||||
| 	case BestSpeed: | ||||
| 		r.Use(gzip.Gzip(gzip.BestSpeed)) | ||||
| 	} | ||||
| 	r.Use(prommetricsGin(), gin.Recovery(), mw.CorsHandler(), mw.GinParseOperationID(), GinParseToken(authRpc)) | ||||
| 	u := NewUserApi(*userRpc) | ||||
| 	m := NewMessageApi(messageRpc, userRpc, config.Share.IMAdminUserID) | ||||
| 	j := jssdk.NewJSSdkApi(userRpc.Client, friendRpc.Client, groupRpc.Client, messageRpc.Client, conversationRpc.Client) | ||||
| 	userRouterGroup := r.Group("/user") | ||||
| 	{ | ||||
| 		userRouterGroup.POST("/user_register", u.UserRegister) | ||||
| @ -99,6 +119,7 @@ func newGinRouter(disCov discovery.SvcDiscoveryRegistry, config *Config) *gin.En | ||||
| 		friendRouterGroup.POST("/set_friend_remark", f.SetFriendRemark) | ||||
| 		friendRouterGroup.POST("/add_black", f.AddBlack) | ||||
| 		friendRouterGroup.POST("/get_black_list", f.GetPaginationBlacks) | ||||
| 		friendRouterGroup.POST("/get_specified_blacks", f.GetSpecifiedBlacks) | ||||
| 		friendRouterGroup.POST("/remove_black", f.RemoveBlack) | ||||
| 		friendRouterGroup.POST("/get_incremental_blacks", f.GetIncrementalBlacks) | ||||
| 		friendRouterGroup.POST("/import_friend", f.ImportFriends) | ||||
| @ -114,7 +135,7 @@ func newGinRouter(disCov discovery.SvcDiscoveryRegistry, config *Config) *gin.En | ||||
| 	{ | ||||
| 		groupRouterGroup.POST("/create_group", g.CreateGroup) | ||||
| 		groupRouterGroup.POST("/set_group_info", g.SetGroupInfo) | ||||
| 		groupRouterGroup.POST("/set_group_info_ex", g.SetGroupInfoEX) | ||||
| 		groupRouterGroup.POST("/set_group_info_ex", g.SetGroupInfoEx) | ||||
| 		groupRouterGroup.POST("/join_group", g.JoinGroup) | ||||
| 		groupRouterGroup.POST("/quit_group", g.QuitGroup) | ||||
| 		groupRouterGroup.POST("/group_application_response", g.ApplicationGroupResponse) | ||||
| @ -122,6 +143,7 @@ func newGinRouter(disCov discovery.SvcDiscoveryRegistry, config *Config) *gin.En | ||||
| 		groupRouterGroup.POST("/get_recv_group_applicationList", g.GetRecvGroupApplicationList) | ||||
| 		groupRouterGroup.POST("/get_user_req_group_applicationList", g.GetUserReqGroupApplicationList) | ||||
| 		groupRouterGroup.POST("/get_group_users_req_application_list", g.GetGroupUsersReqApplicationList) | ||||
| 		groupRouterGroup.POST("/get_specified_user_group_request_info", g.GetSpecifiedUserGroupRequestInfo) | ||||
| 		groupRouterGroup.POST("/get_groups_info", g.GetGroupsInfo) | ||||
| 		groupRouterGroup.POST("/kick_group", g.KickGroupMember) | ||||
| 		groupRouterGroup.POST("/get_group_members_info", g.GetGroupMembersInfo) | ||||
| @ -147,7 +169,7 @@ func newGinRouter(disCov discovery.SvcDiscoveryRegistry, config *Config) *gin.En | ||||
| 	authRouterGroup := r.Group("/auth") | ||||
| 	{ | ||||
| 		a := NewAuthApi(*authRpc) | ||||
| 		authRouterGroup.POST("/user_token", a.UserToken) | ||||
| 		authRouterGroup.POST("/get_admin_token", a.GetAdminToken) | ||||
| 		authRouterGroup.POST("/get_user_token", a.GetUserToken) | ||||
| 		authRouterGroup.POST("/parse_token", a.ParseToken) | ||||
| 		authRouterGroup.POST("/force_logout", a.ForceLogout) | ||||
| @ -214,6 +236,8 @@ func newGinRouter(disCov discovery.SvcDiscoveryRegistry, config *Config) *gin.En | ||||
| 		conversationGroup.POST("/get_full_conversation_ids", c.GetFullOwnerConversationIDs) | ||||
| 		conversationGroup.POST("/get_incremental_conversations", c.GetIncrementalConversation) | ||||
| 		conversationGroup.POST("/get_owner_conversation", c.GetOwnerConversation) | ||||
| 		conversationGroup.POST("/get_not_notify_conversation_ids", c.GetNotNotifyConversationIDs) | ||||
| 		conversationGroup.POST("/get_pinned_conversation_ids", c.GetPinnedConversationIDs) | ||||
| 	} | ||||
| 
 | ||||
| 	statisticsGroup := r.Group("/statistics") | ||||
| @ -223,6 +247,11 @@ func newGinRouter(disCov discovery.SvcDiscoveryRegistry, config *Config) *gin.En | ||||
| 		statisticsGroup.POST("/group/create", g.GroupCreateCount) | ||||
| 		statisticsGroup.POST("/group/active", m.GetActiveGroup) | ||||
| 	} | ||||
| 
 | ||||
| 	jssdk := r.Group("/jssdk") | ||||
| 	jssdk.POST("/get_conversations", j.GetConversations) | ||||
| 	jssdk.POST("/get_active_conversations", j.GetActiveConversations) | ||||
| 
 | ||||
| 	return r | ||||
| } | ||||
| 
 | ||||
| @ -259,7 +288,6 @@ func GinParseToken(authRPC *rpcclient.Auth) gin.HandlerFunc { | ||||
| 
 | ||||
| // Whitelist api not parse token | ||||
| var Whitelist = []string{ | ||||
| 	"/user/user_register", | ||||
| 	"/auth/user_token", | ||||
| 	"/auth/get_admin_token", | ||||
| 	"/auth/parse_token", | ||||
| } | ||||
|  | ||||
| @ -107,14 +107,14 @@ func (u *UserApi) GetUsersOnlineStatus(c *gin.Context) { | ||||
| 			if v2.UserID == v1 { | ||||
| 				flag = true | ||||
| 				res.UserID = v1 | ||||
| 				res.Status = constant.OnlineStatus | ||||
| 				res.Status = constant.Online | ||||
| 				res.DetailPlatformStatus = append(res.DetailPlatformStatus, v2.DetailPlatformStatus...) | ||||
| 				break | ||||
| 			} | ||||
| 		} | ||||
| 		if !flag { | ||||
| 			res.UserID = v1 | ||||
| 			res.Status = constant.OfflineStatus | ||||
| 			res.Status = constant.Offline | ||||
| 		} | ||||
| 		respResult = append(respResult, res) | ||||
| 	} | ||||
| @ -153,26 +153,26 @@ func (u *UserApi) GetUsersOnlineTokenDetail(c *gin.Context) { | ||||
| 	} | ||||
| 
 | ||||
| 	for _, v1 := range req.UserIDs { | ||||
| 		m := make(map[string][]string, 10) | ||||
| 		m := make(map[int32][]string, 10) | ||||
| 		flag = false | ||||
| 		temp := new(msggateway.SingleDetail) | ||||
| 		for _, v2 := range wsResult { | ||||
| 			if v2.UserID == v1 { | ||||
| 				flag = true | ||||
| 				temp.UserID = v1 | ||||
| 				temp.Status = constant.OnlineStatus | ||||
| 				temp.Status = constant.Online | ||||
| 				for _, status := range v2.DetailPlatformStatus { | ||||
| 					if v, ok := m[status.Platform]; ok { | ||||
| 						m[status.Platform] = append(v, status.Token) | ||||
| 					if v, ok := m[status.PlatformID]; ok { | ||||
| 						m[status.PlatformID] = append(v, status.Token) | ||||
| 					} else { | ||||
| 						m[status.Platform] = []string{status.Token} | ||||
| 						m[status.PlatformID] = []string{status.Token} | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 		for p, tokens := range m { | ||||
| 			t := new(msggateway.SinglePlatformToken) | ||||
| 			t.Platform = p | ||||
| 			t.PlatformID = p | ||||
| 			t.Token = tokens | ||||
| 			t.Total = int32(len(tokens)) | ||||
| 			temp.SinglePlatformToken = append(temp.SinglePlatformToken, t) | ||||
|  | ||||
| @ -22,6 +22,8 @@ import ( | ||||
| 	"sync/atomic" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"google.golang.org/protobuf/proto" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/msgprocessor" | ||||
| 	"github.com/openimsdk/protocol/constant" | ||||
| 	"github.com/openimsdk/protocol/sdkws" | ||||
| @ -30,7 +32,6 @@ import ( | ||||
| 	"github.com/openimsdk/tools/log" | ||||
| 	"github.com/openimsdk/tools/mcontext" | ||||
| 	"github.com/openimsdk/tools/utils/stringutil" | ||||
| 	"google.golang.org/protobuf/proto" | ||||
| ) | ||||
| 
 | ||||
| var ( | ||||
| @ -220,6 +221,10 @@ func (c *Client) handleMessage(message []byte) error { | ||||
| 		resp, messageErr = c.longConnServer.SendSignalMessage(ctx, binaryReq) | ||||
| 	case WSPullMsgBySeqList: | ||||
| 		resp, messageErr = c.longConnServer.PullMessageBySeqList(ctx, binaryReq) | ||||
| 	case WSPullMsg: | ||||
| 		resp, messageErr = c.longConnServer.GetSeqMessage(ctx, binaryReq) | ||||
| 	case WSGetConvMaxReadSeq: | ||||
| 		resp, messageErr = c.longConnServer.GetConversationsHasReadAndMaxSeq(ctx, binaryReq) | ||||
| 	case WsLogoutMsg: | ||||
| 		resp, messageErr = c.longConnServer.UserLogout(ctx, binaryReq) | ||||
| 	case WsSetBackgroundStatus: | ||||
|  | ||||
| @ -39,6 +39,8 @@ const ( | ||||
| 	WSPullMsgBySeqList    = 1002 | ||||
| 	WSSendMsg             = 1003 | ||||
| 	WSSendSignalMsg       = 1004 | ||||
| 	WSPullMsg             = 1005 | ||||
| 	WSGetConvMaxReadSeq   = 1006 | ||||
| 	WSPushMsg             = 2001 | ||||
| 	WSKickOnlineMsg       = 2002 | ||||
| 	WsLogoutMsg           = 2003 | ||||
|  | ||||
| @ -66,12 +66,16 @@ func (c *UserConnContext) Value(key any) any { | ||||
| } | ||||
| 
 | ||||
| func newContext(respWriter http.ResponseWriter, req *http.Request) *UserConnContext { | ||||
| 	remoteAddr := req.RemoteAddr | ||||
| 	if forwarded := req.Header.Get("X-Forwarded-For"); forwarded != "" { | ||||
| 		remoteAddr += "_" + forwarded | ||||
| 	} | ||||
| 	return &UserConnContext{ | ||||
| 		RespWriter: respWriter, | ||||
| 		Req:        req, | ||||
| 		Path:       req.URL.Path, | ||||
| 		Method:     req.Method, | ||||
| 		RemoteAddr: req.RemoteAddr, | ||||
| 		RemoteAddr: remoteAddr, | ||||
| 		ConnID:     encrypt.Md5(req.RemoteAddr + "_" + strconv.Itoa(int(timeutil.GetCurrentTimestampByMill()))), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @ -16,6 +16,8 @@ package msggateway | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"sync/atomic" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/servererrs" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/startrpc" | ||||
| @ -30,7 +32,6 @@ import ( | ||||
| 	"github.com/openimsdk/tools/mq/memamq" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| 	"google.golang.org/grpc" | ||||
| 	"sync/atomic" | ||||
| ) | ||||
| 
 | ||||
| func (s *Server) InitServer(ctx context.Context, config *Config, disCov discovery.SvcDiscoveryRegistry, server *grpc.Server) error { | ||||
| @ -111,15 +112,14 @@ func (s *Server) GetUsersOnlineStatus( | ||||
| 			} | ||||
| 
 | ||||
| 			ps := new(msggateway.GetUsersOnlineStatusResp_SuccessDetail) | ||||
| 			ps.Platform = constant.PlatformIDToName(client.PlatformID) | ||||
| 			ps.Status = constant.OnlineStatus | ||||
| 			ps.PlatformID = int32(client.PlatformID) | ||||
| 			ps.ConnID = client.ctx.GetConnID() | ||||
| 			ps.Token = client.token | ||||
| 			ps.IsBackground = client.IsBackground | ||||
| 			uresp.Status = constant.OnlineStatus | ||||
| 			uresp.Status = constant.Online | ||||
| 			uresp.DetailPlatformStatus = append(uresp.DetailPlatformStatus, ps) | ||||
| 		} | ||||
| 		if uresp.Status == constant.OnlineStatus { | ||||
| 		if uresp.Status == constant.Online { | ||||
| 			resp.SuccessResult = append(resp.SuccessResult, uresp) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| @ -58,7 +58,7 @@ func Start(ctx context.Context, index int, conf *Config) error { | ||||
| 	) | ||||
| 
 | ||||
| 	hubServer := NewServer(rpcPort, longServer, conf, func(srv *Server) error { | ||||
| 		longServer.online = rpccache.NewOnlineCache(srv.userRcp, nil, rdb, longServer.subscriberUserOnlineStatusChanges) | ||||
| 		longServer.online, _ = rpccache.NewOnlineCache(srv.userRcp, nil, rdb, false, longServer.subscriberUserOnlineStatusChanges) | ||||
| 		return nil | ||||
| 	}) | ||||
| 
 | ||||
|  | ||||
| @ -19,6 +19,8 @@ import ( | ||||
| 	"sync" | ||||
| 
 | ||||
| 	"github.com/go-playground/validator/v10" | ||||
| 	"google.golang.org/protobuf/proto" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcclient" | ||||
| 	"github.com/openimsdk/protocol/msg" | ||||
| @ -27,7 +29,6 @@ import ( | ||||
| 	"github.com/openimsdk/tools/discovery" | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| 	"github.com/openimsdk/tools/utils/jsonutil" | ||||
| 	"google.golang.org/protobuf/proto" | ||||
| ) | ||||
| 
 | ||||
| type Req struct { | ||||
| @ -94,6 +95,8 @@ type MessageHandler interface { | ||||
| 	SendMessage(context context.Context, data *Req) ([]byte, error) | ||||
| 	SendSignalMessage(context context.Context, data *Req) ([]byte, error) | ||||
| 	PullMessageBySeqList(context context.Context, data *Req) ([]byte, error) | ||||
| 	GetConversationsHasReadAndMaxSeq(context context.Context, data *Req) ([]byte, error) | ||||
| 	GetSeqMessage(context context.Context, data *Req) ([]byte, error) | ||||
| 	UserLogout(context context.Context, data *Req) ([]byte, error) | ||||
| 	SetUserDeviceBackground(context context.Context, data *Req) ([]byte, bool, error) | ||||
| } | ||||
| @ -175,7 +178,7 @@ func (g GrpcHandler) SendSignalMessage(context context.Context, data *Req) ([]by | ||||
| func (g GrpcHandler) PullMessageBySeqList(context context.Context, data *Req) ([]byte, error) { | ||||
| 	req := sdkws.PullMessageBySeqsReq{} | ||||
| 	if err := proto.Unmarshal(data.Data, &req); err != nil { | ||||
| 		return nil, errs.WrapMsg(err, "error unmarshaling request", "action", "unmarshal", "dataType", "PullMessageBySeqsReq") | ||||
| 		return nil, errs.WrapMsg(err, "err proto unmarshal", "action", "unmarshal", "dataType", "PullMessageBySeqsReq") | ||||
| 	} | ||||
| 	if err := g.validate.Struct(data); err != nil { | ||||
| 		return nil, errs.WrapMsg(err, "validation failed", "action", "validate", "dataType", "PullMessageBySeqsReq") | ||||
| @ -191,6 +194,44 @@ func (g GrpcHandler) PullMessageBySeqList(context context.Context, data *Req) ([ | ||||
| 	return c, nil | ||||
| } | ||||
| 
 | ||||
| func (g GrpcHandler) GetConversationsHasReadAndMaxSeq(context context.Context, data *Req) ([]byte, error) { | ||||
| 	req := msg.GetConversationsHasReadAndMaxSeqReq{} | ||||
| 	if err := proto.Unmarshal(data.Data, &req); err != nil { | ||||
| 		return nil, errs.WrapMsg(err, "err proto unmarshal", "action", "unmarshal", "dataType", "GetConversationsHasReadAndMaxSeq") | ||||
| 	} | ||||
| 	if err := g.validate.Struct(data); err != nil { | ||||
| 		return nil, errs.WrapMsg(err, "validation failed", "action", "validate", "dataType", "GetConversationsHasReadAndMaxSeq") | ||||
| 	} | ||||
| 	resp, err := g.msgRpcClient.GetConversationsHasReadAndMaxSeq(context, &req) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	c, err := proto.Marshal(resp) | ||||
| 	if err != nil { | ||||
| 		return nil, errs.WrapMsg(err, "error marshaling response", "action", "marshal", "dataType", "GetConversationsHasReadAndMaxSeq") | ||||
| 	} | ||||
| 	return c, nil | ||||
| } | ||||
| 
 | ||||
| func (g GrpcHandler) GetSeqMessage(context context.Context, data *Req) ([]byte, error) { | ||||
| 	req := msg.GetSeqMessageReq{} | ||||
| 	if err := proto.Unmarshal(data.Data, &req); err != nil { | ||||
| 		return nil, errs.WrapMsg(err, "error unmarshaling request", "action", "unmarshal", "dataType", "GetSeqMessage") | ||||
| 	} | ||||
| 	if err := g.validate.Struct(data); err != nil { | ||||
| 		return nil, errs.WrapMsg(err, "validation failed", "action", "validate", "dataType", "GetSeqMessage") | ||||
| 	} | ||||
| 	resp, err := g.msgRpcClient.GetSeqMessage(context, &req) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	c, err := proto.Marshal(resp) | ||||
| 	if err != nil { | ||||
| 		return nil, errs.WrapMsg(err, "error marshaling response", "action", "marshal", "dataType", "GetSeqMessage") | ||||
| 	} | ||||
| 	return c, nil | ||||
| } | ||||
| 
 | ||||
| func (g GrpcHandler) UserLogout(context context.Context, data *Req) ([]byte, error) { | ||||
| 	req := push.DelUserPushTokenReq{} | ||||
| 	if err := proto.Unmarshal(data.Data, &req); err != nil { | ||||
|  | ||||
| @ -90,6 +90,19 @@ func (ws *WsServer) ChangeOnlineStatus(concurrent int) { | ||||
| 		if _, err := ws.userClient.Client.SetUserOnlineStatus(ctx, req); err != nil { | ||||
| 			log.ZError(ctx, "update user online status", err) | ||||
| 		} | ||||
| 		for _, ss := range req.Status { | ||||
| 			for _, online := range ss.Online { | ||||
| 				client, _, _ := ws.clients.Get(ss.UserID, int(online)) | ||||
| 				back := false | ||||
| 				if len(client) > 0 { | ||||
| 					back = client[0].IsBackground | ||||
| 				} | ||||
| 				ws.webhookAfterUserOnline(ctx, &ws.msgGatewayConfig.WebhooksConfig.AfterUserOnline, ss.UserID, int(online), back, ss.ConnID) | ||||
| 			} | ||||
| 			for _, offline := range ss.Offline { | ||||
| 				ws.webhookAfterUserOffline(ctx, &ws.msgGatewayConfig.WebhooksConfig.AfterUserOffline, ss.UserID, int(offline), ss.ConnID) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	for i := 0; i < concurrent; i++ { | ||||
|  | ||||
| @ -1,17 +1,3 @@ | ||||
| // Copyright © 2023 OpenIM. All rights reserved. | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| //     http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
| 
 | ||||
| package msggateway | ||||
| 
 | ||||
| import ( | ||||
| @ -212,7 +198,6 @@ func (ws *WsServer) sendUserOnlineInfoToOtherNode(ctx context.Context, client *C | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
| 	wg := errgroup.Group{} | ||||
| 	wg.SetLimit(concurrentRequest) | ||||
| 
 | ||||
| @ -265,7 +250,7 @@ func (ws *WsServer) registerClient(client *Client) { | ||||
| 		if clientOK { | ||||
| 			ws.clients.Set(client.UserID, client) | ||||
| 			// There is already a connection to the platform | ||||
| 			log.ZInfo(client.ctx, "repeat login", "userID", client.UserID, "platformID", | ||||
| 			log.ZDebug(client.ctx, "repeat login", "userID", client.UserID, "platformID", | ||||
| 				client.PlatformID, "old remote addr", getRemoteAdders(oldClients)) | ||||
| 			ws.onlineUserConnNum.Add(1) | ||||
| 		} else { | ||||
| @ -293,7 +278,7 @@ func (ws *WsServer) registerClient(client *Client) { | ||||
| 
 | ||||
| 	wg.Wait() | ||||
| 
 | ||||
| 	log.ZInfo( | ||||
| 	log.ZDebug( | ||||
| 		client.ctx, | ||||
| 		"user online", | ||||
| 		"online user Num", | ||||
| @ -321,8 +306,32 @@ func (ws *WsServer) KickUserConn(client *Client) error { | ||||
| } | ||||
| 
 | ||||
| func (ws *WsServer) multiTerminalLoginChecker(clientOK bool, oldClients []*Client, newClient *Client) { | ||||
| 	switch ws.msgGatewayConfig.MsgGateway.MultiLoginPolicy { | ||||
| 	kickTokenFunc := func(kickClients []*Client) { | ||||
| 		var kickTokens []string | ||||
| 		ws.clients.DeleteClients(newClient.UserID, kickClients) | ||||
| 		for _, c := range kickClients { | ||||
| 			kickTokens = append(kickTokens, c.token) | ||||
| 			err := c.KickOnlineMessage() | ||||
| 			if err != nil { | ||||
| 				log.ZWarn(c.ctx, "KickOnlineMessage", err) | ||||
| 			} | ||||
| 		} | ||||
| 		ctx := mcontext.WithMustInfoCtx( | ||||
| 			[]string{newClient.ctx.GetOperationID(), newClient.ctx.GetUserID(), | ||||
| 				constant.PlatformIDToName(newClient.PlatformID), newClient.ctx.GetConnID()}, | ||||
| 		) | ||||
| 		if _, err := ws.authClient.KickTokens(ctx, kickTokens); err != nil { | ||||
| 			log.ZWarn(newClient.ctx, "kickTokens err", err) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	switch ws.msgGatewayConfig.Share.MultiLogin.Policy { | ||||
| 	case constant.DefalutNotKick: | ||||
| 	case constant.WebAndOther: | ||||
| 		if constant.PlatformIDToClass(newClient.PlatformID) == constant.WebPlatformStr { | ||||
| 			return | ||||
| 		} | ||||
| 		fallthrough | ||||
| 	case constant.PCAndOther: | ||||
| 		if constant.PlatformIDToClass(newClient.PlatformID) == constant.TerminalPC { | ||||
| 			return | ||||
| @ -347,6 +356,35 @@ func (ws *WsServer) multiTerminalLoginChecker(clientOK bool, oldClients []*Clien | ||||
| 			log.ZWarn(newClient.ctx, "InvalidateToken err", err, "userID", newClient.UserID, | ||||
| 				"platformID", newClient.PlatformID) | ||||
| 		} | ||||
| 	case constant.PcMobileAndWeb: | ||||
| 		clients, ok := ws.clients.GetAll(newClient.UserID) | ||||
| 		if !ok { | ||||
| 			return | ||||
| 		} | ||||
| 		var ( | ||||
| 			kickClients []*Client | ||||
| 		) | ||||
| 		for _, client := range clients { | ||||
| 			if constant.PlatformIDToClass(client.PlatformID) == constant.PlatformIDToClass(newClient.PlatformID) { | ||||
| 				kickClients = append(kickClients, client) | ||||
| 			} | ||||
| 		} | ||||
| 		kickTokenFunc(kickClients) | ||||
| 
 | ||||
| 	case constant.SingleTerminalLogin: | ||||
| 		clients, ok := ws.clients.GetAll(newClient.UserID) | ||||
| 		if !ok { | ||||
| 			return | ||||
| 		} | ||||
| 		var ( | ||||
| 			kickClients []*Client | ||||
| 		) | ||||
| 		for _, client := range clients { | ||||
| 			kickClients = append(kickClients, client) | ||||
| 		} | ||||
| 		kickTokenFunc(kickClients) | ||||
| 	case constant.Customize: | ||||
| 		// todo | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| @ -360,7 +398,7 @@ func (ws *WsServer) unregisterClient(client *Client) { | ||||
| 	ws.onlineUserConnNum.Add(-1) | ||||
| 	ws.subscription.DelClient(client) | ||||
| 	//ws.SetUserOnlineStatus(client.ctx, client, constant.Offline) | ||||
| 	log.ZInfo(client.ctx, "user offline", "close reason", client.closedErr, "online user Num", | ||||
| 	log.ZDebug(client.ctx, "user offline", "close reason", client.closedErr, "online user Num", | ||||
| 		ws.onlineUserNum.Load(), "online user conn Num", | ||||
| 		ws.onlineUserConnNum.Load(), | ||||
| 	) | ||||
| @ -425,6 +463,7 @@ func (ws *WsServer) wsHandler(w http.ResponseWriter, r *http.Request) { | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	log.ZDebug(connContext, "new conn", "token", connContext.GetToken()) | ||||
| 	// Create a WebSocket long connection object | ||||
| 	wsLongConn := newGWebSocket(WebSocket, ws.handshakeTimeout, ws.writeBufferSize) | ||||
| 	if err := wsLongConn.GenerateLongConn(w, r); err != nil { | ||||
|  | ||||
| @ -18,19 +18,20 @@ import ( | ||||
| 	"context" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 	"os" | ||||
| 	"os/signal" | ||||
| 	"syscall" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/prommetrics" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache/redis" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/database/mgo" | ||||
| 	"github.com/openimsdk/tools/db/mongoutil" | ||||
| 	"github.com/openimsdk/tools/db/redisutil" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| 	"net/http" | ||||
| 	"os" | ||||
| 	"os/signal" | ||||
| 	"syscall" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||
| 	kdisc "github.com/openimsdk/open-im-server/v3/pkg/common/discoveryregister" | ||||
| 	discRegister "github.com/openimsdk/open-im-server/v3/pkg/common/discoveryregister" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/controller" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcclient" | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| @ -65,6 +66,7 @@ type Config struct { | ||||
| func Start(ctx context.Context, index int, config *Config) error { | ||||
| 	log.CInfo(ctx, "MSG-TRANSFER server is initializing", "prometheusPorts", | ||||
| 		config.MsgTransfer.Prometheus.Ports, "index", index) | ||||
| 
 | ||||
| 	mgocli, err := mongoutil.NewMongoDB(ctx, config.MongodbConfig.Build()) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| @ -73,12 +75,13 @@ func Start(ctx context.Context, index int, config *Config) error { | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	client, err := kdisc.NewDiscoveryRegister(&config.Discovery, &config.Share) | ||||
| 	client, err := discRegister.NewDiscoveryRegister(&config.Discovery, &config.Share) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	client.AddOption(mw.GrpcClient(), grpc.WithTransportCredentials(insecure.NewCredentials()), | ||||
| 		grpc.WithDefaultServiceConfig(fmt.Sprintf(`{"LoadBalancingPolicy": "%s"}`, "round_robin"))) | ||||
| 
 | ||||
| 	msgModel := redis.NewMsgCache(rdb) | ||||
| 	msgDocModel, err := mgo.NewMsgMongo(mgocli.GetDB()) | ||||
| 	if err != nil { | ||||
| @ -94,20 +97,21 @@ func Start(ctx context.Context, index int, config *Config) error { | ||||
| 		return err | ||||
| 	} | ||||
| 	seqUserCache := redis.NewSeqUserCacheRedis(rdb, seqUser) | ||||
| 	msgDatabase, err := controller.NewCommonMsgDatabase(msgDocModel, msgModel, seqUserCache, seqConversationCache, &config.KafkaConfig) | ||||
| 	msgTransferDatabase, err := controller.NewMsgTransferDatabase(msgDocModel, msgModel, seqUserCache, seqConversationCache, &config.KafkaConfig) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	conversationRpcClient := rpcclient.NewConversationRpcClient(client, config.Share.RpcRegisterName.Conversation) | ||||
| 	groupRpcClient := rpcclient.NewGroupRpcClient(client, config.Share.RpcRegisterName.Group) | ||||
| 	historyCH, err := NewOnlineHistoryRedisConsumerHandler(&config.KafkaConfig, msgDatabase, &conversationRpcClient, &groupRpcClient) | ||||
| 	historyCH, err := NewOnlineHistoryRedisConsumerHandler(&config.KafkaConfig, msgTransferDatabase, &conversationRpcClient, &groupRpcClient) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	historyMongoCH, err := NewOnlineHistoryMongoConsumerHandler(&config.KafkaConfig, msgDatabase) | ||||
| 	historyMongoCH, err := NewOnlineHistoryMongoConsumerHandler(&config.KafkaConfig, msgTransferDatabase) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
| 	msgTransfer := &MsgTransfer{ | ||||
| 		historyCH:      historyCH, | ||||
| 		historyMongoCH: historyMongoCH, | ||||
|  | ||||
| @ -18,6 +18,10 @@ import ( | ||||
| 	"context" | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/IBM/sarama" | ||||
| 	"github.com/go-redis/redis" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||
| @ -33,9 +37,6 @@ import ( | ||||
| 	"github.com/openimsdk/tools/mq/kafka" | ||||
| 	"github.com/openimsdk/tools/utils/stringutil" | ||||
| 	"google.golang.org/protobuf/proto" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| ) | ||||
| 
 | ||||
| const ( | ||||
| @ -56,19 +57,19 @@ type OnlineHistoryRedisConsumerHandler struct { | ||||
| 
 | ||||
| 	redisMessageBatches *batcher.Batcher[sarama.ConsumerMessage] | ||||
| 
 | ||||
| 	msgDatabase           controller.CommonMsgDatabase | ||||
| 	msgTransferDatabase   controller.MsgTransferDatabase | ||||
| 	conversationRpcClient *rpcclient.ConversationRpcClient | ||||
| 	groupRpcClient        *rpcclient.GroupRpcClient | ||||
| } | ||||
| 
 | ||||
| func NewOnlineHistoryRedisConsumerHandler(kafkaConf *config.Kafka, database controller.CommonMsgDatabase, | ||||
| func NewOnlineHistoryRedisConsumerHandler(kafkaConf *config.Kafka, database controller.MsgTransferDatabase, | ||||
| 	conversationRpcClient *rpcclient.ConversationRpcClient, groupRpcClient *rpcclient.GroupRpcClient) (*OnlineHistoryRedisConsumerHandler, error) { | ||||
| 	historyConsumerGroup, err := kafka.NewMConsumerGroup(kafkaConf.Build(), kafkaConf.ToRedisGroupID, []string{kafkaConf.ToRedisTopic}, false) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	var och OnlineHistoryRedisConsumerHandler | ||||
| 	och.msgDatabase = database | ||||
| 	och.msgTransferDatabase = database | ||||
| 
 | ||||
| 	b := batcher.New[sarama.ConsumerMessage]( | ||||
| 		batcher.WithSize(size), | ||||
| @ -161,7 +162,7 @@ func (och *OnlineHistoryRedisConsumerHandler) doSetReadSeq(ctx context.Context, | ||||
| 		return | ||||
| 	} | ||||
| 	for key, seq := range readSeq { | ||||
| 		if err := och.msgDatabase.SetHasReadSeqToDB(ctx, key.userID, key.conversationID, seq); err != nil { | ||||
| 		if err := och.msgTransferDatabase.SetHasReadSeqToDB(ctx, key.userID, key.conversationID, seq); err != nil { | ||||
| 			log.ZError(ctx, "set read seq to db error", err, "userID", key.userID, "conversationID", key.conversationID, "seq", seq) | ||||
| 		} | ||||
| 	} | ||||
| @ -237,6 +238,11 @@ func (och *OnlineHistoryRedisConsumerHandler) categorizeMessageLists(totalMsgs [ | ||||
| } | ||||
| 
 | ||||
| func (och *OnlineHistoryRedisConsumerHandler) handleMsg(ctx context.Context, key, conversationID string, storageList, notStorageList []*ContextMsg) { | ||||
| 	log.ZInfo(ctx, "handle storage msg") | ||||
| 	for _, storageMsg := range storageList { | ||||
| 		log.ZDebug(ctx, "handle storage msg", "msg", storageMsg.message.String()) | ||||
| 	} | ||||
| 
 | ||||
| 	och.toPushTopic(ctx, key, conversationID, notStorageList) | ||||
| 	var storageMessageList []*sdkws.MsgData | ||||
| 	for _, msg := range storageList { | ||||
| @ -244,21 +250,25 @@ func (och *OnlineHistoryRedisConsumerHandler) handleMsg(ctx context.Context, key | ||||
| 	} | ||||
| 	if len(storageMessageList) > 0 { | ||||
| 		msg := storageMessageList[0] | ||||
| 		lastSeq, isNewConversation, err := och.msgDatabase.BatchInsertChat2Cache(ctx, conversationID, storageMessageList) | ||||
| 		lastSeq, isNewConversation, err := och.msgTransferDatabase.BatchInsertChat2Cache(ctx, conversationID, storageMessageList) | ||||
| 		if err != nil && !errors.Is(errs.Unwrap(err), redis.Nil) { | ||||
| 			log.ZError(ctx, "batch data insert to redis err", err, "storageMsgList", storageMessageList) | ||||
| 			return | ||||
| 		} | ||||
| 		log.ZInfo(ctx, "BatchInsertChat2Cache end") | ||||
| 
 | ||||
| 		if isNewConversation { | ||||
| 			switch msg.SessionType { | ||||
| 			case constant.ReadGroupChatType: | ||||
| 				log.ZInfo(ctx, "group chat first create conversation", "conversationID", | ||||
| 				log.ZDebug(ctx, "group chat first create conversation", "conversationID", | ||||
| 					conversationID) | ||||
| 				userIDs, err := och.groupRpcClient.GetGroupMemberIDs(ctx, msg.GroupID) | ||||
| 				if err != nil { | ||||
| 					log.ZWarn(ctx, "get group member ids error", err, "conversationID", | ||||
| 						conversationID) | ||||
| 				} else { | ||||
| 					log.ZInfo(ctx, "GetGroupMemberIDs end") | ||||
| 
 | ||||
| 					if err := och.conversationRpcClient.GroupChatFirstCreateConversation(ctx, | ||||
| 						msg.GroupID, userIDs); err != nil { | ||||
| 						log.ZWarn(ctx, "single chat first create conversation error", err, | ||||
| @ -277,13 +287,16 @@ func (och *OnlineHistoryRedisConsumerHandler) handleMsg(ctx context.Context, key | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		log.ZDebug(ctx, "success incr to next topic") | ||||
| 		err = och.msgDatabase.MsgToMongoMQ(ctx, key, conversationID, storageMessageList, lastSeq) | ||||
| 		log.ZInfo(ctx, "success incr to next topic") | ||||
| 		err = och.msgTransferDatabase.MsgToMongoMQ(ctx, key, conversationID, storageMessageList, lastSeq) | ||||
| 		if err != nil { | ||||
| 			log.ZError(ctx, "Msg To MongoDB MQ error", err, "conversationID", | ||||
| 				conversationID, "storageList", storageMessageList, "lastSeq", lastSeq) | ||||
| 		} | ||||
| 		log.ZInfo(ctx, "MsgToMongoMQ end") | ||||
| 
 | ||||
| 		och.toPushTopic(ctx, key, conversationID, storageList) | ||||
| 		log.ZInfo(ctx, "toPushTopic end") | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| @ -295,14 +308,14 @@ func (och *OnlineHistoryRedisConsumerHandler) handleNotification(ctx context.Con | ||||
| 		storageMessageList = append(storageMessageList, msg.message) | ||||
| 	} | ||||
| 	if len(storageMessageList) > 0 { | ||||
| 		lastSeq, _, err := och.msgDatabase.BatchInsertChat2Cache(ctx, conversationID, storageMessageList) | ||||
| 		lastSeq, _, err := och.msgTransferDatabase.BatchInsertChat2Cache(ctx, conversationID, storageMessageList) | ||||
| 		if err != nil { | ||||
| 			log.ZError(ctx, "notification batch insert to redis error", err, "conversationID", conversationID, | ||||
| 				"storageList", storageMessageList) | ||||
| 			return | ||||
| 		} | ||||
| 		log.ZDebug(ctx, "success to next topic", "conversationID", conversationID) | ||||
| 		err = och.msgDatabase.MsgToMongoMQ(ctx, key, conversationID, storageMessageList, lastSeq) | ||||
| 		err = och.msgTransferDatabase.MsgToMongoMQ(ctx, key, conversationID, storageMessageList, lastSeq) | ||||
| 		if err != nil { | ||||
| 			log.ZError(ctx, "Msg To MongoDB MQ error", err, "conversationID", | ||||
| 				conversationID, "storageList", storageMessageList, "lastSeq", lastSeq) | ||||
| @ -311,9 +324,10 @@ func (och *OnlineHistoryRedisConsumerHandler) handleNotification(ctx context.Con | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func (och *OnlineHistoryRedisConsumerHandler) toPushTopic(_ context.Context, key, conversationID string, msgs []*ContextMsg) { | ||||
| func (och *OnlineHistoryRedisConsumerHandler) toPushTopic(ctx context.Context, key, conversationID string, msgs []*ContextMsg) { | ||||
| 	for _, v := range msgs { | ||||
| 		och.msgDatabase.MsgToPushMQ(v.ctx, key, conversationID, v.message) | ||||
| 		log.ZDebug(ctx, "push msg to topic", "msg", v.message.String()) | ||||
| 		_, _, _ = och.msgTransferDatabase.MsgToPushMQ(v.ctx, key, conversationID, v.message) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| @ -338,7 +352,7 @@ func (och *OnlineHistoryRedisConsumerHandler) Cleanup(_ sarama.ConsumerGroupSess | ||||
| 
 | ||||
| func (och *OnlineHistoryRedisConsumerHandler) ConsumeClaim(session sarama.ConsumerGroupSession, | ||||
| 	claim sarama.ConsumerGroupClaim) error { // a instance in the consumer group | ||||
| 	log.ZInfo(context.Background(), "online new session msg come", "highWaterMarkOffset", | ||||
| 	log.ZDebug(context.Background(), "online new session msg come", "highWaterMarkOffset", | ||||
| 		claim.HighWaterMarkOffset(), "topic", claim.Topic(), "partition", claim.Partition()) | ||||
| 	och.redisMessageBatches.OnComplete = func(lastMessage *sarama.ConsumerMessage, totalCount int) { | ||||
| 		session.MarkMessage(lastMessage, "") | ||||
|  | ||||
| @ -16,6 +16,7 @@ package msgtransfer | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 
 | ||||
| 	"github.com/IBM/sarama" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/prommetrics" | ||||
| @ -28,10 +29,10 @@ import ( | ||||
| 
 | ||||
| type OnlineHistoryMongoConsumerHandler struct { | ||||
| 	historyConsumerGroup *kafka.MConsumerGroup | ||||
| 	msgDatabase          controller.CommonMsgDatabase | ||||
| 	msgTransferDatabase  controller.MsgTransferDatabase | ||||
| } | ||||
| 
 | ||||
| func NewOnlineHistoryMongoConsumerHandler(kafkaConf *config.Kafka, database controller.CommonMsgDatabase) (*OnlineHistoryMongoConsumerHandler, error) { | ||||
| func NewOnlineHistoryMongoConsumerHandler(kafkaConf *config.Kafka, database controller.MsgTransferDatabase) (*OnlineHistoryMongoConsumerHandler, error) { | ||||
| 	historyConsumerGroup, err := kafka.NewMConsumerGroup(kafkaConf.Build(), kafkaConf.ToMongoGroupID, []string{kafkaConf.ToMongoTopic}, true) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -39,7 +40,7 @@ func NewOnlineHistoryMongoConsumerHandler(kafkaConf *config.Kafka, database cont | ||||
| 
 | ||||
| 	mc := &OnlineHistoryMongoConsumerHandler{ | ||||
| 		historyConsumerGroup: historyConsumerGroup, | ||||
| 		msgDatabase:          database, | ||||
| 		msgTransferDatabase:  database, | ||||
| 	} | ||||
| 	return mc, nil | ||||
| } | ||||
| @ -56,8 +57,8 @@ func (mc *OnlineHistoryMongoConsumerHandler) handleChatWs2Mongo(ctx context.Cont | ||||
| 		log.ZError(ctx, "msgFromMQ.MsgData is empty", nil, "cMsg", cMsg) | ||||
| 		return | ||||
| 	} | ||||
| 	log.ZInfo(ctx, "mongo consumer recv msg", "msgs", msgFromMQ.String()) | ||||
| 	err = mc.msgDatabase.BatchInsertChat2DB(ctx, msgFromMQ.ConversationID, msgFromMQ.MsgData, msgFromMQ.LastSeq) | ||||
| 	log.ZDebug(ctx, "mongo consumer recv msg", "msgs", msgFromMQ.String()) | ||||
| 	err = mc.msgTransferDatabase.BatchInsertChat2DB(ctx, msgFromMQ.ConversationID, msgFromMQ.MsgData, msgFromMQ.LastSeq) | ||||
| 	if err != nil { | ||||
| 		log.ZError( | ||||
| 			ctx, | ||||
| @ -76,7 +77,7 @@ func (mc *OnlineHistoryMongoConsumerHandler) handleChatWs2Mongo(ctx context.Cont | ||||
| 	for _, msg := range msgFromMQ.MsgData { | ||||
| 		seqs = append(seqs, msg.Seq) | ||||
| 	} | ||||
| 	err = mc.msgDatabase.DeleteMessagesFromCache(ctx, msgFromMQ.ConversationID, seqs) | ||||
| 	err = mc.msgTransferDatabase.DeleteMessagesFromCache(ctx, msgFromMQ.ConversationID, seqs) | ||||
| 	if err != nil { | ||||
| 		log.ZError( | ||||
| 			ctx, | ||||
|  | ||||
| @ -1,29 +0,0 @@ | ||||
| package push | ||||
| 
 | ||||
| import ( | ||||
| 	"github.com/openimsdk/protocol/sdkws" | ||||
| 	"testing" | ||||
| ) | ||||
| 
 | ||||
| func TestName(t *testing.T) { | ||||
| 	var c ConsumerHandler | ||||
| 	c.readCh = make(chan *sdkws.MarkAsReadTips) | ||||
| 
 | ||||
| 	go c.loopRead() | ||||
| 
 | ||||
| 	go func() { | ||||
| 		for i := 0; ; i++ { | ||||
| 			seq := int64(i + 1) | ||||
| 			if seq%3 == 0 { | ||||
| 				seq = 1 | ||||
| 			} | ||||
| 			c.readCh <- &sdkws.MarkAsReadTips{ | ||||
| 				ConversationID:   "c100", | ||||
| 				MarkAsReadUserID: "u100", | ||||
| 				HasReadSeq:       seq, | ||||
| 			} | ||||
| 		} | ||||
| 	}() | ||||
| 
 | ||||
| 	select {} | ||||
| } | ||||
| @ -17,6 +17,7 @@ package dummy | ||||
| import ( | ||||
| 	"context" | ||||
| 	"github.com/openimsdk/open-im-server/v3/internal/push/offlinepush/options" | ||||
| 	"github.com/openimsdk/tools/log" | ||||
| ) | ||||
| 
 | ||||
| func NewClient() *Dummy { | ||||
| @ -27,5 +28,6 @@ type Dummy struct { | ||||
| } | ||||
| 
 | ||||
| func (d *Dummy) Push(ctx context.Context, userIDs []string, title, content string, opts *options.Opts) error { | ||||
| 	log.ZDebug(ctx, "dummy push") | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| @ -22,8 +22,8 @@ import ( | ||||
| 	"path/filepath" | ||||
| 	"strings" | ||||
| 
 | ||||
| 	firebase "firebase.google.com/go" | ||||
| 	"firebase.google.com/go/messaging" | ||||
| 	firebase "firebase.google.com/go/v4" | ||||
| 	"firebase.google.com/go/v4/messaging" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache" | ||||
| 	"github.com/openimsdk/protocol/constant" | ||||
| @ -99,7 +99,7 @@ func (f *Fcm) Push(ctx context.Context, userIDs []string, title, content string, | ||||
| 		apns := &messaging.APNSConfig{Payload: &messaging.APNSPayload{Aps: &messaging.Aps{Sound: opts.IOSPushSound}}} | ||||
| 		messageCount := len(messages) | ||||
| 		if messageCount >= SinglePushCountLimit { | ||||
| 			response, err := f.fcmMsgCli.SendAll(ctx, messages) | ||||
| 			response, err := f.fcmMsgCli.SendEach(ctx, messages) | ||||
| 			if err != nil { | ||||
| 				Fail = Fail + messageCount | ||||
| 				// Record push error | ||||
| @ -154,7 +154,7 @@ func (f *Fcm) Push(ctx context.Context, userIDs []string, title, content string, | ||||
| 	} | ||||
| 	messageCount := len(messages) | ||||
| 	if messageCount > 0 { | ||||
| 		response, err := f.fcmMsgCli.SendAll(ctx, messages) | ||||
| 		response, err := f.fcmMsgCli.SendEach(ctx, messages) | ||||
| 		if err != nil { | ||||
| 			Fail = Fail + messageCount | ||||
| 		} else { | ||||
|  | ||||
| @ -18,11 +18,11 @@ import ( | ||||
| 	"context" | ||||
| 	"crypto/sha256" | ||||
| 	"encoding/hex" | ||||
| 	"github.com/openimsdk/open-im-server/v3/internal/push/offlinepush/options" | ||||
| 	"strconv" | ||||
| 	"sync" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/internal/push/offlinepush/options" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache" | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| @ -91,6 +91,15 @@ func (g *Client) Push(ctx context.Context, userIDs []string, title, content stri | ||||
| 			for i, v := range s.GetSplitResult() { | ||||
| 				go func(index int, userIDs []string) { | ||||
| 					defer wg.Done() | ||||
| 					for i := 0; i < len(userIDs); i += maxNum { | ||||
| 						end := i + maxNum | ||||
| 						if end > len(userIDs) { | ||||
| 							end = len(userIDs) | ||||
| 						} | ||||
| 						if err = g.batchPush(ctx, token, userIDs[i:end], pushReq); err != nil { | ||||
| 							log.ZError(ctx, "batchPush failed", err, "index", index, "token", token, "req", pushReq) | ||||
| 						} | ||||
| 					} | ||||
| 					if err = g.batchPush(ctx, token, userIDs, pushReq); err != nil { | ||||
| 						log.ZError(ctx, "batchPush failed", err, "index", index, "token", token, "req", pushReq) | ||||
| 					} | ||||
|  | ||||
| @ -23,10 +23,13 @@ import ( | ||||
| 	"github.com/openimsdk/open-im-server/v3/internal/push/offlinepush/options" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache" | ||||
| 	"github.com/openimsdk/tools/log" | ||||
| 	"github.com/openimsdk/tools/mcontext" | ||||
| 	"strings" | ||||
| ) | ||||
| 
 | ||||
| const ( | ||||
| 	geTUI    = "geTui" | ||||
| 	geTUI    = "getui" | ||||
| 	firebase = "fcm" | ||||
| 	jPush    = "jpush" | ||||
| ) | ||||
| @ -38,6 +41,7 @@ type OfflinePusher interface { | ||||
| 
 | ||||
| func NewOfflinePusher(pushConf *config.Push, cache cache.ThirdCache, fcmConfigPath string) (OfflinePusher, error) { | ||||
| 	var offlinePusher OfflinePusher | ||||
| 	pushConf.Enable = strings.ToLower(pushConf.Enable) | ||||
| 	switch pushConf.Enable { | ||||
| 	case geTUI: | ||||
| 		offlinePusher = getui.NewClient(pushConf, cache) | ||||
| @ -47,6 +51,7 @@ func NewOfflinePusher(pushConf *config.Push, cache cache.ThirdCache, fcmConfigPa | ||||
| 		offlinePusher = jpush.NewClient(pushConf) | ||||
| 	default: | ||||
| 		offlinePusher = dummy.NewClient() | ||||
| 		log.ZWarn(mcontext.WithMustInfoCtx([]string{"push start", "admin", "admin", ""}), "Unknown push config", nil) | ||||
| 	} | ||||
| 	return offlinePusher, nil | ||||
| } | ||||
|  | ||||
							
								
								
									
										125
									
								
								internal/push/offlinepush_handler.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										125
									
								
								internal/push/offlinepush_handler.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,125 @@ | ||||
| package push | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 
 | ||||
| 	"github.com/IBM/sarama" | ||||
| 	"github.com/openimsdk/open-im-server/v3/internal/push/offlinepush" | ||||
| 	"github.com/openimsdk/open-im-server/v3/internal/push/offlinepush/options" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/prommetrics" | ||||
| 	"github.com/openimsdk/protocol/constant" | ||||
| 	pbpush "github.com/openimsdk/protocol/push" | ||||
| 	"github.com/openimsdk/protocol/sdkws" | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| 	"github.com/openimsdk/tools/log" | ||||
| 	"github.com/openimsdk/tools/mq/kafka" | ||||
| 	"github.com/openimsdk/tools/utils/jsonutil" | ||||
| 	"google.golang.org/protobuf/proto" | ||||
| ) | ||||
| 
 | ||||
| type OfflinePushConsumerHandler struct { | ||||
| 	OfflinePushConsumerGroup *kafka.MConsumerGroup | ||||
| 	offlinePusher            offlinepush.OfflinePusher | ||||
| } | ||||
| 
 | ||||
| func NewOfflinePushConsumerHandler(config *Config, offlinePusher offlinepush.OfflinePusher) (*OfflinePushConsumerHandler, error) { | ||||
| 	var offlinePushConsumerHandler OfflinePushConsumerHandler | ||||
| 	var err error | ||||
| 	offlinePushConsumerHandler.offlinePusher = offlinePusher | ||||
| 	offlinePushConsumerHandler.OfflinePushConsumerGroup, err = kafka.NewMConsumerGroup(config.KafkaConfig.Build(), config.KafkaConfig.ToOfflineGroupID, | ||||
| 		[]string{config.KafkaConfig.ToOfflinePushTopic}, true) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &offlinePushConsumerHandler, nil | ||||
| } | ||||
| 
 | ||||
| func (*OfflinePushConsumerHandler) Setup(sarama.ConsumerGroupSession) error   { return nil } | ||||
| func (*OfflinePushConsumerHandler) Cleanup(sarama.ConsumerGroupSession) error { return nil } | ||||
| func (o *OfflinePushConsumerHandler) ConsumeClaim(sess sarama.ConsumerGroupSession, claim sarama.ConsumerGroupClaim) error { | ||||
| 	for msg := range claim.Messages() { | ||||
| 		ctx := o.OfflinePushConsumerGroup.GetContextFromMsg(msg) | ||||
| 		o.handleMsg2OfflinePush(ctx, msg.Value) | ||||
| 		sess.MarkMessage(msg, "") | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func (o *OfflinePushConsumerHandler) handleMsg2OfflinePush(ctx context.Context, msg []byte) { | ||||
| 	offlinePushMsg := pbpush.PushMsgReq{} | ||||
| 	if err := proto.Unmarshal(msg, &offlinePushMsg); err != nil { | ||||
| 		log.ZError(ctx, "offline push Unmarshal msg err", err, "msg", string(msg)) | ||||
| 		return | ||||
| 	} | ||||
| 	if offlinePushMsg.MsgData == nil || offlinePushMsg.UserIDs == nil { | ||||
| 		log.ZError(ctx, "offline push msg is empty", errs.New("offlinePushMsg is empty"), "userIDs", offlinePushMsg.UserIDs, "msg", offlinePushMsg.MsgData) | ||||
| 		return | ||||
| 	} | ||||
| 	if offlinePushMsg.MsgData.Status == constant.MsgStatusSending { | ||||
| 		offlinePushMsg.MsgData.Status = constant.MsgStatusSendSuccess | ||||
| 	} | ||||
| 	log.ZInfo(ctx, "receive to OfflinePush MQ", "userIDs", offlinePushMsg.UserIDs, "msg", offlinePushMsg.MsgData) | ||||
| 
 | ||||
| 	err := o.offlinePushMsg(ctx, offlinePushMsg.MsgData, offlinePushMsg.UserIDs) | ||||
| 	if err != nil { | ||||
| 		log.ZWarn(ctx, "offline push failed", err, "msg", offlinePushMsg.String()) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func (o *OfflinePushConsumerHandler) getOfflinePushInfos(msg *sdkws.MsgData) (title, content string, opts *options.Opts, err error) { | ||||
| 	type AtTextElem struct { | ||||
| 		Text       string   `json:"text,omitempty"` | ||||
| 		AtUserList []string `json:"atUserList,omitempty"` | ||||
| 		IsAtSelf   bool     `json:"isAtSelf"` | ||||
| 	} | ||||
| 
 | ||||
| 	opts = &options.Opts{Signal: &options.Signal{}} | ||||
| 	if msg.OfflinePushInfo != nil { | ||||
| 		opts.IOSBadgeCount = msg.OfflinePushInfo.IOSBadgeCount | ||||
| 		opts.IOSPushSound = msg.OfflinePushInfo.IOSPushSound | ||||
| 		opts.Ex = msg.OfflinePushInfo.Ex | ||||
| 	} | ||||
| 
 | ||||
| 	if msg.OfflinePushInfo != nil { | ||||
| 		title = msg.OfflinePushInfo.Title | ||||
| 		content = msg.OfflinePushInfo.Desc | ||||
| 	} | ||||
| 	if title == "" { | ||||
| 		switch msg.ContentType { | ||||
| 		case constant.Text: | ||||
| 			fallthrough | ||||
| 		case constant.Picture: | ||||
| 			fallthrough | ||||
| 		case constant.Voice: | ||||
| 			fallthrough | ||||
| 		case constant.Video: | ||||
| 			fallthrough | ||||
| 		case constant.File: | ||||
| 			title = constant.ContentType2PushContent[int64(msg.ContentType)] | ||||
| 		case constant.AtText: | ||||
| 			ac := AtTextElem{} | ||||
| 			_ = jsonutil.JsonStringToStruct(string(msg.Content), &ac) | ||||
| 		case constant.SignalingNotification: | ||||
| 			title = constant.ContentType2PushContent[constant.SignalMsg] | ||||
| 		default: | ||||
| 			title = constant.ContentType2PushContent[constant.Common] | ||||
| 		} | ||||
| 	} | ||||
| 	if content == "" { | ||||
| 		content = title | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
| 
 | ||||
| func (o *OfflinePushConsumerHandler) offlinePushMsg(ctx context.Context, msg *sdkws.MsgData, offlinePushUserIDs []string) error { | ||||
| 	title, content, opts, err := o.getOfflinePushInfos(msg) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	err = o.offlinePusher.Push(ctx, offlinePushUserIDs, title, content, opts) | ||||
| 	if err != nil { | ||||
| 		prommetrics.MsgOfflinePushFailedCounter.Inc() | ||||
| 		return err | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
| @ -27,12 +27,12 @@ func newEmptyOnlinePusher() *emptyOnlinePusher { | ||||
| 
 | ||||
| func (emptyOnlinePusher) GetConnsAndOnlinePush(ctx context.Context, msg *sdkws.MsgData, | ||||
| 	pushToUserIDs []string) (wsResults []*msggateway.SingleMsgToUserResults, err error) { | ||||
| 	log.ZWarn(ctx, "emptyOnlinePusher GetConnsAndOnlinePush", nil) | ||||
| 	log.ZInfo(ctx, "emptyOnlinePusher GetConnsAndOnlinePush", nil) | ||||
| 	return nil, nil | ||||
| } | ||||
| func (u emptyOnlinePusher) GetOnlinePushFailedUserIDs(ctx context.Context, msg *sdkws.MsgData, | ||||
| 	wsResults []*msggateway.SingleMsgToUserResults, pushToUserIDs *[]string) []string { | ||||
| 	log.ZWarn(ctx, "emptyOnlinePusher GetOnlinePushFailedUserIDs", nil) | ||||
| 	log.ZInfo(ctx, "emptyOnlinePusher GetOnlinePushFailedUserIDs", nil) | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
|  | ||||
| @ -2,6 +2,7 @@ package push | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/internal/push/offlinepush" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache/redis" | ||||
| @ -17,12 +18,12 @@ type pushServer struct { | ||||
| 	disCov        discovery.SvcDiscoveryRegistry | ||||
| 	offlinePusher offlinepush.OfflinePusher | ||||
| 	pushCh        *ConsumerHandler | ||||
| 	offlinePushCh *OfflinePushConsumerHandler | ||||
| } | ||||
| 
 | ||||
| type Config struct { | ||||
| 	RpcConfig          config.Push | ||||
| 	RedisConfig        config.Redis | ||||
| 	MongodbConfig      config.Mongo | ||||
| 	KafkaConfig        config.Kafka | ||||
| 	NotificationConfig config.Notification | ||||
| 	Share              config.Share | ||||
| @ -55,18 +56,30 @@ func Start(ctx context.Context, config *Config, client discovery.SvcDiscoveryReg | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	database := controller.NewPushDatabase(cacheModel) | ||||
| 
 | ||||
| 	consumer, err := NewConsumerHandler(config, offlinePusher, rdb, client) | ||||
| 	database := controller.NewPushDatabase(cacheModel, &config.KafkaConfig) | ||||
| 
 | ||||
| 	consumer, err := NewConsumerHandler(config, database, offlinePusher, rdb, client) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
| 	offlinePushConsumer, err := NewOfflinePushConsumerHandler(config, offlinePusher) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
| 	pbpush.RegisterPushMsgServiceServer(server, &pushServer{ | ||||
| 		database:      database, | ||||
| 		disCov:        client, | ||||
| 		offlinePusher: offlinePusher, | ||||
| 		pushCh:        consumer, | ||||
| 		offlinePushCh: offlinePushConsumer, | ||||
| 	}) | ||||
| 
 | ||||
| 	go consumer.pushConsumerGroup.RegisterHandleAndConsumer(ctx, consumer) | ||||
| 
 | ||||
| 	go offlinePushConsumer.OfflinePushConsumerGroup.RegisterHandleAndConsumer(ctx, offlinePushConsumer) | ||||
| 
 | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| @ -1,33 +1,20 @@ | ||||
| // Copyright © 2023 OpenIM. All rights reserved. | ||||
| // | ||||
| // Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| // you may not use this file except in compliance with the License. | ||||
| // You may obtain a copy of the License at | ||||
| // | ||||
| //     http://www.apache.org/licenses/LICENSE-2.0 | ||||
| // | ||||
| // Unless required by applicable law or agreed to in writing, software | ||||
| // distributed under the License is distributed on an "AS IS" BASIS, | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| // See the License for the specific language governing permissions and | ||||
| // limitations under the License. | ||||
| 
 | ||||
| package push | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"encoding/json" | ||||
| 
 | ||||
| 	"github.com/IBM/sarama" | ||||
| 	"github.com/openimsdk/open-im-server/v3/internal/push/offlinepush" | ||||
| 	"github.com/openimsdk/open-im-server/v3/internal/push/offlinepush/options" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/prommetrics" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/controller" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/webhook" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/msgprocessor" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpccache" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcclient" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/util/conversationutil" | ||||
| 	"github.com/openimsdk/protocol/constant" | ||||
| 	pbchat "github.com/openimsdk/protocol/msg" | ||||
| 	"github.com/openimsdk/protocol/msggateway" | ||||
| 	pbpush "github.com/openimsdk/protocol/push" | ||||
| 	"github.com/openimsdk/protocol/sdkws" | ||||
| @ -40,12 +27,16 @@ import ( | ||||
| 	"github.com/openimsdk/tools/utils/timeutil" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"google.golang.org/protobuf/proto" | ||||
| 	"math/rand" | ||||
| 	"strconv" | ||||
| 	"time" | ||||
| ) | ||||
| 
 | ||||
| type ConsumerHandler struct { | ||||
| 	pushConsumerGroup      *kafka.MConsumerGroup | ||||
| 	offlinePusher          offlinepush.OfflinePusher | ||||
| 	onlinePusher           OnlinePusher | ||||
| 	pushDatabase           controller.PushDatabase | ||||
| 	onlineCache            *rpccache.OnlineCache | ||||
| 	groupLocalCache        *rpccache.GroupLocalCache | ||||
| 	conversationLocalCache *rpccache.ConversationLocalCache | ||||
| @ -56,7 +47,7 @@ type ConsumerHandler struct { | ||||
| 	config                 *Config | ||||
| } | ||||
| 
 | ||||
| func NewConsumerHandler(config *Config, offlinePusher offlinepush.OfflinePusher, rdb redis.UniversalClient, | ||||
| func NewConsumerHandler(config *Config, database controller.PushDatabase, offlinePusher offlinepush.OfflinePusher, rdb redis.UniversalClient, | ||||
| 	client discovery.SvcDiscoveryRegistry) (*ConsumerHandler, error) { | ||||
| 	var consumerHandler ConsumerHandler | ||||
| 	var err error | ||||
| @ -65,7 +56,9 @@ func NewConsumerHandler(config *Config, offlinePusher offlinepush.OfflinePusher, | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	userRpcClient := rpcclient.NewUserRpcClient(client, config.Share.RpcRegisterName.User, config.Share.IMAdminUserID) | ||||
| 
 | ||||
| 	consumerHandler.offlinePusher = offlinePusher | ||||
| 	consumerHandler.onlinePusher = NewOnlinePusher(client, config) | ||||
| 	consumerHandler.groupRpcClient = rpcclient.NewGroupRpcClient(client, config.Share.RpcRegisterName.Group) | ||||
| @ -75,42 +68,45 @@ func NewConsumerHandler(config *Config, offlinePusher offlinepush.OfflinePusher, | ||||
| 	consumerHandler.conversationLocalCache = rpccache.NewConversationLocalCache(consumerHandler.conversationRpcClient, &config.LocalCacheConfig, rdb) | ||||
| 	consumerHandler.webhookClient = webhook.NewWebhookClient(config.WebhooksConfig.URL) | ||||
| 	consumerHandler.config = config | ||||
| 	consumerHandler.onlineCache = rpccache.NewOnlineCache(userRpcClient, consumerHandler.groupLocalCache, rdb, nil) | ||||
| 	consumerHandler.pushDatabase = database | ||||
| 	consumerHandler.onlineCache, err = rpccache.NewOnlineCache(userRpcClient, consumerHandler.groupLocalCache, rdb, config.RpcConfig.FullUserCache, nil) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &consumerHandler, nil | ||||
| } | ||||
| 
 | ||||
| func (c *ConsumerHandler) handleMs2PsChat(ctx context.Context, msg []byte) { | ||||
| 	msgFromMQ := pbchat.PushMsgDataToMQ{} | ||||
| 	msgFromMQ := pbpush.PushMsgReq{} | ||||
| 	if err := proto.Unmarshal(msg, &msgFromMQ); err != nil { | ||||
| 		log.ZError(ctx, "push Unmarshal msg err", err, "msg", string(msg)) | ||||
| 		return | ||||
| 	} | ||||
| 	pbData := &pbpush.PushMsgReq{ | ||||
| 		MsgData:        msgFromMQ.MsgData, | ||||
| 		ConversationID: msgFromMQ.ConversationID, | ||||
| 	} | ||||
| 
 | ||||
| 	sec := msgFromMQ.MsgData.SendTime / 1000 | ||||
| 	nowSec := timeutil.GetCurrentTimestampBySecond() | ||||
| 
 | ||||
| 	if nowSec-sec > 10 { | ||||
| 		log.ZWarn(ctx, "long time push msg", nil, "msg", pbData.String(), "sec", sec, "nowSec", nowSec, "nowSec-sec", nowSec-sec) | ||||
| 		prommetrics.MsgLoneTimePushCounter.Inc() | ||||
| 		log.ZWarn(ctx, "it’s been a while since the message was sent", nil, "msg", msgFromMQ.String(), "sec", sec, "nowSec", nowSec, "nowSec-sec", nowSec-sec) | ||||
| 	} | ||||
| 	var err error | ||||
| 
 | ||||
| 	switch msgFromMQ.MsgData.SessionType { | ||||
| 	case constant.ReadGroupChatType: | ||||
| 		err = c.Push2Group(ctx, pbData.MsgData.GroupID, pbData.MsgData) | ||||
| 		err = c.Push2Group(ctx, msgFromMQ.MsgData.GroupID, msgFromMQ.MsgData) | ||||
| 	default: | ||||
| 		var pushUserIDList []string | ||||
| 		isSenderSync := datautil.GetSwitchFromOptions(pbData.MsgData.Options, constant.IsSenderSync) | ||||
| 		if !isSenderSync || pbData.MsgData.SendID == pbData.MsgData.RecvID { | ||||
| 			pushUserIDList = append(pushUserIDList, pbData.MsgData.RecvID) | ||||
| 		isSenderSync := datautil.GetSwitchFromOptions(msgFromMQ.MsgData.Options, constant.IsSenderSync) | ||||
| 		if !isSenderSync || msgFromMQ.MsgData.SendID == msgFromMQ.MsgData.RecvID { | ||||
| 			pushUserIDList = append(pushUserIDList, msgFromMQ.MsgData.RecvID) | ||||
| 		} else { | ||||
| 			pushUserIDList = append(pushUserIDList, pbData.MsgData.RecvID, pbData.MsgData.SendID) | ||||
| 			pushUserIDList = append(pushUserIDList, msgFromMQ.MsgData.RecvID, msgFromMQ.MsgData.SendID) | ||||
| 		} | ||||
| 		err = c.Push2User(ctx, pushUserIDList, pbData.MsgData) | ||||
| 		err = c.Push2User(ctx, pushUserIDList, msgFromMQ.MsgData) | ||||
| 	} | ||||
| 	if err != nil { | ||||
| 		log.ZWarn(ctx, "push failed", err, "msg", pbData.String()) | ||||
| 		log.ZWarn(ctx, "push failed", err, "msg", msgFromMQ.String()) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| @ -119,6 +115,14 @@ func (*ConsumerHandler) Setup(sarama.ConsumerGroupSession) error { return nil } | ||||
| func (*ConsumerHandler) Cleanup(sarama.ConsumerGroupSession) error { return nil } | ||||
| 
 | ||||
| func (c *ConsumerHandler) ConsumeClaim(sess sarama.ConsumerGroupSession, claim sarama.ConsumerGroupClaim) error { | ||||
| 	c.onlineCache.Lock.Lock() | ||||
| 	for c.onlineCache.CurrentPhase.Load() < rpccache.DoSubscribeOver { | ||||
| 		c.onlineCache.Cond.Wait() | ||||
| 	} | ||||
| 	c.onlineCache.Lock.Unlock() | ||||
| 	ctx := mcontext.SetOperationID(context.TODO(), strconv.FormatInt(time.Now().UnixNano()+int64(rand.Uint32()), 10)) | ||||
| 	log.ZInfo(ctx, "begin consume messages") | ||||
| 
 | ||||
| 	for msg := range claim.Messages() { | ||||
| 		ctx := c.pushConsumerGroup.GetContextFromMsg(msg) | ||||
| 		c.handleMs2PsChat(ctx, msg.Value) | ||||
| @ -129,20 +133,27 @@ func (c *ConsumerHandler) ConsumeClaim(sess sarama.ConsumerGroupSession, claim s | ||||
| 
 | ||||
| // Push2User Suitable for two types of conversations, one is SingleChatType and the other is NotificationChatType. | ||||
| func (c *ConsumerHandler) Push2User(ctx context.Context, userIDs []string, msg *sdkws.MsgData) (err error) { | ||||
| 	log.ZDebug(ctx, "Get msg from msg_transfer And push msg", "userIDs", userIDs, "msg", msg.String()) | ||||
| 	log.ZInfo(ctx, "Get msg from msg_transfer And push msg", "userIDs", userIDs, "msg", msg.String()) | ||||
| 	defer func(duration time.Time) { | ||||
| 		t := time.Since(duration) | ||||
| 		log.ZInfo(ctx, "Get msg from msg_transfer And push msg", "msg", msg.String(), "time cost", t) | ||||
| 	}(time.Now()) | ||||
| 	if err := c.webhookBeforeOnlinePush(ctx, &c.config.WebhooksConfig.BeforeOnlinePush, userIDs, msg); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	log.ZInfo(ctx, "webhookBeforeOnlinePush end") | ||||
| 
 | ||||
| 	wsResults, err := c.GetConnsAndOnlinePush(ctx, msg, userIDs) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
| 	log.ZDebug(ctx, "single and notification push result", "result", wsResults, "msg", msg, "push_to_userID", userIDs) | ||||
| 	log.ZInfo(ctx, "single and notification push result", "result", wsResults, "msg", msg, "push_to_userID", userIDs) | ||||
| 
 | ||||
| 	if !c.shouldPushOffline(ctx, msg) { | ||||
| 		return nil | ||||
| 	} | ||||
| 	log.ZInfo(ctx, "shouldPushOffline end") | ||||
| 
 | ||||
| 	for _, v := range wsResults { | ||||
| 		//message sender do not need offline push | ||||
| @ -161,7 +172,7 @@ func (c *ConsumerHandler) Push2User(ctx context.Context, userIDs []string, msg * | ||||
| 		offlinePushUserID, msg, nil); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
| 	log.ZInfo(ctx, "webhookBeforeOfflinePush end") | ||||
| 	err = c.offlinePushMsg(ctx, msg, offlinePushUserID) | ||||
| 	if err != nil { | ||||
| 		log.ZWarn(ctx, "offlinePushMsg failed", err, "offlinePushUserID", offlinePushUserID, "msg", msg) | ||||
| @ -183,21 +194,14 @@ func (c *ConsumerHandler) shouldPushOffline(_ context.Context, msg *sdkws.MsgDat | ||||
| } | ||||
| 
 | ||||
| func (c *ConsumerHandler) GetConnsAndOnlinePush(ctx context.Context, msg *sdkws.MsgData, pushToUserIDs []string) ([]*msggateway.SingleMsgToUserResults, error) { | ||||
| 	var ( | ||||
| 		onlineUserIDs  []string | ||||
| 		offlineUserIDs []string | ||||
| 	) | ||||
| 	for _, userID := range pushToUserIDs { | ||||
| 		online, err := c.onlineCache.GetUserOnline(ctx, userID) | ||||
| 	if msg != nil && msg.Status == constant.MsgStatusSending { | ||||
| 		msg.Status = constant.MsgStatusSendSuccess | ||||
| 	} | ||||
| 	onlineUserIDs, offlineUserIDs, err := c.onlineCache.GetUsersOnline(ctx, pushToUserIDs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 		if online { | ||||
| 			onlineUserIDs = append(onlineUserIDs, userID) | ||||
| 		} else { | ||||
| 			offlineUserIDs = append(offlineUserIDs, userID) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	log.ZDebug(ctx, "GetConnsAndOnlinePush online cache", "sendID", msg.SendID, "recvID", msg.RecvID, "groupID", msg.GroupID, "sessionType", msg.SessionType, "clientMsgID", msg.ClientMsgID, "serverMsgID", msg.ServerMsgID, "offlineUserIDs", offlineUserIDs, "onlineUserIDs", onlineUserIDs) | ||||
| 	var result []*msggateway.SingleMsgToUserResults | ||||
| 	if len(onlineUserIDs) > 0 { | ||||
| @ -216,57 +220,70 @@ func (c *ConsumerHandler) GetConnsAndOnlinePush(ctx context.Context, msg *sdkws. | ||||
| } | ||||
| 
 | ||||
| func (c *ConsumerHandler) Push2Group(ctx context.Context, groupID string, msg *sdkws.MsgData) (err error) { | ||||
| 	log.ZDebug(ctx, "Get group msg from msg_transfer and push msg", "msg", msg.String(), "groupID", groupID) | ||||
| 	log.ZInfo(ctx, "Get group msg from msg_transfer and push msg", "msg", msg.String(), "groupID", groupID) | ||||
| 	defer func(duration time.Time) { | ||||
| 		t := time.Since(duration) | ||||
| 		log.ZInfo(ctx, "Get group msg from msg_transfer and push msg end", "msg", msg.String(), "groupID", groupID, "time cost", t) | ||||
| 	}(time.Now()) | ||||
| 	var pushToUserIDs []string | ||||
| 	if err = c.webhookBeforeGroupOnlinePush(ctx, &c.config.WebhooksConfig.BeforeGroupOnlinePush, groupID, msg, | ||||
| 		&pushToUserIDs); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	log.ZInfo(ctx, "webhookBeforeGroupOnlinePush end") | ||||
| 
 | ||||
| 	err = c.groupMessagesHandler(ctx, groupID, &pushToUserIDs, msg) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	log.ZInfo(ctx, "groupMessagesHandler end") | ||||
| 
 | ||||
| 	wsResults, err := c.GetConnsAndOnlinePush(ctx, msg, pushToUserIDs) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
| 	log.ZDebug(ctx, "group push result", "result", wsResults, "msg", msg) | ||||
| 	log.ZInfo(ctx, "group push result", "result", wsResults, "msg", msg) | ||||
| 
 | ||||
| 	if !c.shouldPushOffline(ctx, msg) { | ||||
| 		return nil | ||||
| 	} | ||||
| 	needOfflinePushUserIDs := c.onlinePusher.GetOnlinePushFailedUserIDs(ctx, msg, wsResults, &pushToUserIDs) | ||||
| 
 | ||||
| 	log.ZInfo(ctx, "GetOnlinePushFailedUserIDs end") | ||||
| 	//filter some user, like don not disturb or don't need offline push etc. | ||||
| 	needOfflinePushUserIDs, err = c.filterGroupMessageOfflinePush(ctx, groupID, msg, needOfflinePushUserIDs) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	log.ZInfo(ctx, "filterGroupMessageOfflinePush end") | ||||
| 
 | ||||
| 	// Use offline push messaging | ||||
| 	if len(needOfflinePushUserIDs) > 0 { | ||||
| 		c.asyncOfflinePush(ctx, needOfflinePushUserIDs, msg) | ||||
| 	} | ||||
| 
 | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func (c *ConsumerHandler) asyncOfflinePush(ctx context.Context, needOfflinePushUserIDs []string, msg *sdkws.MsgData) { | ||||
| 	var offlinePushUserIDs []string | ||||
| 		err = c.webhookBeforeOfflinePush(ctx, &c.config.WebhooksConfig.BeforeOfflinePush, needOfflinePushUserIDs, msg, &offlinePushUserIDs) | ||||
| 	err := c.webhookBeforeOfflinePush(ctx, &c.config.WebhooksConfig.BeforeOfflinePush, needOfflinePushUserIDs, msg, &offlinePushUserIDs) | ||||
| 	if err != nil { | ||||
| 			return err | ||||
| 		log.ZWarn(ctx, "webhookBeforeOfflinePush failed", err, "msg", msg) | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	if len(offlinePushUserIDs) > 0 { | ||||
| 		needOfflinePushUserIDs = offlinePushUserIDs | ||||
| 	} | ||||
| 
 | ||||
| 		err = c.offlinePushMsg(ctx, msg, needOfflinePushUserIDs) | ||||
| 		if err != nil { | ||||
| 			log.ZWarn(ctx, "offlinePushMsg failed", err, "groupID", groupID, "msg", msg) | ||||
| 			return nil | ||||
| 	if err := c.pushDatabase.MsgToOfflinePushMQ(ctx, conversationutil.GenConversationUniqueKeyForSingle(msg.SendID, msg.RecvID), needOfflinePushUserIDs, msg); err != nil { | ||||
| 		log.ZError(ctx, "Msg To OfflinePush MQ error", err, "needOfflinePushUserIDs", | ||||
| 			needOfflinePushUserIDs, "msg", msg) | ||||
| 		prommetrics.SingleChatMsgProcessFailedCounter.Inc() | ||||
| 		return | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| 	} | ||||
| 
 | ||||
| 	return nil | ||||
| } | ||||
| func (c *ConsumerHandler) groupMessagesHandler(ctx context.Context, groupID string, pushToUserIDs *[]string, msg *sdkws.MsgData) (err error) { | ||||
| 	if len(*pushToUserIDs) == 0 { | ||||
| 		*pushToUserIDs, err = c.groupLocalCache.GetGroupMemberIDs(ctx, groupID) | ||||
| @ -300,7 +317,7 @@ func (c *ConsumerHandler) groupMessagesHandler(ctx context.Context, groupID stri | ||||
| 				if unmarshalNotificationElem(msg.Content, &tips) != nil { | ||||
| 					return err | ||||
| 				} | ||||
| 				log.ZInfo(ctx, "GroupDismissedNotificationInfo****", "groupID", groupID, "num", len(*pushToUserIDs), "list", pushToUserIDs) | ||||
| 				log.ZDebug(ctx, "GroupDismissedNotificationInfo****", "groupID", groupID, "num", len(*pushToUserIDs), "list", pushToUserIDs) | ||||
| 				if len(c.config.Share.IMAdminUserID) > 0 { | ||||
| 					ctx = mcontext.WithOpUserIDContext(ctx, c.config.Share.IMAdminUserID[0]) | ||||
| 				} | ||||
| @ -384,6 +401,7 @@ func (c *ConsumerHandler) getOfflinePushInfos(msg *sdkws.MsgData) (title, conten | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
| 
 | ||||
| func (c *ConsumerHandler) DeleteMemberAndSetConversationSeq(ctx context.Context, groupID string, userIDs []string) error { | ||||
| 	conversationID := msgprocessor.GetConversationIDBySessionType(constant.ReadGroupChatType, groupID) | ||||
| 	maxSeq, err := c.msgRpcClient.GetConversationMaxSeq(ctx, conversationID) | ||||
| @ -392,6 +410,7 @@ func (c *ConsumerHandler) DeleteMemberAndSetConversationSeq(ctx context.Context, | ||||
| 	} | ||||
| 	return c.conversationRpcClient.SetConversationMaxSeq(ctx, userIDs, conversationID, maxSeq) | ||||
| } | ||||
| 
 | ||||
| func unmarshalNotificationElem(bytes []byte, t any) error { | ||||
| 	var notification sdkws.NotificationElem | ||||
| 	if err := json.Unmarshal(bytes, ¬ification); err != nil { | ||||
|  | ||||
| @ -20,6 +20,7 @@ import ( | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||
| 	redis2 "github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache/redis" | ||||
| 	"github.com/openimsdk/tools/db/redisutil" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||
| @ -64,24 +65,33 @@ func Start(ctx context.Context, config *Config, client discovery.SvcDiscoveryReg | ||||
| 			redis2.NewTokenCacheModel(rdb, config.RpcConfig.TokenPolicy.Expire), | ||||
| 			config.Share.Secret, | ||||
| 			config.RpcConfig.TokenPolicy.Expire, | ||||
| 			config.Share.MultiLogin, | ||||
| 		), | ||||
| 		config: config, | ||||
| 	}) | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func (s *authServer) UserToken(ctx context.Context, req *pbauth.UserTokenReq) (*pbauth.UserTokenResp, error) { | ||||
| 	resp := pbauth.UserTokenResp{} | ||||
| func (s *authServer) GetAdminToken(ctx context.Context, req *pbauth.GetAdminTokenReq) (*pbauth.GetAdminTokenResp, error) { | ||||
| 	resp := pbauth.GetAdminTokenResp{} | ||||
| 	if req.Secret != s.config.Share.Secret { | ||||
| 		return nil, errs.ErrNoPermission.WrapMsg("secret invalid") | ||||
| 	} | ||||
| 
 | ||||
| 	if !datautil.Contain(req.UserID, s.config.Share.IMAdminUserID...) { | ||||
| 		return nil, errs.ErrArgs.WrapMsg("userID is error.", "userID", req.UserID, "adminUserID", s.config.Share.IMAdminUserID) | ||||
| 
 | ||||
| 	} | ||||
| 
 | ||||
| 	if _, err := s.userRpcClient.GetUserInfo(ctx, req.UserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	token, err := s.authDatabase.CreateToken(ctx, req.UserID, int(req.PlatformID)) | ||||
| 
 | ||||
| 	token, err := s.authDatabase.CreateToken(ctx, req.UserID, int(constant.AdminPlatformID)) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	prommetrics.UserLoginCounter.Inc() | ||||
| 	resp.Token = token | ||||
| 	resp.ExpireTimeSeconds = s.config.RpcConfig.TokenPolicy.Expire * 24 * 60 * 60 | ||||
| @ -92,6 +102,11 @@ func (s *authServer) GetUserToken(ctx context.Context, req *pbauth.GetUserTokenR | ||||
| 	if err := authverify.CheckAdmin(ctx, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	if req.PlatformID == constant.AdminPlatformID { | ||||
| 		return nil, errs.ErrNoPermission.WrapMsg("platformID invalid. platformID must not be adminPlatformID") | ||||
| 	} | ||||
| 
 | ||||
| 	resp := pbauth.GetUserTokenResp{} | ||||
| 
 | ||||
| 	if authverify.IsManagerUserID(req.UserID, s.config.Share.IMAdminUserID) { | ||||
| @ -215,3 +230,10 @@ func (s *authServer) InvalidateToken(ctx context.Context, req *pbauth.Invalidate | ||||
| 	} | ||||
| 	return &pbauth.InvalidateTokenResp{}, nil | ||||
| } | ||||
| 
 | ||||
| func (s *authServer) KickTokens(ctx context.Context, req *pbauth.KickTokensReq) (*pbauth.KickTokensResp, error) { | ||||
| 	if err := s.authDatabase.BatchSetTokenMapByUidPid(ctx, req.Tokens); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &pbauth.KickTokensResp{}, nil | ||||
| } | ||||
|  | ||||
| @ -278,8 +278,8 @@ func (c *conversationServer) SetConversations(ctx context.Context, req *pbconver | ||||
| 		if req.Conversation.MsgDestructTime != nil { | ||||
| 			m["msg_destruct_time"] = req.Conversation.MsgDestructTime.Value | ||||
| 		} | ||||
| 		if req.Conversation.MsgDestructTime != nil { | ||||
| 			m["msg_destruct_time"] = req.Conversation.MsgDestructTime.Value | ||||
| 		if req.Conversation.IsMsgDestruct != nil { | ||||
| 			m["is_msg_destruct"] = req.Conversation.IsMsgDestruct.Value | ||||
| 		} | ||||
| 		if req.Conversation.BurnDuration != nil { | ||||
| 			m["burn_duration"] = req.Conversation.BurnDuration.Value | ||||
| @ -710,3 +710,19 @@ func (c *conversationServer) GetConversationsNeedDestructMsgs(ctx context.Contex | ||||
| 
 | ||||
| 	return &pbconversation.GetConversationsNeedDestructMsgsResp{Conversations: convert.ConversationsDB2Pb(temp)}, nil | ||||
| } | ||||
| 
 | ||||
| func (c *conversationServer) GetNotNotifyConversationIDs(ctx context.Context, req *pbconversation.GetNotNotifyConversationIDsReq) (*pbconversation.GetNotNotifyConversationIDsResp, error) { | ||||
| 	conversationIDs, err := c.conversationDatabase.GetNotNotifyConversationIDs(ctx, req.UserID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &pbconversation.GetNotNotifyConversationIDsResp{ConversationIDs: conversationIDs}, nil | ||||
| } | ||||
| 
 | ||||
| func (c *conversationServer) GetPinnedConversationIDs(ctx context.Context, req *pbconversation.GetPinnedConversationIDsReq) (*pbconversation.GetPinnedConversationIDsResp, error) { | ||||
| 	conversationIDs, err := c.conversationDatabase.GetPinnedConversationIDs(ctx, req.UserID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &pbconversation.GetPinnedConversationIDsResp{ConversationIDs: conversationIDs}, nil | ||||
| } | ||||
| @ -218,6 +218,7 @@ func (s *groupServer) webhookAfterKickGroupMember(ctx context.Context, after *co | ||||
| 		CallbackCommand: callbackstruct.CallbackAfterKickGroupCommand, | ||||
| 		GroupID:         req.GroupID, | ||||
| 		KickedUserIDs:   req.KickedUserIDs, | ||||
| 		Reason:          req.Reason, | ||||
| 	} | ||||
| 	s.webhookClient.AsyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, &callbackstruct.CallbackKillGroupMemberResp{}, after) | ||||
| } | ||||
| @ -359,73 +360,73 @@ func (s *groupServer) webhookAfterSetGroupInfo(ctx context.Context, after *confi | ||||
| 	s.webhookClient.AsyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, &callbackstruct.CallbackAfterSetGroupInfoResp{}, after) | ||||
| } | ||||
| 
 | ||||
| func (s *groupServer) webhookBeforeSetGroupInfoEX(ctx context.Context, before *config.BeforeConfig, req *group.SetGroupInfoEXReq) error { | ||||
| func (s *groupServer) webhookBeforeSetGroupInfoEx(ctx context.Context, before *config.BeforeConfig, req *group.SetGroupInfoExReq) error { | ||||
| 	return webhook.WithCondition(ctx, before, func(ctx context.Context) error { | ||||
| 		cbReq := &callbackstruct.CallbackBeforeSetGroupInfoEXReq{ | ||||
| 			CallbackCommand: callbackstruct.CallbackBeforeSetGroupInfoCommand, | ||||
| 			GroupID:         req.GroupInfoForSetEX.GroupID, | ||||
| 			GroupName:       req.GroupInfoForSetEX.GroupName, | ||||
| 			Notification:    req.GroupInfoForSetEX.Notification, | ||||
| 			Introduction:    req.GroupInfoForSetEX.Introduction, | ||||
| 			FaceURL:         req.GroupInfoForSetEX.FaceURL, | ||||
| 		cbReq := &callbackstruct.CallbackBeforeSetGroupInfoExReq{ | ||||
| 			CallbackCommand: callbackstruct.CallbackBeforeSetGroupInfoExCommand, | ||||
| 			GroupID:         req.GroupID, | ||||
| 			GroupName:       req.GroupName, | ||||
| 			Notification:    req.Notification, | ||||
| 			Introduction:    req.Introduction, | ||||
| 			FaceURL:         req.FaceURL, | ||||
| 		} | ||||
| 
 | ||||
| 		if req.GroupInfoForSetEX.Ex != nil { | ||||
| 			cbReq.Ex = req.GroupInfoForSetEX.Ex | ||||
| 		if req.Ex != nil { | ||||
| 			cbReq.Ex = req.Ex | ||||
| 		} | ||||
| 		log.ZDebug(ctx, "debug CallbackBeforeSetGroupInfoEX", "ex", cbReq.Ex) | ||||
| 		log.ZDebug(ctx, "debug CallbackBeforeSetGroupInfoEx", "ex", cbReq.Ex) | ||||
| 
 | ||||
| 		if req.GroupInfoForSetEX.NeedVerification != nil { | ||||
| 			cbReq.NeedVerification = req.GroupInfoForSetEX.NeedVerification | ||||
| 		if req.NeedVerification != nil { | ||||
| 			cbReq.NeedVerification = req.NeedVerification | ||||
| 		} | ||||
| 		if req.GroupInfoForSetEX.LookMemberInfo != nil { | ||||
| 			cbReq.LookMemberInfo = req.GroupInfoForSetEX.LookMemberInfo | ||||
| 		if req.LookMemberInfo != nil { | ||||
| 			cbReq.LookMemberInfo = req.LookMemberInfo | ||||
| 		} | ||||
| 		if req.GroupInfoForSetEX.ApplyMemberFriend != nil { | ||||
| 			cbReq.ApplyMemberFriend = req.GroupInfoForSetEX.ApplyMemberFriend | ||||
| 		if req.ApplyMemberFriend != nil { | ||||
| 			cbReq.ApplyMemberFriend = req.ApplyMemberFriend | ||||
| 		} | ||||
| 
 | ||||
| 		resp := &callbackstruct.CallbackBeforeSetGroupInfoEXResp{} | ||||
| 		resp := &callbackstruct.CallbackBeforeSetGroupInfoExResp{} | ||||
| 
 | ||||
| 		if err := s.webhookClient.SyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, resp, before); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 
 | ||||
| 		datautil.NotNilReplace(&req.GroupInfoForSetEX.GroupID, &resp.GroupID) | ||||
| 		datautil.NotNilReplace(&req.GroupInfoForSetEX.GroupName, &resp.GroupName) | ||||
| 		datautil.NotNilReplace(&req.GroupInfoForSetEX.FaceURL, &resp.FaceURL) | ||||
| 		datautil.NotNilReplace(&req.GroupInfoForSetEX.Introduction, &resp.Introduction) | ||||
| 		datautil.NotNilReplace(&req.GroupInfoForSetEX.Ex, &resp.Ex) | ||||
| 		datautil.NotNilReplace(&req.GroupInfoForSetEX.NeedVerification, &resp.NeedVerification) | ||||
| 		datautil.NotNilReplace(&req.GroupInfoForSetEX.LookMemberInfo, &resp.LookMemberInfo) | ||||
| 		datautil.NotNilReplace(&req.GroupInfoForSetEX.ApplyMemberFriend, &resp.ApplyMemberFriend) | ||||
| 		datautil.NotNilReplace(&req.GroupID, &resp.GroupID) | ||||
| 		datautil.NotNilReplace(&req.GroupName, &resp.GroupName) | ||||
| 		datautil.NotNilReplace(&req.FaceURL, &resp.FaceURL) | ||||
| 		datautil.NotNilReplace(&req.Introduction, &resp.Introduction) | ||||
| 		datautil.NotNilReplace(&req.Ex, &resp.Ex) | ||||
| 		datautil.NotNilReplace(&req.NeedVerification, &resp.NeedVerification) | ||||
| 		datautil.NotNilReplace(&req.LookMemberInfo, &resp.LookMemberInfo) | ||||
| 		datautil.NotNilReplace(&req.ApplyMemberFriend, &resp.ApplyMemberFriend) | ||||
| 
 | ||||
| 		return nil | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func (s *groupServer) webhookAfterSetGroupInfoEX(ctx context.Context, after *config.AfterConfig, req *group.SetGroupInfoEXReq) { | ||||
| 	cbReq := &callbackstruct.CallbackAfterSetGroupInfoEXReq{ | ||||
| 		CallbackCommand: callbackstruct.CallbackAfterSetGroupInfoCommand, | ||||
| 		GroupID:         req.GroupInfoForSetEX.GroupID, | ||||
| 		GroupName:       req.GroupInfoForSetEX.GroupName, | ||||
| 		Notification:    req.GroupInfoForSetEX.Notification, | ||||
| 		Introduction:    req.GroupInfoForSetEX.Introduction, | ||||
| 		FaceURL:         req.GroupInfoForSetEX.FaceURL, | ||||
| func (s *groupServer) webhookAfterSetGroupInfoEx(ctx context.Context, after *config.AfterConfig, req *group.SetGroupInfoExReq) { | ||||
| 	cbReq := &callbackstruct.CallbackAfterSetGroupInfoExReq{ | ||||
| 		CallbackCommand: callbackstruct.CallbackAfterSetGroupInfoExCommand, | ||||
| 		GroupID:         req.GroupID, | ||||
| 		GroupName:       req.GroupName, | ||||
| 		Notification:    req.Notification, | ||||
| 		Introduction:    req.Introduction, | ||||
| 		FaceURL:         req.FaceURL, | ||||
| 	} | ||||
| 
 | ||||
| 	if req.GroupInfoForSetEX.Ex != nil { | ||||
| 		cbReq.Ex = req.GroupInfoForSetEX.Ex | ||||
| 	if req.Ex != nil { | ||||
| 		cbReq.Ex = req.Ex | ||||
| 	} | ||||
| 	if req.GroupInfoForSetEX.NeedVerification != nil { | ||||
| 		cbReq.NeedVerification = req.GroupInfoForSetEX.NeedVerification | ||||
| 	if req.NeedVerification != nil { | ||||
| 		cbReq.NeedVerification = req.NeedVerification | ||||
| 	} | ||||
| 	if req.GroupInfoForSetEX.LookMemberInfo != nil { | ||||
| 		cbReq.LookMemberInfo = req.GroupInfoForSetEX.LookMemberInfo | ||||
| 	if req.LookMemberInfo != nil { | ||||
| 		cbReq.LookMemberInfo = req.LookMemberInfo | ||||
| 	} | ||||
| 	if req.GroupInfoForSetEX.ApplyMemberFriend != nil { | ||||
| 		cbReq.ApplyMemberFriend = req.GroupInfoForSetEX.ApplyMemberFriend | ||||
| 	if req.ApplyMemberFriend != nil { | ||||
| 		cbReq.ApplyMemberFriend = req.ApplyMemberFriend | ||||
| 	} | ||||
| 
 | ||||
| 	s.webhookClient.AsyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, &callbackstruct.CallbackAfterSetGroupInfoEXResp{}, after) | ||||
| 	s.webhookClient.AsyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, &callbackstruct.CallbackAfterSetGroupInfoExResp{}, after) | ||||
| } | ||||
|  | ||||
| @ -20,6 +20,7 @@ import ( | ||||
| 
 | ||||
| 	pbgroup "github.com/openimsdk/protocol/group" | ||||
| 	"github.com/openimsdk/protocol/sdkws" | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| 	"github.com/openimsdk/tools/mcontext" | ||||
| ) | ||||
| 
 | ||||
| @ -54,11 +55,15 @@ func UpdateGroupInfoMap(ctx context.Context, group *sdkws.GroupInfoForSet) map[s | ||||
| 	return m | ||||
| } | ||||
| 
 | ||||
| func UpdateGroupInfoEXMap(ctx context.Context, group *sdkws.GroupInfoForSetEX) map[string]any { | ||||
| func UpdateGroupInfoExMap(ctx context.Context, group *pbgroup.SetGroupInfoExReq) (map[string]any, error) { | ||||
| 	m := make(map[string]any) | ||||
| 
 | ||||
| 	if group.GroupName != "" { | ||||
| 		m["group_name"] = group.GroupName | ||||
| 	if group.GroupName != nil { | ||||
| 		if group.GroupName.Value != "" { | ||||
| 			m["group_name"] = group.GroupName.Value | ||||
| 		} else { | ||||
| 			return nil, errs.ErrArgs.WrapMsg("group name is empty") | ||||
| 		} | ||||
| 	} | ||||
| 	if group.Notification != nil { | ||||
| 		m["notification"] = group.Notification.Value | ||||
| @ -84,7 +89,7 @@ func UpdateGroupInfoEXMap(ctx context.Context, group *sdkws.GroupInfoForSetEX) m | ||||
| 		m["ex"] = group.Ex.Value | ||||
| 	} | ||||
| 
 | ||||
| 	return m | ||||
| 	return m, nil | ||||
| } | ||||
| 
 | ||||
| func UpdateGroupStatusMap(status int) map[string]any { | ||||
|  | ||||
| @ -167,11 +167,11 @@ func (g *groupServer) CheckGroupAdmin(ctx context.Context, groupID string) error | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func (g *groupServer) GetPublicUserInfoMap(ctx context.Context, userIDs []string, complete bool) (map[string]*sdkws.PublicUserInfo, error) { | ||||
| func (g *groupServer) GetPublicUserInfoMap(ctx context.Context, userIDs []string) (map[string]*sdkws.PublicUserInfo, error) { | ||||
| 	if len(userIDs) == 0 { | ||||
| 		return map[string]*sdkws.PublicUserInfo{}, nil | ||||
| 	} | ||||
| 	users, err := g.user.GetPublicUserInfos(ctx, userIDs, complete) | ||||
| 	users, err := g.user.GetPublicUserInfos(ctx, userIDs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| @ -304,6 +304,13 @@ func (g *groupServer) CreateGroup(ctx context.Context, req *pbgroup.CreateGroupR | ||||
| 	} | ||||
| 	g.notification.GroupCreatedNotification(ctx, tips) | ||||
| 
 | ||||
| 	if req.GroupInfo.Notification != "" { | ||||
| 		g.notification.GroupInfoSetAnnouncementNotification(ctx, &sdkws.GroupInfoSetAnnouncementTips{ | ||||
| 			Group:  tips.Group, | ||||
| 			OpUser: tips.OpUser, | ||||
| 		}) | ||||
| 	} | ||||
| 
 | ||||
| 	reqCallBackAfter := &pbgroup.CreateGroupReq{ | ||||
| 		MemberUserIDs: userIDs, | ||||
| 		GroupInfo:     resp.GroupInfo, | ||||
| @ -458,7 +465,7 @@ func (g *groupServer) InviteUserToGroup(ctx context.Context, req *pbgroup.Invite | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	if err = g.notification.MemberEnterNotification(ctx, req.GroupID, req.InvitedUserIDs...); err != nil { | ||||
| 	if err = g.notification.GroupApplicationAgreeMemberEnterNotification(ctx, req.GroupID, opUserID, req.InvitedUserIDs...); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &pbgroup.InviteUserToGroupResp{}, nil | ||||
| @ -689,7 +696,7 @@ func (g *groupServer) GetGroupApplicationList(ctx context.Context, req *pbgroup. | ||||
| 		userIDs = append(userIDs, gr.UserID) | ||||
| 	} | ||||
| 	userIDs = datautil.Distinct(userIDs) | ||||
| 	userMap, err := g.user.GetPublicUserInfoMap(ctx, userIDs, true) | ||||
| 	userMap, err := g.user.GetPublicUserInfoMap(ctx, userIDs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| @ -833,7 +840,7 @@ func (g *groupServer) GroupApplicationResponse(ctx context.Context, req *pbgroup | ||||
| 		if member == nil { | ||||
| 			log.ZDebug(ctx, "GroupApplicationResponse", "member is nil") | ||||
| 		} else { | ||||
| 			if err = g.notification.MemberEnterNotification(ctx, req.GroupID, req.FromUserID); err != nil { | ||||
| 			if err = g.notification.GroupApplicationAgreeMemberEnterNotification(ctx, req.GroupID, groupRequest.InviterUserID, req.FromUserID); err != nil { | ||||
| 				return nil, err | ||||
| 			} | ||||
| 		} | ||||
| @ -1028,12 +1035,12 @@ func (g *groupServer) SetGroupInfo(ctx context.Context, req *pbgroup.SetGroupInf | ||||
| 			} | ||||
| 			resp, err := g.GetGroupMemberUserIDs(ctx, &pbgroup.GetGroupMemberUserIDsReq{GroupID: req.GroupInfoForSet.GroupID}) | ||||
| 			if err != nil { | ||||
| 				log.ZWarn(ctx, "GetGroupMemberIDs", err) | ||||
| 				log.ZWarn(ctx, "GetGroupMemberIDs is failed.", err) | ||||
| 				return | ||||
| 			} | ||||
| 			conversation.GroupAtType = &wrapperspb.Int32Value{Value: constant.GroupNotification} | ||||
| 			if err := g.conversationRpcClient.SetConversations(ctx, resp.UserIDs, conversation); err != nil { | ||||
| 				log.ZWarn(ctx, "SetConversations", err, resp.UserIDs, conversation) | ||||
| 				log.ZWarn(ctx, "SetConversations", err, "UserIDs", resp.UserIDs, "conversation", conversation) | ||||
| 			} | ||||
| 		}() | ||||
| 		g.notification.GroupInfoSetAnnouncementNotification(ctx, &sdkws.GroupInfoSetAnnouncementTips{Group: tips.Group, OpUser: tips.OpUser}) | ||||
| @ -1051,13 +1058,13 @@ func (g *groupServer) SetGroupInfo(ctx context.Context, req *pbgroup.SetGroupInf | ||||
| 	return &pbgroup.SetGroupInfoResp{}, nil | ||||
| } | ||||
| 
 | ||||
| func (g *groupServer) SetGroupInfoEX(ctx context.Context, req *pbgroup.SetGroupInfoEXReq) (*pbgroup.SetGroupInfoEXResp, error) { | ||||
| func (g *groupServer) SetGroupInfoEx(ctx context.Context, req *pbgroup.SetGroupInfoExReq) (*pbgroup.SetGroupInfoExResp, error) { | ||||
| 	var opMember *model.GroupMember | ||||
| 
 | ||||
| 	if !authverify.IsAppManagerUid(ctx, g.config.Share.IMAdminUserID) { | ||||
| 		var err error | ||||
| 
 | ||||
| 		opMember, err = g.db.TakeGroupMember(ctx, req.GroupInfoForSetEX.GroupID, mcontext.GetOpUserID(ctx)) | ||||
| 		opMember, err = g.db.TakeGroupMember(ctx, req.GroupID, mcontext.GetOpUserID(ctx)) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| @ -1071,11 +1078,11 @@ func (g *groupServer) SetGroupInfoEX(ctx context.Context, req *pbgroup.SetGroupI | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	if err := g.webhookBeforeSetGroupInfoEX(ctx, &g.config.WebhooksConfig.BeforeSetGroupInfoEX, req); err != nil && err != servererrs.ErrCallbackContinue { | ||||
| 	if err := g.webhookBeforeSetGroupInfoEx(ctx, &g.config.WebhooksConfig.BeforeSetGroupInfoEx, req); err != nil && err != servererrs.ErrCallbackContinue { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	group, err := g.db.TakeGroup(ctx, req.GroupInfoForSetEX.GroupID) | ||||
| 	group, err := g.db.TakeGroup(ctx, req.GroupID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| @ -1097,16 +1104,20 @@ func (g *groupServer) SetGroupInfoEX(ctx context.Context, req *pbgroup.SetGroupI | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	updatedData := UpdateGroupInfoEXMap(ctx, req.GroupInfoForSetEX) | ||||
| 	updatedData, err := UpdateGroupInfoExMap(ctx, req) | ||||
| 	if len(updatedData) == 0 { | ||||
| 		return &pbgroup.SetGroupInfoEXResp{}, nil | ||||
| 		return &pbgroup.SetGroupInfoExResp{}, nil | ||||
| 	} | ||||
| 
 | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	if err := g.db.UpdateGroup(ctx, group.GroupID, updatedData); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	group, err = g.db.TakeGroup(ctx, req.GroupInfoForSetEX.GroupID) | ||||
| 	group, err = g.db.TakeGroup(ctx, req.GroupID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| @ -1122,43 +1133,46 @@ func (g *groupServer) SetGroupInfoEX(ctx context.Context, req *pbgroup.SetGroupI | ||||
| 	} | ||||
| 
 | ||||
| 	num := len(updatedData) | ||||
| 	if req.GroupInfoForSetEX.Notification != nil { | ||||
| 	if req.Notification != nil { | ||||
| 		num-- | ||||
| 
 | ||||
| 		if req.Notification.Value != "" { | ||||
| 			func() { | ||||
| 				conversation := &pbconversation.ConversationReq{ | ||||
| 				ConversationID:   msgprocessor.GetConversationIDBySessionType(constant.ReadGroupChatType, req.GroupInfoForSetEX.GroupID), | ||||
| 					ConversationID:   msgprocessor.GetConversationIDBySessionType(constant.ReadGroupChatType, req.GroupID), | ||||
| 					ConversationType: constant.ReadGroupChatType, | ||||
| 				GroupID:          req.GroupInfoForSetEX.GroupID, | ||||
| 					GroupID:          req.GroupID, | ||||
| 				} | ||||
| 
 | ||||
| 			resp, err := g.GetGroupMemberUserIDs(ctx, &pbgroup.GetGroupMemberUserIDsReq{GroupID: req.GroupInfoForSetEX.GroupID}) | ||||
| 				resp, err := g.GetGroupMemberUserIDs(ctx, &pbgroup.GetGroupMemberUserIDsReq{GroupID: req.GroupID}) | ||||
| 				if err != nil { | ||||
| 				log.ZWarn(ctx, "GetGroupMemberIDs", err) | ||||
| 					log.ZWarn(ctx, "GetGroupMemberIDs is failed.", err) | ||||
| 					return | ||||
| 				} | ||||
| 
 | ||||
| 				conversation.GroupAtType = &wrapperspb.Int32Value{Value: constant.GroupNotification} | ||||
| 
 | ||||
| 				if err := g.conversationRpcClient.SetConversations(ctx, resp.UserIDs, conversation); err != nil { | ||||
| 				log.ZWarn(ctx, "SetConversations", err, resp.UserIDs, conversation) | ||||
| 					log.ZWarn(ctx, "SetConversations", err, "UserIDs", resp.UserIDs, "conversation", conversation) | ||||
| 				} | ||||
| 			}() | ||||
| 
 | ||||
| 			g.notification.GroupInfoSetAnnouncementNotification(ctx, &sdkws.GroupInfoSetAnnouncementTips{Group: tips.Group, OpUser: tips.OpUser}) | ||||
| 		} | ||||
| 	if req.GroupInfoForSetEX.GroupName != "" { | ||||
| 		num-- | ||||
| 	} | ||||
| 
 | ||||
| 	if req.GroupName != nil { | ||||
| 		num-- | ||||
| 		g.notification.GroupInfoSetNameNotification(ctx, &sdkws.GroupInfoSetNameTips{Group: tips.Group, OpUser: tips.OpUser}) | ||||
| 	} | ||||
| 
 | ||||
| 	if num > 0 { | ||||
| 		g.notification.GroupInfoSetNotification(ctx, tips) | ||||
| 	} | ||||
| 
 | ||||
| 	g.webhookAfterSetGroupInfoEX(ctx, &g.config.WebhooksConfig.AfterSetGroupInfoEX, req) | ||||
| 	g.webhookAfterSetGroupInfoEx(ctx, &g.config.WebhooksConfig.AfterSetGroupInfoEx, req) | ||||
| 
 | ||||
| 	return &pbgroup.SetGroupInfoEXResp{}, nil | ||||
| 	return &pbgroup.SetGroupInfoExResp{}, nil | ||||
| } | ||||
| 
 | ||||
| func (g *groupServer) TransferGroupOwner(ctx context.Context, req *pbgroup.TransferGroupOwnerReq) (*pbgroup.TransferGroupOwnerResp, error) { | ||||
| @ -1471,9 +1485,6 @@ func (g *groupServer) SetGroupMemberInfo(ctx context.Context, req *pbgroup.SetGr | ||||
| 		return nil, errs.ErrNoPermission.WrapMsg("no op user id") | ||||
| 	} | ||||
| 	isAppManagerUid := authverify.IsAppManagerUid(ctx, g.config.Share.IMAdminUserID) | ||||
| 	for i := range req.Members { | ||||
| 		req.Members[i].FaceURL = nil | ||||
| 	} | ||||
| 	groupMembers := make(map[string][]*pbgroup.SetGroupMemberInfo) | ||||
| 	for i, member := range req.Members { | ||||
| 		if member.RoleLevel != nil { | ||||
| @ -1515,10 +1526,23 @@ func (g *groupServer) SetGroupMemberInfo(ctx context.Context, req *pbgroup.SetGr | ||||
| 		case 0: | ||||
| 			if !isAppManagerUid { | ||||
| 				roleLevel := dbMembers[opUserIndex].RoleLevel | ||||
| 				if roleLevel != constant.GroupOwner { | ||||
| 				var ( | ||||
| 					dbSelf  = &model.GroupMember{} | ||||
| 					reqSelf *pbgroup.SetGroupMemberInfo | ||||
| 				) | ||||
| 				switch roleLevel { | ||||
| 				case constant.GroupOwner: | ||||
| 					for _, member := range dbMembers { | ||||
| 						if member.UserID == opUserID { | ||||
| 							dbSelf = member | ||||
| 							break | ||||
| 						} | ||||
| 					} | ||||
| 				case constant.GroupAdmin: | ||||
| 					for _, member := range dbMembers { | ||||
| 						if member.UserID == opUserID { | ||||
| 							dbSelf = member | ||||
| 						} | ||||
| 						if member.RoleLevel == constant.GroupOwner { | ||||
| 							return nil, errs.ErrNoPermission.WrapMsg("admin can not change group owner") | ||||
| 						} | ||||
| @ -1528,17 +1552,36 @@ func (g *groupServer) SetGroupMemberInfo(ctx context.Context, req *pbgroup.SetGr | ||||
| 					} | ||||
| 				case constant.GroupOrdinaryUsers: | ||||
| 					for _, member := range dbMembers { | ||||
| 						if member.UserID == opUserID { | ||||
| 							dbSelf = member | ||||
| 						} | ||||
| 						if !(member.RoleLevel == constant.GroupOrdinaryUsers && member.UserID == opUserID) { | ||||
| 							return nil, errs.ErrNoPermission.WrapMsg("ordinary users can not change other role level") | ||||
| 						} | ||||
| 					} | ||||
| 				default: | ||||
| 					for _, member := range dbMembers { | ||||
| 						if member.UserID == opUserID { | ||||
| 							dbSelf = member | ||||
| 						} | ||||
| 						if member.RoleLevel >= roleLevel { | ||||
| 							return nil, errs.ErrNoPermission.WrapMsg("can not change higher role level") | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 				for _, member := range req.Members { | ||||
| 					if member.UserID == opUserID { | ||||
| 						reqSelf = member | ||||
| 						break | ||||
| 					} | ||||
| 				} | ||||
| 				if reqSelf != nil && reqSelf.RoleLevel != nil { | ||||
| 					if reqSelf.RoleLevel.GetValue() > dbSelf.RoleLevel { | ||||
| 						return nil, errs.ErrNoPermission.WrapMsg("can not improve role level by self") | ||||
| 					} | ||||
| 					if roleLevel == constant.GroupOwner { | ||||
| 						return nil, errs.ErrArgs.WrapMsg("group owner can not change own role level") // Prevent the absence of a group owner | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		case 1: | ||||
| @ -1578,7 +1621,7 @@ func (g *groupServer) SetGroupMemberInfo(ctx context.Context, req *pbgroup.SetGr | ||||
| 				g.notification.GroupMemberSetToOrdinaryUserNotification(ctx, member.GroupID, member.UserID) | ||||
| 			} | ||||
| 		} | ||||
| 		if member.Nickname != nil || member.FaceURL != nil || member.Ex != nil { | ||||
| 		if member.Nickname != nil || member.FaceURL != nil || member.Ex != nil || member.RoleLevel != nil { | ||||
| 			g.notification.GroupMemberInfoSetNotification(ctx, member.GroupID, member.UserID) | ||||
| 		} | ||||
| 	} | ||||
| @ -1671,36 +1714,51 @@ func (g *groupServer) GetGroupUsersReqApplicationList(ctx context.Context, req * | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	if len(requests) == 0 { | ||||
| 		return &pbgroup.GetGroupUsersReqApplicationListResp{}, nil | ||||
| 	} | ||||
| 
 | ||||
| 	groupIDs := datautil.Distinct(datautil.Slice(requests, func(e *model.GroupRequest) string { | ||||
| 		return e.GroupID | ||||
| 	})) | ||||
| 
 | ||||
| 	groups, err := g.db.FindGroup(ctx, groupIDs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	groupMap := datautil.SliceToMap(groups, func(e *model.Group) string { | ||||
| 		return e.GroupID | ||||
| 	}) | ||||
| 
 | ||||
| 	if ids := datautil.Single(groupIDs, datautil.Keys(groupMap)); len(ids) > 0 { | ||||
| 		return nil, servererrs.ErrGroupIDNotFound.WrapMsg(strings.Join(ids, ",")) | ||||
| 	} | ||||
| 
 | ||||
| 	userMap, err := g.user.GetPublicUserInfoMap(ctx, req.UserIDs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	owners, err := g.db.FindGroupsOwner(ctx, groupIDs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	if err := g.PopulateGroupMember(ctx, owners...); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	ownerMap := datautil.SliceToMap(owners, func(e *model.GroupMember) string { | ||||
| 		return e.GroupID | ||||
| 	}) | ||||
| 
 | ||||
| 	groupMemberNum, err := g.db.MapGroupMemberNum(ctx, groupIDs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	return &pbgroup.GetGroupUsersReqApplicationListResp{ | ||||
| 		Total: int64(len(requests)), | ||||
| 		GroupRequests: datautil.Slice(requests, func(e *model.GroupRequest) *sdkws.GroupRequest { | ||||
| @ -1708,7 +1766,72 @@ func (g *groupServer) GetGroupUsersReqApplicationList(ctx context.Context, req * | ||||
| 			if owner, ok := ownerMap[e.GroupID]; ok { | ||||
| 				ownerUserID = owner.UserID | ||||
| 			} | ||||
| 			return convert.Db2PbGroupRequest(e, nil, convert.Db2PbGroupInfo(groupMap[e.GroupID], ownerUserID, groupMemberNum[e.GroupID])) | ||||
| 
 | ||||
| 			var userInfo *sdkws.PublicUserInfo | ||||
| 			if user, ok := userMap[e.UserID]; !ok { | ||||
| 				userInfo = user | ||||
| 			} | ||||
| 
 | ||||
| 			return convert.Db2PbGroupRequest(e, userInfo, convert.Db2PbGroupInfo(groupMap[e.GroupID], ownerUserID, groupMemberNum[e.GroupID])) | ||||
| 		}), | ||||
| 	}, nil | ||||
| } | ||||
| 
 | ||||
| func (g *groupServer) GetSpecifiedUserGroupRequestInfo(ctx context.Context, req *pbgroup.GetSpecifiedUserGroupRequestInfoReq) (*pbgroup.GetSpecifiedUserGroupRequestInfoResp, error) { | ||||
| 	opUserID := mcontext.GetOpUserID(ctx) | ||||
| 
 | ||||
| 	owners, err := g.db.FindGroupsOwner(ctx, []string{req.GroupID}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	if req.UserID != opUserID { | ||||
| 		req.UserID = mcontext.GetOpUserID(ctx) | ||||
| 		adminIDs, err := g.db.GetGroupRoleLevelMemberIDs(ctx, req.GroupID, constant.GroupAdmin) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 
 | ||||
| 		adminIDs = append(adminIDs, owners[0].UserID) | ||||
| 		adminIDs = append(adminIDs, g.config.Share.IMAdminUserID...) | ||||
| 
 | ||||
| 		if !datautil.Contain(req.UserID, adminIDs...) { | ||||
| 			return nil, errs.ErrNoPermission.WrapMsg("opUser no permission") | ||||
| 		} | ||||
| 	} | ||||
| 	requests, err := g.db.FindGroupRequests(ctx, req.GroupID, []string{req.UserID}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	if len(requests) == 0 { | ||||
| 		return &pbgroup.GetSpecifiedUserGroupRequestInfoResp{}, nil | ||||
| 	} | ||||
| 
 | ||||
| 	groups, err := g.db.FindGroup(ctx, []string{req.GroupID}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	userInfos, err := g.user.GetPublicUserInfos(ctx, []string{req.UserID}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	groupMemberNum, err := g.db.MapGroupMemberNum(ctx, []string{req.GroupID}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	resp := &pbgroup.GetSpecifiedUserGroupRequestInfoResp{ | ||||
| 		GroupRequests: make([]*sdkws.GroupRequest, 0, len(requests)), | ||||
| 	} | ||||
| 
 | ||||
| 	for _, request := range requests { | ||||
| 		resp.GroupRequests = append(resp.GroupRequests, convert.Db2PbGroupRequest(request, userInfos[0], convert.Db2PbGroupInfo(groups[0], owners[0].UserID, groupMemberNum[groups[0].GroupID]))) | ||||
| 	} | ||||
| 
 | ||||
| 	resp.Total = uint32(len(requests)) | ||||
| 
 | ||||
| 	return resp, nil | ||||
| } | ||||
|  | ||||
| @ -16,27 +16,29 @@ package group | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/convert" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/servererrs" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/controller" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/database" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/versionctx" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/msgprocessor" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcclient/notification" | ||||
| 	"github.com/openimsdk/protocol/msg" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/servererrs" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/controller" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcclient" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcclient/notification" | ||||
| 	"github.com/openimsdk/protocol/constant" | ||||
| 	pbgroup "github.com/openimsdk/protocol/group" | ||||
| 	"github.com/openimsdk/protocol/msg" | ||||
| 	"github.com/openimsdk/protocol/sdkws" | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| 	"github.com/openimsdk/tools/log" | ||||
| 	"github.com/openimsdk/tools/mcontext" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| 	"github.com/openimsdk/tools/utils/stringutil" | ||||
| 	"go.mongodb.org/mongo-driver/mongo" | ||||
| 	"time" | ||||
| ) | ||||
| 
 | ||||
| // GroupApplicationReceiver | ||||
| @ -227,10 +229,13 @@ func (g *GroupNotificationSender) groupMemberDB2PB(member *model.GroupMember, ap | ||||
| } */ | ||||
| 
 | ||||
| func (g *GroupNotificationSender) fillOpUser(ctx context.Context, opUser **sdkws.GroupMemberFullInfo, groupID string) (err error) { | ||||
| 	return g.fillOpUserByUserID(ctx, mcontext.GetOpUserID(ctx), opUser, groupID) | ||||
| } | ||||
| 
 | ||||
| func (g *GroupNotificationSender) fillOpUserByUserID(ctx context.Context, userID string, opUser **sdkws.GroupMemberFullInfo, groupID string) error { | ||||
| 	if opUser == nil { | ||||
| 		return errs.ErrInternalServer.WrapMsg("**sdkws.GroupMemberFullInfo is nil") | ||||
| 	} | ||||
| 	userID := mcontext.GetOpUserID(ctx) | ||||
| 	if groupID != "" { | ||||
| 		if authverify.IsManagerUserID(userID, g.config.Share.IMAdminUserID) { | ||||
| 			*opUser = &sdkws.GroupMemberFullInfo{ | ||||
| @ -243,7 +248,7 @@ func (g *GroupNotificationSender) fillOpUser(ctx context.Context, opUser **sdkws | ||||
| 			member, err := g.db.TakeGroupMember(ctx, groupID, userID) | ||||
| 			if err == nil { | ||||
| 				*opUser = g.groupMemberDB2PB(member, 0) | ||||
| 			} else if !errs.ErrRecordNotFound.Is(err) { | ||||
| 			} else if !(errors.Is(err, mongo.ErrNoDocuments) || errs.ErrRecordNotFound.Is(err)) { | ||||
| 				return err | ||||
| 			} | ||||
| 		} | ||||
| @ -509,7 +514,7 @@ func (g *GroupNotificationSender) MemberKickedNotification(ctx context.Context, | ||||
| 	g.Notification(ctx, mcontext.GetOpUserID(ctx), tips.Group.GroupID, constant.MemberKickedNotification, tips) | ||||
| } | ||||
| 
 | ||||
| func (g *GroupNotificationSender) MemberEnterNotification(ctx context.Context, groupID string, entrantUserID ...string) error { | ||||
| func (g *GroupNotificationSender) GroupApplicationAgreeMemberEnterNotification(ctx context.Context, groupID string, invitedOpUserID string, entrantUserID ...string) error { | ||||
| 	var err error | ||||
| 	defer func() { | ||||
| 		if err != nil { | ||||
| @ -546,15 +551,75 @@ func (g *GroupNotificationSender) MemberEnterNotification(ctx context.Context, g | ||||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
| 	tips := &sdkws.MemberInvitedTips{Group: group, InvitedUserList: users} | ||||
| 	if err = g.fillOpUser(ctx, &tips.OpUser, tips.Group.GroupID); err != nil { | ||||
| 	tips := &sdkws.MemberInvitedTips{ | ||||
| 		Group:           group, | ||||
| 		InvitedUserList: users, | ||||
| 	} | ||||
| 	opUserID := mcontext.GetOpUserID(ctx) | ||||
| 	if err = g.fillOpUserByUserID(ctx, opUserID, &tips.OpUser, tips.Group.GroupID); err != nil { | ||||
| 		return nil | ||||
| 	} | ||||
| 	switch { | ||||
| 	case invitedOpUserID == "": | ||||
| 	case invitedOpUserID == opUserID: | ||||
| 		tips.InviterUser = tips.OpUser | ||||
| 	default: | ||||
| 		if err = g.fillOpUserByUserID(ctx, invitedOpUserID, &tips.InviterUser, tips.Group.GroupID); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	g.setVersion(ctx, &tips.GroupMemberVersion, &tips.GroupMemberVersionID, database.GroupMemberVersionName, tips.Group.GroupID) | ||||
| 	g.Notification(ctx, mcontext.GetOpUserID(ctx), group.GroupID, constant.MemberInvitedNotification, tips) | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func (g *GroupNotificationSender) MemberEnterNotification(ctx context.Context, groupID string, entrantUserID string) error { | ||||
| 	var err error | ||||
| 	defer func() { | ||||
| 		if err != nil { | ||||
| 			log.ZError(ctx, stringutil.GetFuncName(1)+" failed", err) | ||||
| 		} | ||||
| 	}() | ||||
| 
 | ||||
| 	if !g.config.RpcConfig.EnableHistoryForNewMembers { | ||||
| 		conversationID := msgprocessor.GetConversationIDBySessionType(constant.ReadGroupChatType, groupID) | ||||
| 		maxSeq, err := g.msgRpcClient.GetConversationMaxSeq(ctx, conversationID) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		if _, err = g.msgRpcClient.SetUserConversationsMinSeq(ctx, &msg.SetUserConversationsMinSeqReq{ | ||||
| 			UserIDs:        []string{entrantUserID}, | ||||
| 			ConversationID: conversationID, | ||||
| 			Seq:            maxSeq, | ||||
| 		}); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	if err := g.conversationRpcClient.GroupChatFirstCreateConversation(ctx, groupID, []string{entrantUserID}); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
| 	var group *sdkws.GroupInfo | ||||
| 	group, err = g.getGroupInfo(ctx, groupID) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	user, err := g.getGroupMember(ctx, groupID, entrantUserID) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
| 	tips := &sdkws.MemberEnterTips{ | ||||
| 		Group:         group, | ||||
| 		EntrantUser:   user, | ||||
| 		OperationTime: time.Now().UnixMilli(), | ||||
| 	} | ||||
| 	g.setVersion(ctx, &tips.GroupMemberVersion, &tips.GroupMemberVersionID, database.GroupMemberVersionName, tips.Group.GroupID) | ||||
| 	g.Notification(ctx, mcontext.GetOpUserID(ctx), group.GroupID, constant.MemberEnterNotification, tips) | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func (g *GroupNotificationSender) GroupDismissedNotification(ctx context.Context, tips *sdkws.GroupDismissedTips) { | ||||
| 	var err error | ||||
| 	defer func() { | ||||
|  | ||||
| @ -55,7 +55,7 @@ func (m *msgServer) GetConversationsHasReadAndMaxSeq(ctx context.Context, req *m | ||||
| 			conversationMaxSeqMap[conversation.ConversationID] = conversation.MaxSeq | ||||
| 		} | ||||
| 	} | ||||
| 	maxSeqs, err := m.MsgDatabase.GetMaxSeqs(ctx, conversationIDs) | ||||
| 	maxSeqs, err := m.MsgDatabase.GetMaxSeqsWithTime(ctx, conversationIDs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| @ -63,7 +63,8 @@ func (m *msgServer) GetConversationsHasReadAndMaxSeq(ctx context.Context, req *m | ||||
| 	for conversationID, maxSeq := range maxSeqs { | ||||
| 		resp.Seqs[conversationID] = &msg.Seqs{ | ||||
| 			HasReadSeq: hasReadSeqs[conversationID], | ||||
| 			MaxSeq:     maxSeq, | ||||
| 			MaxSeq:     maxSeq.Seq, | ||||
| 			MaxSeqTime: maxSeq.Time, | ||||
| 		} | ||||
| 		if v, ok := conversationMaxSeqMap[conversationID]; ok { | ||||
| 			resp.Seqs[conversationID].MaxSeq = v | ||||
|  | ||||
| @ -67,6 +67,9 @@ func (m *msgServer) webhookBeforeSendSingleMsg(ctx context.Context, before *conf | ||||
| 		if msg.MsgData.ContentType == constant.Typing { | ||||
| 			return nil | ||||
| 		} | ||||
| 		if !filterBeforeMsg(msg, before) { | ||||
| 			return nil | ||||
| 		} | ||||
| 		cbReq := &cbapi.CallbackBeforeSendSingleMsgReq{ | ||||
| 			CommonCallbackReq: toCommonCallback(ctx, msg, cbapi.CallbackBeforeSendSingleMsgCommand), | ||||
| 			RecvID:            msg.MsgData.RecvID, | ||||
| @ -84,9 +87,7 @@ func (m *msgServer) webhookAfterSendSingleMsg(ctx context.Context, after *config | ||||
| 	if msg.MsgData.ContentType == constant.Typing { | ||||
| 		return | ||||
| 	} | ||||
| 	// According to the attentionIds configuration, only some users are sent | ||||
| 	attentionIds := after.AttentionIds | ||||
| 	if attentionIds != nil && !datautil.Contain(msg.MsgData.RecvID, attentionIds...) && !datautil.Contain(msg.MsgData.SendID, attentionIds...) { | ||||
| 	if !filterAfterMsg(msg, after) { | ||||
| 		return | ||||
| 	} | ||||
| 	cbReq := &cbapi.CallbackAfterSendSingleMsgReq{ | ||||
| @ -98,6 +99,9 @@ func (m *msgServer) webhookAfterSendSingleMsg(ctx context.Context, after *config | ||||
| 
 | ||||
| func (m *msgServer) webhookBeforeSendGroupMsg(ctx context.Context, before *config.BeforeConfig, msg *pbchat.SendMsgReq) error { | ||||
| 	return webhook.WithCondition(ctx, before, func(ctx context.Context) error { | ||||
| 		if !filterBeforeMsg(msg, before) { | ||||
| 			return nil | ||||
| 		} | ||||
| 		if msg.MsgData.ContentType == constant.Typing { | ||||
| 			return nil | ||||
| 		} | ||||
| @ -117,6 +121,9 @@ func (m *msgServer) webhookAfterSendGroupMsg(ctx context.Context, after *config. | ||||
| 	if msg.MsgData.ContentType == constant.Typing { | ||||
| 		return | ||||
| 	} | ||||
| 	if !filterAfterMsg(msg, after) { | ||||
| 		return | ||||
| 	} | ||||
| 	cbReq := &cbapi.CallbackAfterSendGroupMsgReq{ | ||||
| 		CommonCallbackReq: toCommonCallback(ctx, msg, cbapi.CallbackAfterSendGroupMsgCommand), | ||||
| 		GroupID:           msg.MsgData.GroupID, | ||||
| @ -129,6 +136,9 @@ func (m *msgServer) webhookBeforeMsgModify(ctx context.Context, before *config.B | ||||
| 		if msg.MsgData.ContentType != constant.Text { | ||||
| 			return nil | ||||
| 		} | ||||
| 		if !filterBeforeMsg(msg, before) { | ||||
| 			return nil | ||||
| 		} | ||||
| 		cbReq := &cbapi.CallbackMsgModifyCommandReq{ | ||||
| 			CommonCallbackReq: toCommonCallback(ctx, msg, cbapi.CallbackBeforeMsgModifyCommand), | ||||
| 		} | ||||
|  | ||||
| @ -67,7 +67,7 @@ func (m *msgServer) ClearMsg(ctx context.Context, req *msg.ClearMsgReq) (_ *msg. | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	log.ZInfo(ctx, "clearing message", "docNum", docNum, "msgNum", msgNum, "cost", time.Since(start)) | ||||
| 	log.ZDebug(ctx, "clearing message", "docNum", docNum, "msgNum", msgNum, "cost", time.Since(start)) | ||||
| 
 | ||||
| 	return &msg.ClearMsgResp{}, nil | ||||
| } | ||||
|  | ||||
							
								
								
									
										67
									
								
								internal/rpc/msg/filter.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								internal/rpc/msg/filter.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,67 @@ | ||||
| package msg | ||||
| 
 | ||||
| import ( | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||
| 	pbchat "github.com/openimsdk/protocol/msg" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| ) | ||||
| 
 | ||||
| const ( | ||||
| 	separator = "-" | ||||
| ) | ||||
| 
 | ||||
| func filterAfterMsg(msg *pbchat.SendMsgReq, after *config.AfterConfig) bool { | ||||
| 	return filterMsg(msg, after.AttentionIds, after.AllowedTypes, after.DeniedTypes) | ||||
| } | ||||
| 
 | ||||
| func filterBeforeMsg(msg *pbchat.SendMsgReq, before *config.BeforeConfig) bool { | ||||
| 	return filterMsg(msg, nil, before.AllowedTypes, before.DeniedTypes) | ||||
| } | ||||
| 
 | ||||
| func filterMsg(msg *pbchat.SendMsgReq, attentionIds, allowedTypes, deniedTypes []string) bool { | ||||
| 	// According to the attentionIds configuration, only some users are sent | ||||
| 	if len(attentionIds) != 0 && !datautil.Contains([]string{msg.MsgData.SendID, msg.MsgData.RecvID}, attentionIds...) { | ||||
| 		return false | ||||
| 	} | ||||
| 	if len(allowedTypes) != 0 && !isInInterval(msg.MsgData.ContentType, allowedTypes) { | ||||
| 		return false | ||||
| 	} | ||||
| 	if len(deniedTypes) != 0 && isInInterval(msg.MsgData.ContentType, deniedTypes) { | ||||
| 		return false | ||||
| 	} | ||||
| 	return true | ||||
| } | ||||
| 
 | ||||
| func isInInterval(contentType int32, interval []string) bool { | ||||
| 	for _, v := range interval { | ||||
| 		if strings.Contains(v, separator) { | ||||
| 			// is interval | ||||
| 			bounds := strings.Split(v, separator) | ||||
| 			if len(bounds) != 2 { | ||||
| 				continue | ||||
| 			} | ||||
| 			bottom, err := strconv.Atoi(bounds[0]) | ||||
| 			if err != nil { | ||||
| 				continue | ||||
| 			} | ||||
| 			top, err := strconv.Atoi(bounds[1]) | ||||
| 			if err != nil { | ||||
| 				continue | ||||
| 			} | ||||
| 			if datautil.BetweenEq(int(contentType), bottom, top) { | ||||
| 				return true | ||||
| 			} | ||||
| 		} else { | ||||
| 			iv, err := strconv.Atoi(v) | ||||
| 			if err != nil { | ||||
| 				continue | ||||
| 			} | ||||
| 			if int(contentType) == iv { | ||||
| 				return true | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return false | ||||
| } | ||||
| @ -16,10 +16,10 @@ package msg | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	pbmsg "github.com/openimsdk/protocol/msg" | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 
 | ||||
| 	pbmsg "github.com/openimsdk/protocol/msg" | ||||
| 	"sort" | ||||
| ) | ||||
| 
 | ||||
| func (m *msgServer) GetConversationMaxSeq(ctx context.Context, req *pbmsg.GetConversationMaxSeqReq) (*pbmsg.GetConversationMaxSeqResp, error) { | ||||
| @ -62,3 +62,25 @@ func (m *msgServer) SetUserConversationsMinSeq(ctx context.Context, req *pbmsg.S | ||||
| 	} | ||||
| 	return &pbmsg.SetUserConversationsMinSeqResp{}, nil | ||||
| } | ||||
| 
 | ||||
| func (m *msgServer) GetActiveConversation(ctx context.Context, req *pbmsg.GetActiveConversationReq) (*pbmsg.GetActiveConversationResp, error) { | ||||
| 	res, err := m.MsgDatabase.GetCacheMaxSeqWithTime(ctx, req.ConversationIDs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	conversations := make([]*pbmsg.ActiveConversation, 0, len(res)) | ||||
| 	for conversationID, val := range res { | ||||
| 		conversations = append(conversations, &pbmsg.ActiveConversation{ | ||||
| 			MaxSeq:         val.Seq, | ||||
| 			LastTime:       val.Time, | ||||
| 			ConversationID: conversationID, | ||||
| 		}) | ||||
| 	} | ||||
| 	if req.Limit > 0 { | ||||
| 		sort.Sort(activeConversations(conversations)) | ||||
| 		if len(conversations) > int(req.Limit) { | ||||
| 			conversations = conversations[:req.Limit] | ||||
| 		} | ||||
| 	} | ||||
| 	return &pbmsg.GetActiveConversationResp{Conversations: conversations}, nil | ||||
| } | ||||
|  | ||||
| @ -16,16 +16,16 @@ package msg | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/util/conversationutil" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| 	"github.com/openimsdk/tools/utils/timeutil" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/msgprocessor" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/util/conversationutil" | ||||
| 	"github.com/openimsdk/protocol/constant" | ||||
| 	"github.com/openimsdk/protocol/msg" | ||||
| 	"github.com/openimsdk/protocol/sdkws" | ||||
| 	"github.com/openimsdk/tools/log" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| 	"github.com/openimsdk/tools/utils/timeutil" | ||||
| ) | ||||
| 
 | ||||
| func (m *msgServer) PullMessageBySeqs(ctx context.Context, req *sdkws.PullMessageBySeqsReq) (*sdkws.PullMessageBySeqsResp, error) { | ||||
| @ -86,6 +86,35 @@ func (m *msgServer) PullMessageBySeqs(ctx context.Context, req *sdkws.PullMessag | ||||
| 	return resp, nil | ||||
| } | ||||
| 
 | ||||
| func (m *msgServer) GetSeqMessage(ctx context.Context, req *msg.GetSeqMessageReq) (*msg.GetSeqMessageResp, error) { | ||||
| 	resp := &msg.GetSeqMessageResp{ | ||||
| 		Msgs:             make(map[string]*sdkws.PullMsgs), | ||||
| 		NotificationMsgs: make(map[string]*sdkws.PullMsgs), | ||||
| 	} | ||||
| 	for _, conv := range req.Conversations { | ||||
| 		_, _, msgs, err := m.MsgDatabase.GetMsgBySeqs(ctx, req.UserID, conv.ConversationID, conv.Seqs) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		var pullMsgs *sdkws.PullMsgs | ||||
| 		if ok := false; conversationutil.IsNotificationConversationID(conv.ConversationID) { | ||||
| 			pullMsgs, ok = resp.NotificationMsgs[conv.ConversationID] | ||||
| 			if !ok { | ||||
| 				pullMsgs = &sdkws.PullMsgs{} | ||||
| 				resp.NotificationMsgs[conv.ConversationID] = pullMsgs | ||||
| 			} | ||||
| 		} else { | ||||
| 			pullMsgs, ok = resp.Msgs[conv.ConversationID] | ||||
| 			if !ok { | ||||
| 				pullMsgs = &sdkws.PullMsgs{} | ||||
| 				resp.Msgs[conv.ConversationID] = pullMsgs | ||||
| 			} | ||||
| 		} | ||||
| 		pullMsgs.Msgs = append(pullMsgs.Msgs, msgs...) | ||||
| 	} | ||||
| 	return resp, nil | ||||
| } | ||||
| 
 | ||||
| func (m *msgServer) GetMaxSeq(ctx context.Context, req *sdkws.GetMaxSeqReq) (*sdkws.GetMaxSeqResp, error) { | ||||
| 	if err := authverify.CheckAccessV3(ctx, req.UserID, m.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| @ -104,13 +133,20 @@ func (m *msgServer) GetMaxSeq(ctx context.Context, req *sdkws.GetMaxSeqReq) (*sd | ||||
| 		log.ZWarn(ctx, "GetMaxSeqs error", err, "conversationIDs", conversationIDs, "maxSeqs", maxSeqs) | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	// avoid pulling messages from sessions with a large number of max seq values of 0 | ||||
| 	for conversationID, seq := range maxSeqs { | ||||
| 		if seq == 0 { | ||||
| 			delete(maxSeqs, conversationID) | ||||
| 		} | ||||
| 	} | ||||
| 	resp := new(sdkws.GetMaxSeqResp) | ||||
| 	resp.MaxSeqs = maxSeqs | ||||
| 	return resp, nil | ||||
| } | ||||
| 
 | ||||
| func (m *msgServer) SearchMessage(ctx context.Context, req *msg.SearchMessageReq) (resp *msg.SearchMessageResp, err error) { | ||||
| 	var chatLogs []*sdkws.MsgData | ||||
| 	// var chatLogs []*sdkws.MsgData | ||||
| 	var chatLogs []*msg.SearchedMsgData | ||||
| 	var total int64 | ||||
| 	resp = &msg.SearchMessageResp{} | ||||
| 	if total, chatLogs, err = m.MsgDatabase.SearchMessage(ctx, req); err != nil { | ||||
| @ -125,17 +161,19 @@ func (m *msgServer) SearchMessage(ctx context.Context, req *msg.SearchMessageReq | ||||
| 		recvMap  = make(map[string]string) | ||||
| 		groupMap = make(map[string]*sdkws.GroupInfo) | ||||
| 	) | ||||
| 
 | ||||
| 	for _, chatLog := range chatLogs { | ||||
| 		if chatLog.SenderNickname == "" { | ||||
| 			sendIDs = append(sendIDs, chatLog.SendID) | ||||
| 		if chatLog.MsgData.SenderNickname == "" { | ||||
| 			sendIDs = append(sendIDs, chatLog.MsgData.SendID) | ||||
| 		} | ||||
| 		switch chatLog.SessionType { | ||||
| 		switch chatLog.MsgData.SessionType { | ||||
| 		case constant.SingleChatType, constant.NotificationChatType: | ||||
| 			recvIDs = append(recvIDs, chatLog.RecvID) | ||||
| 			recvIDs = append(recvIDs, chatLog.MsgData.RecvID) | ||||
| 		case constant.WriteGroupChatType, constant.ReadGroupChatType: | ||||
| 			groupIDs = append(groupIDs, chatLog.GroupID) | ||||
| 			groupIDs = append(groupIDs, chatLog.MsgData.GroupID) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	// Retrieve sender and receiver information | ||||
| 	if len(sendIDs) != 0 { | ||||
| 		sendInfos, err := m.UserLocalCache.GetUsersInfo(ctx, sendIDs) | ||||
| @ -146,6 +184,7 @@ func (m *msgServer) SearchMessage(ctx context.Context, req *msg.SearchMessageReq | ||||
| 			sendMap[sendInfo.UserID] = sendInfo.Nickname | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	if len(recvIDs) != 0 { | ||||
| 		recvInfos, err := m.UserLocalCache.GetUsersInfo(ctx, recvIDs) | ||||
| 		if err != nil { | ||||
| @ -171,20 +210,21 @@ func (m *msgServer) SearchMessage(ctx context.Context, req *msg.SearchMessageReq | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	// Construct response with updated information | ||||
| 	for _, chatLog := range chatLogs { | ||||
| 		pbchatLog := &msg.ChatLog{} | ||||
| 		datautil.CopyStructFields(pbchatLog, chatLog) | ||||
| 		pbchatLog.SendTime = chatLog.SendTime | ||||
| 		pbchatLog.CreateTime = chatLog.CreateTime | ||||
| 		if chatLog.SenderNickname == "" { | ||||
| 			pbchatLog.SenderNickname = sendMap[chatLog.SendID] | ||||
| 		datautil.CopyStructFields(pbchatLog, chatLog.MsgData) | ||||
| 		pbchatLog.SendTime = chatLog.MsgData.SendTime | ||||
| 		pbchatLog.CreateTime = chatLog.MsgData.CreateTime | ||||
| 		if chatLog.MsgData.SenderNickname == "" { | ||||
| 			pbchatLog.SenderNickname = sendMap[chatLog.MsgData.SendID] | ||||
| 		} | ||||
| 		switch chatLog.SessionType { | ||||
| 		switch chatLog.MsgData.SessionType { | ||||
| 		case constant.SingleChatType, constant.NotificationChatType: | ||||
| 			pbchatLog.RecvNickname = recvMap[chatLog.RecvID] | ||||
| 		case constant.WriteGroupChatType, constant.ReadGroupChatType: | ||||
| 			groupInfo := groupMap[chatLog.GroupID] | ||||
| 			pbchatLog.RecvNickname = recvMap[chatLog.MsgData.RecvID] | ||||
| 		case constant.ReadGroupChatType: | ||||
| 			groupInfo := groupMap[chatLog.MsgData.GroupID] | ||||
| 			pbchatLog.SenderFaceURL = groupInfo.FaceURL | ||||
| 			pbchatLog.GroupMemberCount = groupInfo.MemberCount // Reflects actual member count | ||||
| 			pbchatLog.RecvID = groupInfo.GroupID | ||||
| @ -192,7 +232,9 @@ func (m *msgServer) SearchMessage(ctx context.Context, req *msg.SearchMessageReq | ||||
| 			pbchatLog.GroupOwner = groupInfo.OwnerUserID | ||||
| 			pbchatLog.GroupType = groupInfo.GroupType | ||||
| 		} | ||||
| 		resp.ChatLogs = append(resp.ChatLogs, pbchatLog) | ||||
| 		searchChatLog := &msg.SearchChatLog{ChatLog: pbchatLog, IsRevoked: chatLog.IsRevoked} | ||||
| 
 | ||||
| 		resp.ChatLogs = append(resp.ChatLogs, searchChatLog) | ||||
| 	} | ||||
| 	resp.ChatLogsNum = int32(total) | ||||
| 	return resp, nil | ||||
|  | ||||
| @ -15,6 +15,7 @@ | ||||
| package msg | ||||
| 
 | ||||
| import ( | ||||
| 	"github.com/openimsdk/protocol/msg" | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"go.mongodb.org/mongo-driver/mongo" | ||||
| @ -28,3 +29,63 @@ func IsNotFound(err error) bool { | ||||
| 		return false | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| type activeConversations []*msg.ActiveConversation | ||||
| 
 | ||||
| func (s activeConversations) Len() int { | ||||
| 	return len(s) | ||||
| } | ||||
| 
 | ||||
| func (s activeConversations) Less(i, j int) bool { | ||||
| 	return s[i].LastTime > s[j].LastTime | ||||
| } | ||||
| 
 | ||||
| func (s activeConversations) Swap(i, j int) { | ||||
| 	s[i], s[j] = s[j], s[i] | ||||
| } | ||||
| 
 | ||||
| //type seqTime struct { | ||||
| //	ConversationID string | ||||
| //	Seq            int64 | ||||
| //	Time           int64 | ||||
| //	Unread         int64 | ||||
| //	Pinned         bool | ||||
| //} | ||||
| // | ||||
| //func (s seqTime) String() string { | ||||
| //	return fmt.Sprintf("<Time_%d,Unread_%d,Pinned_%t>", s.Time, s.Unread, s.Pinned) | ||||
| //} | ||||
| // | ||||
| //type seqTimes []seqTime | ||||
| // | ||||
| //func (s seqTimes) Len() int { | ||||
| //	return len(s) | ||||
| //} | ||||
| // | ||||
| //// Less sticky priority, unread priority, time descending | ||||
| //func (s seqTimes) Less(i, j int) bool { | ||||
| //	iv, jv := s[i], s[j] | ||||
| //	if iv.Pinned && (!jv.Pinned) { | ||||
| //		return true | ||||
| //	} | ||||
| //	if jv.Pinned && (!iv.Pinned) { | ||||
| //		return false | ||||
| //	} | ||||
| //	if iv.Unread > 0 && jv.Unread == 0 { | ||||
| //		return true | ||||
| //	} | ||||
| //	if jv.Unread > 0 && iv.Unread == 0 { | ||||
| //		return false | ||||
| //	} | ||||
| //	return iv.Time > jv.Time | ||||
| //} | ||||
| // | ||||
| //func (s seqTimes) Swap(i, j int) { | ||||
| //	s[i], s[j] = s[j], s[i] | ||||
| //} | ||||
| // | ||||
| //type conversationStatus struct { | ||||
| //	ConversationID string | ||||
| //	Pinned         bool | ||||
| //	Recv           bool | ||||
| //} | ||||
|  | ||||
| @ -23,13 +23,17 @@ import ( | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/convert" | ||||
| 	"github.com/openimsdk/protocol/relation" | ||||
| 	"github.com/openimsdk/protocol/sdkws" | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| 	"github.com/openimsdk/tools/mcontext" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| ) | ||||
| 
 | ||||
| func (s *friendServer) GetPaginationBlacks(ctx context.Context, req *relation.GetPaginationBlacksReq) (resp *relation.GetPaginationBlacksResp, err error) { | ||||
| 	if err := s.userRpcClient.Access(ctx, req.UserID); err != nil { | ||||
| 	if err := authverify.CheckAccessV3(ctx, req.UserID, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	total, blacks, err := s.blackDatabase.FindOwnerBlacks(ctx, req.UserID, req.Pagination) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -55,7 +59,7 @@ func (s *friendServer) IsBlack(ctx context.Context, req *relation.IsBlackReq) (* | ||||
| } | ||||
| 
 | ||||
| func (s *friendServer) RemoveBlack(ctx context.Context, req *relation.RemoveBlackReq) (*relation.RemoveBlackResp, error) { | ||||
| 	if err := s.userRpcClient.Access(ctx, req.OwnerUserID); err != nil { | ||||
| 	if err := authverify.CheckAccessV3(ctx, req.OwnerUserID, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| @ -64,6 +68,7 @@ func (s *friendServer) RemoveBlack(ctx context.Context, req *relation.RemoveBlac | ||||
| 	} | ||||
| 
 | ||||
| 	s.notificationSender.BlackDeletedNotification(ctx, req) | ||||
| 	s.webhookAfterRemoveBlack(ctx, &s.config.WebhooksConfig.AfterRemoveBlack, req) | ||||
| 
 | ||||
| 	return &relation.RemoveBlackResp{}, nil | ||||
| } | ||||
| @ -72,6 +77,11 @@ func (s *friendServer) AddBlack(ctx context.Context, req *relation.AddBlackReq) | ||||
| 	if err := authverify.CheckAccessV3(ctx, req.OwnerUserID, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	if err := s.webhookBeforeAddBlack(ctx, &s.config.WebhooksConfig.BeforeAddBlack, req); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	_, err := s.userRpcClient.GetUsersInfo(ctx, []string{req.OwnerUserID, req.BlackUserID}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -90,3 +100,53 @@ func (s *friendServer) AddBlack(ctx context.Context, req *relation.AddBlackReq) | ||||
| 	s.notificationSender.BlackAddedNotification(ctx, req) | ||||
| 	return &relation.AddBlackResp{}, nil | ||||
| } | ||||
| 
 | ||||
| func (s *friendServer) GetSpecifiedBlacks(ctx context.Context, req *relation.GetSpecifiedBlacksReq) (*relation.GetSpecifiedBlacksResp, error) { | ||||
| 	if err := authverify.CheckAccessV3(ctx, req.OwnerUserID, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	if len(req.UserIDList) == 0 { | ||||
| 		return nil, errs.ErrArgs.WrapMsg("userIDList is empty") | ||||
| 	} | ||||
| 
 | ||||
| 	if datautil.Duplicate(req.UserIDList) { | ||||
| 		return nil, errs.ErrArgs.WrapMsg("userIDList repeated") | ||||
| 	} | ||||
| 
 | ||||
| 	userMap, err := s.userRpcClient.GetPublicUserInfoMap(ctx, req.UserIDList) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	blacks, err := s.blackDatabase.FindBlackInfos(ctx, req.OwnerUserID, req.UserIDList) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	blackMap := datautil.SliceToMap(blacks, func(e *model.Black) string { | ||||
| 		return e.BlockUserID | ||||
| 	}) | ||||
| 
 | ||||
| 	resp := &relation.GetSpecifiedBlacksResp{ | ||||
| 		Blacks: make([]*sdkws.BlackInfo, 0, len(req.UserIDList)), | ||||
| 	} | ||||
| 
 | ||||
| 	for _, userID := range req.UserIDList { | ||||
| 		if black := blackMap[userID]; black != nil { | ||||
| 			resp.Blacks = append(resp.Blacks, | ||||
| 				&sdkws.BlackInfo{ | ||||
| 					OwnerUserID:    black.OwnerUserID, | ||||
| 					CreateTime:     black.CreateTime.UnixMilli(), | ||||
| 					BlackUserInfo:  userMap[userID], | ||||
| 					AddSource:      black.AddSource, | ||||
| 					OperatorUserID: black.OperatorUserID, | ||||
| 					Ex:             black.Ex, | ||||
| 				}) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	resp.Total = int32(len(resp.Blacks)) | ||||
| 
 | ||||
| 	return resp, nil | ||||
| } | ||||
|  | ||||
| @ -138,6 +138,18 @@ func (s *friendServer) webhookBeforeAddFriendAgree(ctx context.Context, before * | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func (s *friendServer) webhookAfterAddFriendAgree(ctx context.Context, after *config.AfterConfig, req *relation.RespondFriendApplyReq) { | ||||
| 	cbReq := &cbapi.CallbackAfterAddFriendAgreeReq{ | ||||
| 		CallbackCommand: cbapi.CallbackAfterAddFriendAgreeCommand, | ||||
| 		FromUserID:      req.FromUserID, | ||||
| 		ToUserID:        req.ToUserID, | ||||
| 		HandleMsg:       req.HandleMsg, | ||||
| 		HandleResult:    req.HandleResult, | ||||
| 	} | ||||
| 	resp := &cbapi.CallbackAfterAddFriendAgreeResp{} | ||||
| 	s.webhookClient.AsyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, resp, after) | ||||
| } | ||||
| 
 | ||||
| func (s *friendServer) webhookBeforeImportFriends(ctx context.Context, before *config.BeforeConfig, req *relation.ImportFriendReq) error { | ||||
| 	return webhook.WithCondition(ctx, before, func(ctx context.Context) error { | ||||
| 		cbReq := &cbapi.CallbackBeforeImportFriendsReq{ | ||||
|  | ||||
| @ -212,6 +212,7 @@ func (s *friendServer) RespondFriendApply(ctx context.Context, req *relation.Res | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		s.webhookAfterAddFriendAgree(ctx, &s.config.WebhooksConfig.AfterAddFriendAgree, req) | ||||
| 		s.notificationSender.FriendApplicationAgreedNotification(ctx, req) | ||||
| 		return resp, nil | ||||
| 	} | ||||
| @ -228,20 +229,23 @@ func (s *friendServer) RespondFriendApply(ctx context.Context, req *relation.Res | ||||
| 
 | ||||
| // ok. | ||||
| func (s *friendServer) DeleteFriend(ctx context.Context, req *relation.DeleteFriendReq) (resp *relation.DeleteFriendResp, err error) { | ||||
| 	resp = &relation.DeleteFriendResp{} | ||||
| 	if err := s.userRpcClient.Access(ctx, req.OwnerUserID); err != nil { | ||||
| 	if err := authverify.CheckAccessV3(ctx, req.OwnerUserID, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	_, err = s.db.FindFriendsWithError(ctx, req.OwnerUserID, []string{req.FriendUserID}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	if err := s.db.Delete(ctx, req.OwnerUserID, []string{req.FriendUserID}); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	s.notificationSender.FriendDeletedNotification(ctx, req) | ||||
| 	s.webhookAfterDeleteFriend(ctx, &s.config.WebhooksConfig.AfterDeleteFriend, req) | ||||
| 	return resp, nil | ||||
| 
 | ||||
| 	return &relation.DeleteFriendResp{}, nil | ||||
| } | ||||
| 
 | ||||
| // ok. | ||||
| @ -249,23 +253,34 @@ func (s *friendServer) SetFriendRemark(ctx context.Context, req *relation.SetFri | ||||
| 	if err = s.webhookBeforeSetFriendRemark(ctx, &s.config.WebhooksConfig.BeforeSetFriendRemark, req); err != nil && err != servererrs.ErrCallbackContinue { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	resp = &relation.SetFriendRemarkResp{} | ||||
| 	if err := s.userRpcClient.Access(ctx, req.OwnerUserID); err != nil { | ||||
| 
 | ||||
| 	if err := authverify.CheckAccessV3(ctx, req.OwnerUserID, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	_, err = s.db.FindFriendsWithError(ctx, req.OwnerUserID, []string{req.FriendUserID}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	if err := s.db.UpdateRemark(ctx, req.OwnerUserID, req.FriendUserID, req.Remark); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	s.webhookAfterSetFriendRemark(ctx, &s.config.WebhooksConfig.AfterSetFriendRemark, req) | ||||
| 	s.notificationSender.FriendRemarkSetNotification(ctx, req.OwnerUserID, req.FriendUserID) | ||||
| 	return resp, nil | ||||
| 
 | ||||
| 	return &relation.SetFriendRemarkResp{}, nil | ||||
| } | ||||
| 
 | ||||
| func (s *friendServer) GetFriendInfo(ctx context.Context, req *relation.GetFriendInfoReq) (*relation.GetFriendInfoResp, error) { | ||||
| 	friends, err := s.db.FindFriendsWithError(ctx, req.OwnerUserID, req.FriendUserIDs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &relation.GetFriendInfoResp{FriendInfos: convert.FriendOnlyDB2PbOnly(friends)}, nil | ||||
| } | ||||
| 
 | ||||
| // ok. | ||||
| func (s *friendServer) GetDesignatedFriends(ctx context.Context, req *relation.GetDesignatedFriendsReq) (resp *relation.GetDesignatedFriendsResp, err error) { | ||||
| 	resp = &relation.GetDesignatedFriendsResp{} | ||||
| 	if datautil.Duplicate(req.FriendUserIDs) { | ||||
| @ -309,36 +324,45 @@ func (s *friendServer) GetDesignatedFriendsApply(ctx context.Context, | ||||
| 
 | ||||
| // Get received friend requests (i.e., those initiated by others). | ||||
| func (s *friendServer) GetPaginationFriendsApplyTo(ctx context.Context, req *relation.GetPaginationFriendsApplyToReq) (resp *relation.GetPaginationFriendsApplyToResp, err error) { | ||||
| 	if err := s.userRpcClient.Access(ctx, req.UserID); err != nil { | ||||
| 	if err := authverify.CheckAccessV3(ctx, req.UserID, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	total, friendRequests, err := s.db.PageFriendRequestToMe(ctx, req.UserID, req.Pagination) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	resp = &relation.GetPaginationFriendsApplyToResp{} | ||||
| 	resp.FriendRequests, err = convert.FriendRequestDB2Pb(ctx, friendRequests, s.userRpcClient.GetUsersInfoMap) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	resp.Total = int32(total) | ||||
| 
 | ||||
| 	return resp, nil | ||||
| } | ||||
| 
 | ||||
| func (s *friendServer) GetPaginationFriendsApplyFrom(ctx context.Context, req *relation.GetPaginationFriendsApplyFromReq) (resp *relation.GetPaginationFriendsApplyFromResp, err error) { | ||||
| 	resp = &relation.GetPaginationFriendsApplyFromResp{} | ||||
| 	if err := s.userRpcClient.Access(ctx, req.UserID); err != nil { | ||||
| 
 | ||||
| 	if err := authverify.CheckAccessV3(ctx, req.UserID, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	total, friendRequests, err := s.db.PageFriendRequestFromMe(ctx, req.UserID, req.Pagination) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	resp.FriendRequests, err = convert.FriendRequestDB2Pb(ctx, friendRequests, s.userRpcClient.GetUsersInfoMap) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	resp.Total = int32(total) | ||||
| 
 | ||||
| 	return resp, nil | ||||
| } | ||||
| 
 | ||||
| @ -353,31 +377,37 @@ func (s *friendServer) IsFriend(ctx context.Context, req *relation.IsFriendReq) | ||||
| } | ||||
| 
 | ||||
| func (s *friendServer) GetPaginationFriends(ctx context.Context, req *relation.GetPaginationFriendsReq) (resp *relation.GetPaginationFriendsResp, err error) { | ||||
| 	if err := s.userRpcClient.Access(ctx, req.UserID); err != nil { | ||||
| 	if err := authverify.CheckAccessV3(ctx, req.UserID, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	total, friends, err := s.db.PageOwnerFriends(ctx, req.UserID, req.Pagination) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	resp = &relation.GetPaginationFriendsResp{} | ||||
| 	resp.FriendsInfo, err = convert.FriendsDB2Pb(ctx, friends, s.userRpcClient.GetUsersInfoMap) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	resp.Total = int32(total) | ||||
| 
 | ||||
| 	return resp, nil | ||||
| } | ||||
| 
 | ||||
| func (s *friendServer) GetFriendIDs(ctx context.Context, req *relation.GetFriendIDsReq) (resp *relation.GetFriendIDsResp, err error) { | ||||
| 	if err := s.userRpcClient.Access(ctx, req.UserID); err != nil { | ||||
| 	if err := authverify.CheckAccessV3(ctx, req.UserID, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	resp = &relation.GetFriendIDsResp{} | ||||
| 	resp.FriendIDs, err = s.db.FindFriendUserIDs(ctx, req.UserID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	return resp, nil | ||||
| } | ||||
| 
 | ||||
| @ -385,35 +415,45 @@ func (s *friendServer) GetSpecifiedFriendsInfo(ctx context.Context, req *relatio | ||||
| 	if len(req.UserIDList) == 0 { | ||||
| 		return nil, errs.ErrArgs.WrapMsg("userIDList is empty") | ||||
| 	} | ||||
| 
 | ||||
| 	if datautil.Duplicate(req.UserIDList) { | ||||
| 		return nil, errs.ErrArgs.WrapMsg("userIDList repeated") | ||||
| 	} | ||||
| 
 | ||||
| 	userMap, err := s.userRpcClient.GetUsersInfoMap(ctx, req.UserIDList) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	friends, err := s.db.FindFriendsWithError(ctx, req.OwnerUserID, req.UserIDList) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	blacks, err := s.blackDatabase.FindBlackInfos(ctx, req.OwnerUserID, req.UserIDList) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	friendMap := datautil.SliceToMap(friends, func(e *model.Friend) string { | ||||
| 		return e.FriendUserID | ||||
| 	}) | ||||
| 
 | ||||
| 	blackMap := datautil.SliceToMap(blacks, func(e *model.Black) string { | ||||
| 		return e.BlockUserID | ||||
| 	}) | ||||
| 
 | ||||
| 	resp := &relation.GetSpecifiedFriendsInfoResp{ | ||||
| 		Infos: make([]*relation.GetSpecifiedFriendsInfoInfo, 0, len(req.UserIDList)), | ||||
| 	} | ||||
| 
 | ||||
| 	for _, userID := range req.UserIDList { | ||||
| 		user := userMap[userID] | ||||
| 
 | ||||
| 		if user == nil { | ||||
| 			continue | ||||
| 		} | ||||
| 
 | ||||
| 		var friendInfo *sdkws.FriendInfo | ||||
| 		if friend := friendMap[userID]; friend != nil { | ||||
| 			friendInfo = &sdkws.FriendInfo{ | ||||
| @ -426,6 +466,7 @@ func (s *friendServer) GetSpecifiedFriendsInfo(ctx context.Context, req *relatio | ||||
| 				IsPinned:       friend.IsPinned, | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		var blackInfo *sdkws.BlackInfo | ||||
| 		if black := blackMap[userID]; black != nil { | ||||
| 			blackInfo = &sdkws.BlackInfo{ | ||||
| @ -436,12 +477,14 @@ func (s *friendServer) GetSpecifiedFriendsInfo(ctx context.Context, req *relatio | ||||
| 				Ex:             black.Ex, | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		resp.Infos = append(resp.Infos, &relation.GetSpecifiedFriendsInfoInfo{ | ||||
| 			UserInfo:   user, | ||||
| 			FriendInfo: friendInfo, | ||||
| 			BlackInfo:  blackInfo, | ||||
| 		}) | ||||
| 	} | ||||
| 
 | ||||
| 	return resp, nil | ||||
| } | ||||
| 
 | ||||
|  | ||||
| @ -17,9 +17,10 @@ package third | ||||
| import ( | ||||
| 	"context" | ||||
| 	"crypto/rand" | ||||
| 	relationtb "github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||
| 	"time" | ||||
| 
 | ||||
| 	relationtb "github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/servererrs" | ||||
| 	"github.com/openimsdk/protocol/constant" | ||||
|  | ||||
| @ -16,6 +16,7 @@ package user | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/webhook" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| 
 | ||||
| @ -88,7 +89,6 @@ func (s *userServer) webhookBeforeUserRegister(ctx context.Context, before *conf | ||||
| 	return webhook.WithCondition(ctx, before, func(ctx context.Context) error { | ||||
| 		cbReq := &cbapi.CallbackBeforeUserRegisterReq{ | ||||
| 			CallbackCommand: cbapi.CallbackBeforeUserRegisterCommand, | ||||
| 			Secret:          req.Secret, | ||||
| 			Users:           req.Users, | ||||
| 		} | ||||
| 
 | ||||
| @ -108,7 +108,6 @@ func (s *userServer) webhookBeforeUserRegister(ctx context.Context, before *conf | ||||
| func (s *userServer) webhookAfterUserRegister(ctx context.Context, after *config.AfterConfig, req *pbuser.UserRegisterReq) { | ||||
| 	cbReq := &cbapi.CallbackAfterUserRegisterReq{ | ||||
| 		CallbackCommand: cbapi.CallbackAfterUserRegisterCommand, | ||||
| 		Secret:          req.Secret, | ||||
| 		Users:           req.Users, | ||||
| 	} | ||||
| 
 | ||||
|  | ||||
| @ -2,6 +2,9 @@ package user | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 
 | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| 
 | ||||
| 	"github.com/openimsdk/protocol/constant" | ||||
| 	pbuser "github.com/openimsdk/protocol/user" | ||||
| ) | ||||
| @ -59,7 +62,7 @@ func (s *userServer) SetUserStatus(ctx context.Context, req *pbuser.SetUserStatu | ||||
| 	case constant.Online: | ||||
| 		online = []int32{req.PlatformID} | ||||
| 	case constant.Offline: | ||||
| 		online = []int32{req.PlatformID} | ||||
| 		offline = []int32{req.PlatformID} | ||||
| 	} | ||||
| 	if err := s.online.SetUserOnline(ctx, req.UserID, online, offline); err != nil { | ||||
| 		return nil, err | ||||
| @ -80,3 +83,22 @@ func (s *userServer) SetUserOnlineStatus(ctx context.Context, req *pbuser.SetUse | ||||
| 	} | ||||
| 	return &pbuser.SetUserOnlineStatusResp{}, nil | ||||
| } | ||||
| 
 | ||||
| func (s *userServer) GetAllOnlineUsers(ctx context.Context, req *pbuser.GetAllOnlineUsersReq) (*pbuser.GetAllOnlineUsersResp, error) { | ||||
| 	resMap, nextCursor, err := s.online.GetAllOnlineUsers(ctx, req.Cursor) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	resp := &pbuser.GetAllOnlineUsersResp{ | ||||
| 		StatusList: make([]*pbuser.OnlineStatus, 0, len(resMap)), | ||||
| 		NextCursor: nextCursor, | ||||
| 	} | ||||
| 	for userID, plats := range resMap { | ||||
| 		resp.StatusList = append(resp.StatusList, &pbuser.OnlineStatus{ | ||||
| 			UserID:      userID, | ||||
| 			Status:      int32(datautil.If(len(plats) > 0, constant.Online, constant.Offline)), | ||||
| 			PlatformIDs: plats, | ||||
| 		}) | ||||
| 	} | ||||
| 	return resp, nil | ||||
| } | ||||
|  | ||||
| @ -47,7 +47,6 @@ import ( | ||||
| 	"github.com/openimsdk/tools/db/pagination" | ||||
| 	registry "github.com/openimsdk/tools/discovery" | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| 	"github.com/openimsdk/tools/log" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| 	"google.golang.org/grpc" | ||||
| ) | ||||
| @ -263,10 +262,11 @@ func (s *userServer) UserRegister(ctx context.Context, req *pbuser.UserRegisterR | ||||
| 	if len(req.Users) == 0 { | ||||
| 		return nil, errs.ErrArgs.WrapMsg("users is empty") | ||||
| 	} | ||||
| 	if req.Secret != s.config.Share.Secret { | ||||
| 		log.ZDebug(ctx, "UserRegister", s.config.Share.Secret, req.Secret) | ||||
| 		return nil, errs.ErrNoPermission.WrapMsg("secret invalid") | ||||
| 
 | ||||
| 	if err = authverify.CheckAdmin(ctx, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	if datautil.DuplicateAny(req.Users, func(e *sdkws.UserInfo) string { return e.UserID }) { | ||||
| 		return nil, errs.ErrArgs.WrapMsg("userID repeated") | ||||
| 	} | ||||
|  | ||||
| @ -79,13 +79,13 @@ func Start(ctx context.Context, config *CronTaskConfig) error { | ||||
| 		now := time.Now() | ||||
| 		deltime := now.Add(-time.Hour * 24 * time.Duration(config.CronTask.RetainChatRecords)) | ||||
| 		ctx := mcontext.SetOperationID(ctx, fmt.Sprintf("cron_%d_%d", os.Getpid(), deltime.UnixMilli())) | ||||
| 		log.ZInfo(ctx, "clear chat records", "deltime", deltime, "timestamp", deltime.UnixMilli()) | ||||
| 		log.ZDebug(ctx, "clear chat records", "deltime", deltime, "timestamp", deltime.UnixMilli()) | ||||
| 
 | ||||
| 		if _, err := msgClient.ClearMsg(ctx, &msg.ClearMsgReq{Timestamp: deltime.UnixMilli()}); err != nil { | ||||
| 			log.ZError(ctx, "cron clear chat records failed", err, "deltime", deltime, "cont", time.Since(now)) | ||||
| 			return | ||||
| 		} | ||||
| 		log.ZInfo(ctx, "cron clear chat records success", "deltime", deltime, "cont", time.Since(now)) | ||||
| 		log.ZDebug(ctx, "cron clear chat records success", "deltime", deltime, "cont", time.Since(now)) | ||||
| 	} | ||||
| 	if _, err := crontab.AddFunc(config.CronTask.CronExecuteTime, clearMsgFunc); err != nil { | ||||
| 		return errs.Wrap(err) | ||||
| @ -95,7 +95,7 @@ func Start(ctx context.Context, config *CronTaskConfig) error { | ||||
| 	msgDestructFunc := func() { | ||||
| 		now := time.Now() | ||||
| 		ctx := mcontext.SetOperationID(ctx, fmt.Sprintf("cron_%d_%d", os.Getpid(), now.UnixMilli())) | ||||
| 		log.ZInfo(ctx, "msg destruct cron start", "now", now) | ||||
| 		log.ZDebug(ctx, "msg destruct cron start", "now", now) | ||||
| 
 | ||||
| 		conversations, err := conversationClient.GetConversationsNeedDestructMsgs(ctx, &pbconversation.GetConversationsNeedDestructMsgsReq{}) | ||||
| 		if err != nil { | ||||
| @ -108,7 +108,7 @@ func Start(ctx context.Context, config *CronTaskConfig) error { | ||||
| 				return | ||||
| 			} | ||||
| 		} | ||||
| 		log.ZInfo(ctx, "msg destruct cron task completed", "cont", time.Since(now)) | ||||
| 		log.ZDebug(ctx, "msg destruct cron task completed", "cont", time.Since(now)) | ||||
| 	} | ||||
| 	if _, err := crontab.AddFunc(config.CronTask.CronExecuteTime, msgDestructFunc); err != nil { | ||||
| 		return errs.Wrap(err) | ||||
| @ -119,18 +119,18 @@ func Start(ctx context.Context, config *CronTaskConfig) error { | ||||
| 	// 	now := time.Now() | ||||
| 	// 	deleteTime := now.Add(-time.Hour * 24 * time.Duration(config.CronTask.FileExpireTime)) | ||||
| 	// 	ctx := mcontext.SetOperationID(ctx, fmt.Sprintf("cron_%d_%d", os.Getpid(), deleteTime.UnixMilli())) | ||||
| 	// 	log.ZInfo(ctx, "deleteoutDatedData ", "deletetime", deleteTime, "timestamp", deleteTime.UnixMilli()) | ||||
| 	// 	log.ZDebug(ctx, "deleteoutDatedData ", "deletetime", deleteTime, "timestamp", deleteTime.UnixMilli()) | ||||
| 	// 	if _, err := thirdClient.DeleteOutdatedData(ctx, &third.DeleteOutdatedDataReq{ExpireTime: deleteTime.UnixMilli()}); err != nil { | ||||
| 	// 		log.ZError(ctx, "cron deleteoutDatedData failed", err, "deleteTime", deleteTime, "cont", time.Since(now)) | ||||
| 	// 		return | ||||
| 	// 	} | ||||
| 	// 	log.ZInfo(ctx, "cron deleteoutDatedData success", "deltime", deleteTime, "cont", time.Since(now)) | ||||
| 	// 	log.ZDebug(ctx, "cron deleteoutDatedData success", "deltime", deleteTime, "cont", time.Since(now)) | ||||
| 	// } | ||||
| 	// if _, err := crontab.AddFunc(config.CronTask.CronExecuteTime, deleteObjectFunc); err != nil { | ||||
| 	// 	return errs.Wrap(err) | ||||
| 	// } | ||||
| 
 | ||||
| 	log.ZInfo(ctx, "start cron task", "CronExecuteTime", config.CronTask.CronExecuteTime) | ||||
| 	log.ZDebug(ctx, "start cron task", "CronExecuteTime", config.CronTask.CronExecuteTime) | ||||
| 	crontab.Start() | ||||
| 	<-ctx.Done() | ||||
| 	return nil | ||||
|  | ||||
| @ -18,13 +18,14 @@ const ( | ||||
| 	CallbackBeforeInviteJoinGroupCommand    = "callbackBeforeInviteJoinGroupCommand" | ||||
| 	CallbackAfterJoinGroupCommand           = "callbackAfterJoinGroupCommand" | ||||
| 	CallbackAfterSetGroupInfoCommand        = "callbackAfterSetGroupInfoCommand" | ||||
| 	CallbackAfterSetGroupInfoEXCommand      = "callbackAfterSetGroupInfoCommandEX" | ||||
| 	CallbackAfterSetGroupInfoExCommand      = "callbackAfterSetGroupInfoExCommand" | ||||
| 	CallbackBeforeSetGroupInfoCommand       = "callbackBeforeSetGroupInfoCommand" | ||||
| 	CallbackBeforeSetGroupInfoEXCommand     = "callbackBeforeSetGroupInfoEXCommand" | ||||
| 	CallbackBeforeSetGroupInfoExCommand     = "callbackBeforeSetGroupInfoExCommand" | ||||
| 	CallbackAfterRevokeMsgCommand           = "callbackBeforeAfterMsgCommand" | ||||
| 	CallbackBeforeAddBlackCommand           = "callbackBeforeAddBlackCommand" | ||||
| 	CallbackAfterAddFriendCommand           = "callbackAfterAddFriendCommand" | ||||
| 	CallbackBeforeAddFriendAgreeCommand     = "callbackBeforeAddFriendAgreeCommand" | ||||
| 	CallbackAfterAddFriendAgreeCommand      = "callbackAfterAddFriendAgreeCommand" | ||||
| 	CallbackAfterDeleteFriendCommand        = "callbackAfterDeleteFriendCommand" | ||||
| 	CallbackBeforeImportFriendsCommand      = "callbackBeforeImportFriendsCommand" | ||||
| 	CallbackAfterImportFriendsCommand       = "callbackAfterImportFriendsCommand" | ||||
|  | ||||
| @ -90,6 +90,18 @@ type CallbackBeforeAddFriendAgreeResp struct { | ||||
| 	CommonCallbackResp | ||||
| } | ||||
| 
 | ||||
| type CallbackAfterAddFriendAgreeReq struct { | ||||
| 	CallbackCommand `json:"callbackCommand"` | ||||
| 	FromUserID      string `json:"fromUserID" ` | ||||
| 	ToUserID        string `json:"blackUserID"` | ||||
| 	HandleResult    int32  `json:"HandleResult"` | ||||
| 	HandleMsg       string `json:"HandleMsg"` | ||||
| } | ||||
| 
 | ||||
| type CallbackAfterAddFriendAgreeResp struct { | ||||
| 	CommonCallbackResp | ||||
| } | ||||
| 
 | ||||
| type CallbackAfterDeleteFriendReq struct { | ||||
| 	CallbackCommand `json:"callbackCommand"` | ||||
| 	OwnerUserID     string `json:"ownerUserID" ` | ||||
|  | ||||
| @ -244,11 +244,11 @@ type CallbackAfterSetGroupInfoResp struct { | ||||
| 	CommonCallbackResp | ||||
| } | ||||
| 
 | ||||
| type CallbackBeforeSetGroupInfoEXReq struct { | ||||
| type CallbackBeforeSetGroupInfoExReq struct { | ||||
| 	CallbackCommand   `json:"callbackCommand"` | ||||
| 	OperationID       string                  `json:"operationID"` | ||||
| 	GroupID           string                  `json:"groupID"` | ||||
| 	GroupName         string                  `json:"groupName"` | ||||
| 	GroupName         *wrapperspb.StringValue `json:"groupName"` | ||||
| 	Notification      *wrapperspb.StringValue `json:"notification"` | ||||
| 	Introduction      *wrapperspb.StringValue `json:"introduction"` | ||||
| 	FaceURL           *wrapperspb.StringValue `json:"faceURL"` | ||||
| @ -258,10 +258,10 @@ type CallbackBeforeSetGroupInfoEXReq struct { | ||||
| 	ApplyMemberFriend *wrapperspb.Int32Value  `json:"applyMemberFriend"` | ||||
| } | ||||
| 
 | ||||
| type CallbackBeforeSetGroupInfoEXResp struct { | ||||
| type CallbackBeforeSetGroupInfoExResp struct { | ||||
| 	CommonCallbackResp | ||||
| 	GroupID           string                  `json:"groupID"` | ||||
| 	GroupName         string                  `json:"groupName"` | ||||
| 	GroupName         *wrapperspb.StringValue `json:"groupName"` | ||||
| 	Notification      *wrapperspb.StringValue `json:"notification"` | ||||
| 	Introduction      *wrapperspb.StringValue `json:"introduction"` | ||||
| 	FaceURL           *wrapperspb.StringValue `json:"faceURL"` | ||||
| @ -271,11 +271,11 @@ type CallbackBeforeSetGroupInfoEXResp struct { | ||||
| 	ApplyMemberFriend *wrapperspb.Int32Value  `json:"applyMemberFriend"` | ||||
| } | ||||
| 
 | ||||
| type CallbackAfterSetGroupInfoEXReq struct { | ||||
| type CallbackAfterSetGroupInfoExReq struct { | ||||
| 	CallbackCommand   `json:"callbackCommand"` | ||||
| 	OperationID       string                  `json:"operationID"` | ||||
| 	GroupID           string                  `json:"groupID"` | ||||
| 	GroupName         string                  `json:"groupName"` | ||||
| 	GroupName         *wrapperspb.StringValue `json:"groupName"` | ||||
| 	Notification      *wrapperspb.StringValue `json:"notification"` | ||||
| 	Introduction      *wrapperspb.StringValue `json:"introduction"` | ||||
| 	FaceURL           *wrapperspb.StringValue `json:"faceURL"` | ||||
| @ -285,6 +285,6 @@ type CallbackAfterSetGroupInfoEXReq struct { | ||||
| 	ApplyMemberFriend *wrapperspb.Int32Value  `json:"applyMemberFriend"` | ||||
| } | ||||
| 
 | ||||
| type CallbackAfterSetGroupInfoEXResp struct { | ||||
| type CallbackAfterSetGroupInfoExResp struct { | ||||
| 	CommonCallbackResp | ||||
| } | ||||
|  | ||||
| @ -72,7 +72,6 @@ type CallbackAfterUpdateUserInfoExResp struct { | ||||
| 
 | ||||
| type CallbackBeforeUserRegisterReq struct { | ||||
| 	CallbackCommand `json:"callbackCommand"` | ||||
| 	Secret          string            `json:"secret"` | ||||
| 	Users           []*sdkws.UserInfo `json:"users"` | ||||
| } | ||||
| 
 | ||||
| @ -83,7 +82,6 @@ type CallbackBeforeUserRegisterResp struct { | ||||
| 
 | ||||
| type CallbackAfterUserRegisterReq struct { | ||||
| 	CallbackCommand `json:"callbackCommand"` | ||||
| 	Secret          string            `json:"secret"` | ||||
| 	Users           []*sdkws.UserInfo `json:"users"` | ||||
| } | ||||
| 
 | ||||
|  | ||||
| @ -15,13 +15,14 @@ | ||||
| package cmd | ||||
| 
 | ||||
| import ( | ||||
| 	"math" | ||||
| 	"testing" | ||||
| 
 | ||||
| 	"github.com/openimsdk/protocol/auth" | ||||
| 	"github.com/openimsdk/tools/apiresp" | ||||
| 	"github.com/openimsdk/tools/utils/jsonutil" | ||||
| 	"github.com/stretchr/testify/mock" | ||||
| 	"go.mongodb.org/mongo-driver/bson/primitive" | ||||
| 	"math" | ||||
| 	"testing" | ||||
| ) | ||||
| 
 | ||||
| // MockRootCmd is a mock type for the RootCmd type | ||||
| @ -39,7 +40,7 @@ func TestName(t *testing.T) { | ||||
| 		ErrCode: 1234, | ||||
| 		ErrMsg:  "test", | ||||
| 		ErrDlt:  "4567", | ||||
| 		Data: &auth.UserTokenResp{ | ||||
| 		Data: &auth.GetUserTokenResp{ | ||||
| 			Token:             "1234567", | ||||
| 			ExpireTimeSeconds: math.MaxInt64, | ||||
| 		}, | ||||
| @ -51,7 +52,7 @@ func TestName(t *testing.T) { | ||||
| 	t.Log(string(data)) | ||||
| 
 | ||||
| 	var rReso apiresp.ApiResponse | ||||
| 	rReso.Data = &auth.UserTokenResp{} | ||||
| 	rReso.Data = &auth.GetUserTokenResp{} | ||||
| 
 | ||||
| 	if err := jsonutil.JsonUnmarshal(data, &rReso); err != nil { | ||||
| 		panic(err) | ||||
|  | ||||
| @ -37,7 +37,6 @@ func NewPushRpcCmd() *PushRpcCmd { | ||||
| 	ret.configMap = map[string]any{ | ||||
| 		OpenIMPushCfgFileName:    &pushConfig.RpcConfig, | ||||
| 		RedisConfigFileName:      &pushConfig.RedisConfig, | ||||
| 		MongodbConfigFileName:    &pushConfig.MongodbConfig, | ||||
| 		KafkaConfigFileName:      &pushConfig.KafkaConfig, | ||||
| 		ShareFileName:            &pushConfig.Share, | ||||
| 		NotificationFileName:     &pushConfig.NotificationConfig, | ||||
|  | ||||
| @ -129,10 +129,11 @@ func (r *RootCmd) applyOptions(opts ...func(*CmdOpts)) *CmdOpts { | ||||
| } | ||||
| 
 | ||||
| func (r *RootCmd) initializeLogger(cmdOpts *CmdOpts) error { | ||||
| 	err := log.InitFromConfig( | ||||
| 	err := log.InitLoggerFromConfig( | ||||
| 
 | ||||
| 		cmdOpts.loggerPrefixName, | ||||
| 		r.processName, | ||||
| 		"", "", | ||||
| 		r.log.RemainLogLevel, | ||||
| 		r.log.IsStdout, | ||||
| 		r.log.IsJson, | ||||
|  | ||||
| @ -81,9 +81,12 @@ type Kafka struct { | ||||
| 	ToRedisTopic       string   `mapstructure:"toRedisTopic"` | ||||
| 	ToMongoTopic       string   `mapstructure:"toMongoTopic"` | ||||
| 	ToPushTopic        string   `mapstructure:"toPushTopic"` | ||||
| 	ToOfflinePushTopic string   `mapstructure:"toOfflinePushTopic"` | ||||
| 	ToRedisGroupID     string   `mapstructure:"toRedisGroupID"` | ||||
| 	ToMongoGroupID     string   `mapstructure:"toMongoGroupID"` | ||||
| 	ToPushGroupID      string   `mapstructure:"toPushGroupID"` | ||||
| 	ToOfflineGroupID   string   `mapstructure:"toOfflinePushGroupID"` | ||||
| 
 | ||||
| 	Tls TLSConfig `mapstructure:"tls"` | ||||
| } | ||||
| type TLSConfig struct { | ||||
| @ -99,6 +102,7 @@ type API struct { | ||||
| 	Api struct { | ||||
| 		ListenIP         string `mapstructure:"listenIP"` | ||||
| 		Ports            []int  `mapstructure:"ports"` | ||||
| 		CompressionLevel int    `mapstructure:"compressionLevel"` | ||||
| 	} `mapstructure:"api"` | ||||
| 	Prometheus struct { | ||||
| 		Enable     bool   `mapstructure:"enable"` | ||||
| @ -181,7 +185,6 @@ type MsgGateway struct { | ||||
| 		WebsocketMaxMsgLen  int   `mapstructure:"websocketMaxMsgLen"` | ||||
| 		WebsocketTimeout    int   `mapstructure:"websocketTimeout"` | ||||
| 	} `mapstructure:"longConnSvr"` | ||||
| 	MultiLoginPolicy int `mapstructure:"multiLoginPolicy"` | ||||
| } | ||||
| 
 | ||||
| type MsgTransfer struct { | ||||
| @ -220,6 +223,7 @@ type Push struct { | ||||
| 		BadgeCount bool   `mapstructure:"badgeCount"` | ||||
| 		Production bool   `mapstructure:"production"` | ||||
| 	} `mapstructure:"iosPush"` | ||||
| 	FullUserCache bool `mapstructure:"fullUserCache"` | ||||
| } | ||||
| 
 | ||||
| type Auth struct { | ||||
| @ -336,26 +340,50 @@ type Redis struct { | ||||
| 	Password    string   `mapstructure:"password"` | ||||
| 	ClusterMode bool     `mapstructure:"clusterMode"` | ||||
| 	DB          int      `mapstructure:"storage"` | ||||
| 	MaxRetry    int      `mapstructure:"MaxRetry"` | ||||
| 	MaxRetry    int      `mapstructure:"maxRetry"` | ||||
| 	PoolSize    int      `mapstructure:"poolSize"` | ||||
| } | ||||
| 
 | ||||
| type BeforeConfig struct { | ||||
| 	Enable         bool     `mapstructure:"enable"` | ||||
| 	Timeout        int      `mapstructure:"timeout"` | ||||
| 	FailedContinue bool     `mapstructure:"failedContinue"` | ||||
| 	AllowedTypes   []string `mapstructure:"allowedTypes"` | ||||
| 	DeniedTypes    []string `mapstructure:"deniedTypes"` | ||||
| } | ||||
| 
 | ||||
| type AfterConfig struct { | ||||
| 	Enable       bool     `mapstructure:"enable"` | ||||
| 	Timeout      int      `mapstructure:"timeout"` | ||||
| 	AttentionIds []string `mapstructure:"attentionIds"` | ||||
| 	AllowedTypes []string `mapstructure:"allowedTypes"` | ||||
| 	DeniedTypes  []string `mapstructure:"deniedTypes"` | ||||
| } | ||||
| 
 | ||||
| type Share struct { | ||||
| 	Secret          string          `mapstructure:"secret"` | ||||
| 	RpcRegisterName RpcRegisterName `mapstructure:"rpcRegisterName"` | ||||
| 	IMAdminUserID   []string        `mapstructure:"imAdminUserID"` | ||||
| 	MultiLogin      MultiLogin      `mapstructure:"multiLogin"` | ||||
| } | ||||
| 
 | ||||
| type MultiLogin struct { | ||||
| 	Policy            int `mapstructure:"policy"` | ||||
| 	MaxNumOneEnd      int `mapstructure:"maxNumOneEnd"` | ||||
| 	CustomizeLoginNum struct { | ||||
| 		IOS     int `mapstructure:"ios"` | ||||
| 		Android int `mapstructure:"android"` | ||||
| 		Windows int `mapstructure:"windows"` | ||||
| 		OSX     int `mapstructure:"osx"` | ||||
| 		Web     int `mapstructure:"web"` | ||||
| 		MiniWeb int `mapstructure:"miniWeb"` | ||||
| 		Linux   int `mapstructure:"linux"` | ||||
| 		APad    int `mapstructure:"aPad"` | ||||
| 		IPad    int `mapstructure:"iPad"` | ||||
| 		Admin   int `mapstructure:"admin"` | ||||
| 	} `mapstructure:"customizeLoginNum"` | ||||
| } | ||||
| 
 | ||||
| type RpcRegisterName struct { | ||||
| 	User           string `mapstructure:"user"` | ||||
| 	Friend         string `mapstructure:"friend"` | ||||
| @ -422,12 +450,13 @@ type Webhooks struct { | ||||
| 	BeforeInviteUserToGroup  BeforeConfig `mapstructure:"beforeInviteUserToGroup"` | ||||
| 	AfterSetGroupInfo        AfterConfig  `mapstructure:"afterSetGroupInfo"` | ||||
| 	BeforeSetGroupInfo       BeforeConfig `mapstructure:"beforeSetGroupInfo"` | ||||
| 	AfterSetGroupInfoEX      AfterConfig  `mapstructure:"afterSetGroupInfoEX"` | ||||
| 	BeforeSetGroupInfoEX     BeforeConfig `mapstructure:"beforeSetGroupInfoEX"` | ||||
| 	AfterSetGroupInfoEx      AfterConfig  `mapstructure:"afterSetGroupInfoEx"` | ||||
| 	BeforeSetGroupInfoEx     BeforeConfig `mapstructure:"beforeSetGroupInfoEx"` | ||||
| 	AfterRevokeMsg           AfterConfig  `mapstructure:"afterRevokeMsg"` | ||||
| 	BeforeAddBlack           BeforeConfig `mapstructure:"beforeAddBlack"` | ||||
| 	AfterAddFriend           AfterConfig  `mapstructure:"afterAddFriend"` | ||||
| 	BeforeAddFriendAgree     BeforeConfig `mapstructure:"beforeAddFriendAgree"` | ||||
| 	AfterAddFriendAgree      AfterConfig  `mapstructure:"afterAddFriendAgree"` | ||||
| 	AfterDeleteFriend        AfterConfig  `mapstructure:"afterDeleteFriend"` | ||||
| 	BeforeImportFriends      BeforeConfig `mapstructure:"beforeImportFriends"` | ||||
| 	AfterImportFriends       AfterConfig  `mapstructure:"afterImportFriends"` | ||||
| @ -474,6 +503,7 @@ func (r *Redis) Build() *redisutil.Config { | ||||
| 		Password:    r.Password, | ||||
| 		DB:          r.DB, | ||||
| 		MaxRetry:    r.MaxRetry, | ||||
| 		PoolSize:    r.PoolSize, | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
|  | ||||
| @ -18,6 +18,7 @@ import ( | ||||
| 	"context" | ||||
| 	"fmt" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||
| 	"github.com/openimsdk/protocol/relation" | ||||
| 
 | ||||
| 	"github.com/openimsdk/protocol/sdkws" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| @ -35,9 +36,7 @@ func FriendPb2DB(friend *sdkws.FriendInfo) *model.Friend { | ||||
| 	return dbFriend | ||||
| } | ||||
| 
 | ||||
| func FriendDB2Pb(ctx context.Context, friendDB *model.Friend, | ||||
| 	getUsers func(ctx context.Context, userIDs []string) (map[string]*sdkws.UserInfo, error), | ||||
| ) (*sdkws.FriendInfo, error) { | ||||
| func FriendDB2Pb(ctx context.Context, friendDB *model.Friend, getUsers func(ctx context.Context, userIDs []string) (map[string]*sdkws.UserInfo, error)) (*sdkws.FriendInfo, error) { | ||||
| 	users, err := getUsers(ctx, []string{friendDB.FriendUserID}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -53,11 +52,7 @@ func FriendDB2Pb(ctx context.Context, friendDB *model.Friend, | ||||
| 	}, nil | ||||
| } | ||||
| 
 | ||||
| func FriendsDB2Pb( | ||||
| 	ctx context.Context, | ||||
| 	friendsDB []*model.Friend, | ||||
| 	getUsers func(ctx context.Context, userIDs []string) (map[string]*sdkws.UserInfo, error), | ||||
| ) (friendsPb []*sdkws.FriendInfo, err error) { | ||||
| func FriendsDB2Pb(ctx context.Context, friendsDB []*model.Friend, getUsers func(ctx context.Context, userIDs []string) (map[string]*sdkws.UserInfo, error)) (friendsPb []*sdkws.FriendInfo, err error) { | ||||
| 	if len(friendsDB) == 0 { | ||||
| 		return nil, nil | ||||
| 	} | ||||
| @ -86,7 +81,21 @@ func FriendsDB2Pb( | ||||
| 		friendsPb = append(friendsPb, friendPb) | ||||
| 	} | ||||
| 	return friendsPb, nil | ||||
| } | ||||
| 
 | ||||
| func FriendOnlyDB2PbOnly(friendsDB []*model.Friend) []*relation.FriendInfoOnly { | ||||
| 	return datautil.Slice(friendsDB, func(f *model.Friend) *relation.FriendInfoOnly { | ||||
| 		return &relation.FriendInfoOnly{ | ||||
| 			OwnerUserID:    f.OwnerUserID, | ||||
| 			FriendUserID:   f.FriendUserID, | ||||
| 			Remark:         f.Remark, | ||||
| 			CreateTime:     f.CreateTime.UnixMilli(), | ||||
| 			AddSource:      f.AddSource, | ||||
| 			OperatorUserID: f.OperatorUserID, | ||||
| 			Ex:             f.Ex, | ||||
| 			IsPinned:       f.IsPinned, | ||||
| 		} | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func FriendRequestDB2Pb(ctx context.Context, friendRequests []*model.FriendRequest, getUsers func(ctx context.Context, userIDs []string) (map[string]*sdkws.UserInfo, error)) ([]*sdkws.FriendRequest, error) { | ||||
|  | ||||
| @ -23,4 +23,8 @@ var ( | ||||
| 		Name: "msg_offline_push_failed_total", | ||||
| 		Help: "The number of msg failed offline pushed", | ||||
| 	}) | ||||
| 	MsgLoneTimePushCounter = prometheus.NewCounter(prometheus.CounterOpts{ | ||||
| 		Name: "msg_long_time_push_total", | ||||
| 		Help: "The number of messages with a push time exceeding 10 seconds", | ||||
| 	}) | ||||
| ) | ||||
|  | ||||
| @ -47,9 +47,17 @@ func GetGrpcCusMetrics(registerName string, share *config.Share) []prometheus.Co | ||||
| 	case share.RpcRegisterName.MessageGateway: | ||||
| 		return []prometheus.Collector{OnlineUserGauge} | ||||
| 	case share.RpcRegisterName.Msg: | ||||
| 		return []prometheus.Collector{SingleChatMsgProcessSuccessCounter, SingleChatMsgProcessFailedCounter, GroupChatMsgProcessSuccessCounter, GroupChatMsgProcessFailedCounter} | ||||
| 		return []prometheus.Collector{ | ||||
| 			SingleChatMsgProcessSuccessCounter, | ||||
| 			SingleChatMsgProcessFailedCounter, | ||||
| 			GroupChatMsgProcessSuccessCounter, | ||||
| 			GroupChatMsgProcessFailedCounter, | ||||
| 		} | ||||
| 	case share.RpcRegisterName.Push: | ||||
| 		return []prometheus.Collector{MsgOfflinePushFailedCounter} | ||||
| 		return []prometheus.Collector{ | ||||
| 			MsgOfflinePushFailedCounter, | ||||
| 			MsgLoneTimePushCounter, | ||||
| 		} | ||||
| 	case share.RpcRegisterName.Auth: | ||||
| 		return []prometheus.Collector{UserLoginCounter} | ||||
| 	case share.RpcRegisterName.User: | ||||
|  | ||||
| @ -25,7 +25,6 @@ import ( | ||||
| 	"os" | ||||
| 	"os/signal" | ||||
| 	"strconv" | ||||
| 	"sync" | ||||
| 	"syscall" | ||||
| 	"time" | ||||
| 
 | ||||
| @ -35,7 +34,6 @@ import ( | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| 	"github.com/openimsdk/tools/log" | ||||
| 	"github.com/openimsdk/tools/mw" | ||||
| 	"github.com/openimsdk/tools/system/program" | ||||
| 	"github.com/openimsdk/tools/utils/network" | ||||
| 	"google.golang.org/grpc" | ||||
| 	"google.golang.org/grpc/credentials/insecure" | ||||
| @ -54,6 +52,7 @@ func Start[T any](ctx context.Context, discovery *config.Discovery, prometheusCo | ||||
| 	log.CInfo(ctx, "RPC server is initializing", "rpcRegisterName", rpcRegisterName, "rpcPort", rpcPort, | ||||
| 		"prometheusPorts", prometheusConfig.Ports) | ||||
| 	rpcTcpAddr := net.JoinHostPort(network.GetListenIP(listenIP), strconv.Itoa(rpcPort)) | ||||
| 
 | ||||
| 	listener, err := net.Listen( | ||||
| 		"tcp", | ||||
| 		rpcTcpAddr, | ||||
| @ -61,7 +60,6 @@ func Start[T any](ctx context.Context, discovery *config.Discovery, prometheusCo | ||||
| 	if err != nil { | ||||
| 		return errs.WrapMsg(err, "listen err", "rpcTcpAddr", rpcTcpAddr) | ||||
| 	} | ||||
| 
 | ||||
| 	defer listener.Close() | ||||
| 	client, err := kdisc.NewDiscoveryRegister(discovery, share) | ||||
| 	if err != nil { | ||||
| @ -92,10 +90,6 @@ func Start[T any](ctx context.Context, discovery *config.Discovery, prometheusCo | ||||
| 	} | ||||
| 
 | ||||
| 	srv := grpc.NewServer(options...) | ||||
| 	once := sync.Once{} | ||||
| 	defer func() { | ||||
| 		once.Do(srv.GracefulStop) | ||||
| 	}() | ||||
| 
 | ||||
| 	err = rpcFn(ctx, config, client, srv) | ||||
| 	if err != nil { | ||||
| @ -115,7 +109,6 @@ func Start[T any](ctx context.Context, discovery *config.Discovery, prometheusCo | ||||
| 	var ( | ||||
| 		netDone = make(chan struct{}, 2) | ||||
| 		netErr  error | ||||
| 		httpServer *http.Server | ||||
| 	) | ||||
| 	if prometheusConfig.Enable { | ||||
| 		go func() { | ||||
| @ -152,18 +145,11 @@ func Start[T any](ctx context.Context, discovery *config.Discovery, prometheusCo | ||||
| 	signal.Notify(sigs, syscall.SIGTERM) | ||||
| 	select { | ||||
| 	case <-sigs: | ||||
| 		program.SIGTERMExit() | ||||
| 		ctx, cancel := context.WithTimeout(context.Background(), 15*time.Second) | ||||
| 		ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) | ||||
| 		defer cancel() | ||||
| 		if err := gracefulStopWithCtx(ctx, srv.GracefulStop); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		ctx, cancel = context.WithTimeout(context.Background(), 15*time.Second) | ||||
| 		defer cancel() | ||||
| 		err := httpServer.Shutdown(ctx) | ||||
| 		if err != nil { | ||||
| 			return errs.WrapMsg(err, "shutdown err") | ||||
| 		} | ||||
| 		return nil | ||||
| 	case <-netDone: | ||||
| 		return netErr | ||||
|  | ||||
| @ -17,6 +17,8 @@ package cachekey | ||||
| const ( | ||||
| 	ConversationKey                          = "CONVERSATION:" | ||||
| 	ConversationIDsKey                       = "CONVERSATION_IDS:" | ||||
| 	NotNotifyConversationIDsKey              = "NOT_NOTIFY_CONVERSATION_IDS:" | ||||
| 	PinnedConversationIDsKey                 = "PINNED_CONVERSATION_IDS:" | ||||
| 	ConversationIDsHashKey                   = "CONVERSATION_IDS_HASH:" | ||||
| 	ConversationHasReadSeqKey                = "CONVERSATION_HAS_READ_SEQ:" | ||||
| 	RecvMsgOptKey                            = "RECV_MSG_OPT:" | ||||
| @ -34,6 +36,14 @@ func GetConversationIDsKey(ownerUserID string) string { | ||||
| 	return ConversationIDsKey + ownerUserID | ||||
| } | ||||
| 
 | ||||
| func GetNotNotifyConversationIDsKey(ownerUserID string) string { | ||||
| 	return NotNotifyConversationIDsKey + ownerUserID | ||||
| } | ||||
| 
 | ||||
| func GetPinnedConversationIDs(ownerUserID string) string { | ||||
| 	return PinnedConversationIDsKey + ownerUserID | ||||
| } | ||||
| 
 | ||||
| func GetSuperGroupRecvNotNotifyUserIDsKey(groupID string) string { | ||||
| 	return SuperGroupRecvMsgNotNotifyUserIDsKey + groupID | ||||
| } | ||||
|  | ||||
							
								
								
									
										1
									
								
								pkg/common/storage/cache/cachekey/group.go
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								pkg/common/storage/cache/cachekey/group.go
									
									
									
									
										vendored
									
									
								
							| @ -28,6 +28,7 @@ const ( | ||||
| 	JoinedGroupsKey             = "JOIN_GROUPS_KEY:" | ||||
| 	GroupMemberNumKey           = "GROUP_MEMBER_NUM_CACHE:" | ||||
| 	GroupRoleLevelMemberIDsKey  = "GROUP_ROLE_LEVEL_MEMBER_IDS:" | ||||
| 	GroupAdminLevelMemberIDsKey = "GROUP_ADMIN_LEVEL_MEMBER_IDS:" | ||||
| 	GroupMemberMaxVersionKey    = "GROUP_MEMBER_MAX_VERSION:" | ||||
| 	GroupJoinMaxVersionKey      = "GROUP_JOIN_MAX_VERSION:" | ||||
| ) | ||||
|  | ||||
							
								
								
									
										9
									
								
								pkg/common/storage/cache/cachekey/online.go
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								pkg/common/storage/cache/cachekey/online.go
									
									
									
									
										vendored
									
									
								
							| @ -1,6 +1,9 @@ | ||||
| package cachekey | ||||
| 
 | ||||
| import "time" | ||||
| import ( | ||||
| 	"strings" | ||||
| 	"time" | ||||
| ) | ||||
| 
 | ||||
| const ( | ||||
| 	OnlineKey     = "ONLINE:" | ||||
| @ -11,3 +14,7 @@ const ( | ||||
| func GetOnlineKey(userID string) string { | ||||
| 	return OnlineKey + userID | ||||
| } | ||||
| 
 | ||||
| func GetOnlineKeyUserID(key string) string { | ||||
| 	return strings.TrimPrefix(key, OnlineKey) | ||||
| } | ||||
|  | ||||
							
								
								
									
										19
									
								
								pkg/common/storage/cache/cachekey/token.go
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										19
									
								
								pkg/common/storage/cache/cachekey/token.go
									
									
									
									
										vendored
									
									
								
							| @ -1,6 +1,9 @@ | ||||
| package cachekey | ||||
| 
 | ||||
| import "github.com/openimsdk/protocol/constant" | ||||
| import ( | ||||
| 	"github.com/openimsdk/protocol/constant" | ||||
| 	"strings" | ||||
| ) | ||||
| 
 | ||||
| const ( | ||||
| 	UidPidToken = "UID_PID_TOKEN_STATUS:" | ||||
| @ -9,3 +12,17 @@ const ( | ||||
| func GetTokenKey(userID string, platformID int) string { | ||||
| 	return UidPidToken + userID + ":" + constant.PlatformIDToName(platformID) | ||||
| } | ||||
| 
 | ||||
| func GetAllPlatformTokenKey(userID string) []string { | ||||
| 	res := make([]string, len(constant.PlatformID2Name)) | ||||
| 	for k := range constant.PlatformID2Name { | ||||
| 		res[k-1] = GetTokenKey(userID, k) | ||||
| 	} | ||||
| 	return res | ||||
| } | ||||
| 
 | ||||
| func GetPlatformIDByTokenKey(key string) int { | ||||
| 	splitKey := strings.Split(key, ":") | ||||
| 	platform := splitKey[len(splitKey)-1] | ||||
| 	return constant.PlatformNameToID(platform) | ||||
| } | ||||
|  | ||||
							
								
								
									
										5
									
								
								pkg/common/storage/cache/conversation.go
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								pkg/common/storage/cache/conversation.go
									
									
									
									
										vendored
									
									
								
							| @ -25,6 +25,8 @@ type ConversationCache interface { | ||||
| 	CloneConversationCache() ConversationCache | ||||
| 	// get user's conversationIDs from msgCache | ||||
| 	GetUserConversationIDs(ctx context.Context, ownerUserID string) ([]string, error) | ||||
| 	GetUserNotNotifyConversationIDs(ctx context.Context, userID string) ([]string, error) | ||||
| 	GetPinnedConversationIDs(ctx context.Context, userID string) ([]string, error) | ||||
| 	DelConversationIDs(userIDs ...string) ConversationCache | ||||
| 
 | ||||
| 	GetUserConversationIDsHash(ctx context.Context, ownerUserID string) (hash uint64, err error) | ||||
| @ -54,7 +56,8 @@ type ConversationCache interface { | ||||
| 
 | ||||
| 	GetConversationNotReceiveMessageUserIDs(ctx context.Context, conversationID string) ([]string, error) | ||||
| 	DelConversationNotReceiveMessageUserIDs(conversationIDs ...string) ConversationCache | ||||
| 
 | ||||
| 	DelConversationNotNotifyMessageUserIDs(userIDs ...string) ConversationCache | ||||
| 	DelConversationPinnedMessageUserIDs(userIDs ...string) ConversationCache | ||||
| 	DelConversationVersionUserIDs(userIDs ...string) ConversationCache | ||||
| 
 | ||||
| 	FindMaxConversationUserVersion(ctx context.Context, userID string) (*relationtb.VersionLog, error) | ||||
|  | ||||
							
								
								
									
										1
									
								
								pkg/common/storage/cache/group.go
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								pkg/common/storage/cache/group.go
									
									
									
									
										vendored
									
									
								
							| @ -16,6 +16,7 @@ package cache | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/common" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||
| ) | ||||
|  | ||||
							
								
								
									
										1
									
								
								pkg/common/storage/cache/online.go
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								pkg/common/storage/cache/online.go
									
									
									
									
										vendored
									
									
								
							| @ -5,4 +5,5 @@ import "context" | ||||
| type OnlineCache interface { | ||||
| 	GetOnline(ctx context.Context, userID string) ([]int32, error) | ||||
| 	SetUserOnline(ctx context.Context, userID string, online, offline []int32) error | ||||
| 	GetAllOnlineUsers(ctx context.Context, cursor uint64) (map[string][]int32, uint64, error) | ||||
| } | ||||
|  | ||||
							
								
								
									
										6
									
								
								pkg/common/storage/cache/redis/batch.go
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								pkg/common/storage/cache/redis/batch.go
									
									
									
									
										vendored
									
									
								
							| @ -4,6 +4,7 @@ import ( | ||||
| 	"context" | ||||
| 	"encoding/json" | ||||
| 	"github.com/dtm-labs/rockscache" | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| 	"github.com/openimsdk/tools/log" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"golang.org/x/sync/singleflight" | ||||
| @ -65,6 +66,7 @@ func batchGetCache2[K comparable, V any](ctx context.Context, rcClient *rockscac | ||||
| 				} | ||||
| 				bs, err := json.Marshal(value) | ||||
| 				if err != nil { | ||||
| 					log.ZError(ctx, "marshal failed", err) | ||||
| 					return nil, err | ||||
| 				} | ||||
| 				cacheIndex[index] = string(bs) | ||||
| @ -72,7 +74,7 @@ func batchGetCache2[K comparable, V any](ctx context.Context, rcClient *rockscac | ||||
| 			return cacheIndex, nil | ||||
| 		}) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 			return nil, errs.WrapMsg(err, "FetchBatch2 failed") | ||||
| 		} | ||||
| 		for index, data := range indexCache { | ||||
| 			if data == "" { | ||||
| @ -80,7 +82,7 @@ func batchGetCache2[K comparable, V any](ctx context.Context, rcClient *rockscac | ||||
| 			} | ||||
| 			var value V | ||||
| 			if err := json.Unmarshal([]byte(data), &value); err != nil { | ||||
| 				return nil, err | ||||
| 				return nil, errs.WrapMsg(err, "Unmarshal failed") | ||||
| 			} | ||||
| 			if cb, ok := any(&value).(BatchCacheCallback[K]); ok { | ||||
| 				cb.BatchCache(keyId[keys[index]]) | ||||
|  | ||||
| @ -28,6 +28,10 @@ import ( | ||||
| 	"time" | ||||
| ) | ||||
| 
 | ||||
| const ( | ||||
| 	rocksCacheTimeout = 11 * time.Second | ||||
| ) | ||||
| 
 | ||||
| // BatchDeleterRedis is a concrete implementation of the BatchDeleter interface based on Redis and RocksCache. | ||||
| type BatchDeleterRedis struct { | ||||
| 	redisClient    redis.UniversalClient | ||||
| @ -106,6 +110,8 @@ func (c *BatchDeleterRedis) AddKeys(keys ...string) { | ||||
| // GetRocksCacheOptions returns the default configuration options for RocksCache. | ||||
| func GetRocksCacheOptions() *rockscache.Options { | ||||
| 	opts := rockscache.NewDefaultOptions() | ||||
| 	opts.LockExpire = rocksCacheTimeout | ||||
| 	opts.WaitReplicasTimeout = rocksCacheTimeout | ||||
| 	opts.StrongConsistency = true | ||||
| 	opts.RandomExpireAdjustment = 0.2 | ||||
| 
 | ||||
| @ -118,7 +124,7 @@ func getCache[T any](ctx context.Context, rcClient *rockscache.Client, key strin | ||||
| 	v, err := rcClient.Fetch2(ctx, key, expire, func() (s string, err error) { | ||||
| 		t, err = fn(ctx) | ||||
| 		if err != nil { | ||||
| 			log.ZError(ctx, "getCache query database failed", err, "key", key) | ||||
| 			//log.ZError(ctx, "getCache query database failed", err, "key", key) | ||||
| 			return "", err | ||||
| 		} | ||||
| 		bs, err := json.Marshal(t) | ||||
|  | ||||
							
								
								
									
										36
									
								
								pkg/common/storage/cache/redis/conversation.go
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										36
									
								
								pkg/common/storage/cache/redis/conversation.go
									
									
									
									
										vendored
									
									
								
							| @ -71,6 +71,14 @@ func (c *ConversationRedisCache) getConversationIDsKey(ownerUserID string) strin | ||||
| 	return cachekey.GetConversationIDsKey(ownerUserID) | ||||
| } | ||||
| 
 | ||||
| func (c *ConversationRedisCache) getNotNotifyConversationIDsKey(ownerUserID string) string { | ||||
| 	return cachekey.GetNotNotifyConversationIDsKey(ownerUserID) | ||||
| } | ||||
| 
 | ||||
| func (c *ConversationRedisCache) getPinnedConversationIDsKey(ownerUserID string) string { | ||||
| 	return cachekey.GetPinnedConversationIDs(ownerUserID) | ||||
| } | ||||
| 
 | ||||
| func (c *ConversationRedisCache) getSuperGroupRecvNotNotifyUserIDsKey(groupID string) string { | ||||
| 	return cachekey.GetSuperGroupRecvNotNotifyUserIDsKey(groupID) | ||||
| } | ||||
| @ -105,6 +113,18 @@ func (c *ConversationRedisCache) GetUserConversationIDs(ctx context.Context, own | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func (c *ConversationRedisCache) GetUserNotNotifyConversationIDs(ctx context.Context, userID string) ([]string, error) { | ||||
| 	return getCache(ctx, c.rcClient, c.getNotNotifyConversationIDsKey(userID), c.expireTime, func(ctx context.Context) ([]string, error) { | ||||
| 		return c.conversationDB.FindUserIDAllNotNotifyConversationID(ctx, userID) | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func (c *ConversationRedisCache) GetPinnedConversationIDs(ctx context.Context, userID string) ([]string, error) { | ||||
| 	return getCache(ctx, c.rcClient, c.getPinnedConversationIDsKey(userID), c.expireTime, func(ctx context.Context) ([]string, error) { | ||||
| 		return c.conversationDB.FindUserIDAllPinnedConversationID(ctx, userID) | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func (c *ConversationRedisCache) DelConversationIDs(userIDs ...string) cache.ConversationCache { | ||||
| 	keys := make([]string, 0, len(userIDs)) | ||||
| 	for _, userID := range userIDs { | ||||
| @ -242,6 +262,22 @@ func (c *ConversationRedisCache) DelConversationNotReceiveMessageUserIDs(convers | ||||
| 	return cache | ||||
| } | ||||
| 
 | ||||
| func (c *ConversationRedisCache) DelConversationNotNotifyMessageUserIDs(userIDs ...string) cache.ConversationCache { | ||||
| 	cache := c.CloneConversationCache() | ||||
| 	for _, userID := range userIDs { | ||||
| 		cache.AddKeys(c.getNotNotifyConversationIDsKey(userID)) | ||||
| 	} | ||||
| 	return cache | ||||
| } | ||||
| 
 | ||||
| func (c *ConversationRedisCache) DelConversationPinnedMessageUserIDs(userIDs ...string) cache.ConversationCache { | ||||
| 	cache := c.CloneConversationCache() | ||||
| 	for _, userID := range userIDs { | ||||
| 		cache.AddKeys(c.getPinnedConversationIDsKey(userID)) | ||||
| 	} | ||||
| 	return cache | ||||
| } | ||||
| 
 | ||||
| func (c *ConversationRedisCache) DelConversationVersionUserIDs(userIDs ...string) cache.ConversationCache { | ||||
| 	cache := c.CloneConversationCache() | ||||
| 	for _, userID := range userIDs { | ||||
|  | ||||
							
								
								
									
										32
									
								
								pkg/common/storage/cache/redis/online.go
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										32
									
								
								pkg/common/storage/cache/redis/online.go
									
									
									
									
										vendored
									
									
								
							| @ -2,8 +2,10 @@ package redis | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"fmt" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache/cachekey" | ||||
| 	"github.com/openimsdk/protocol/constant" | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| 	"github.com/openimsdk/tools/log" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| @ -49,6 +51,36 @@ func (s *userOnline) GetOnline(ctx context.Context, userID string) ([]int32, err | ||||
| 	return platformIDs, nil | ||||
| } | ||||
| 
 | ||||
| func (s *userOnline) GetAllOnlineUsers(ctx context.Context, cursor uint64) (map[string][]int32, uint64, error) { | ||||
| 	result := make(map[string][]int32) | ||||
| 
 | ||||
| 	keys, nextCursor, err := s.rdb.Scan(ctx, cursor, fmt.Sprintf("%s*", cachekey.OnlineKey), constant.ParamMaxLength).Result() | ||||
| 	if err != nil { | ||||
| 		return nil, 0, err | ||||
| 	} | ||||
| 
 | ||||
| 	for _, key := range keys { | ||||
| 		userID := cachekey.GetOnlineKeyUserID(key) | ||||
| 		strValues, err := s.rdb.ZRange(ctx, key, 0, -1).Result() | ||||
| 		if err != nil { | ||||
| 			return nil, 0, err | ||||
| 		} | ||||
| 
 | ||||
| 		values := make([]int32, 0, len(strValues)) | ||||
| 		for _, value := range strValues { | ||||
| 			intValue, err := strconv.Atoi(value) | ||||
| 			if err != nil { | ||||
| 				return nil, 0, errs.Wrap(err) | ||||
| 			} | ||||
| 			values = append(values, int32(intValue)) | ||||
| 		} | ||||
| 
 | ||||
| 		result[userID] = values | ||||
| 	} | ||||
| 
 | ||||
| 	return result, nextCursor, nil | ||||
| } | ||||
| 
 | ||||
| func (s *userOnline) SetUserOnline(ctx context.Context, userID string, online, offline []int32) error { | ||||
| 	script := ` | ||||
| 	local key = KEYS[1] | ||||
|  | ||||
							
								
								
									
										228
									
								
								pkg/common/storage/cache/redis/seq_conversation.go
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										228
									
								
								pkg/common/storage/cache/redis/seq_conversation.go
									
									
									
									
										vendored
									
									
								
							| @ -12,6 +12,7 @@ import ( | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| 	"github.com/openimsdk/tools/log" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"strconv" | ||||
| 	"time" | ||||
| ) | ||||
| 
 | ||||
| @ -57,6 +58,14 @@ func (s *seqConversationCacheRedis) getSingleMaxSeq(ctx context.Context, convers | ||||
| 	return map[string]int64{conversationID: seq}, nil | ||||
| } | ||||
| 
 | ||||
| func (s *seqConversationCacheRedis) getSingleMaxSeqWithTime(ctx context.Context, conversationID string) (map[string]database.SeqTime, error) { | ||||
| 	seq, err := s.GetMaxSeqWithTime(ctx, conversationID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return map[string]database.SeqTime{conversationID: seq}, nil | ||||
| } | ||||
| 
 | ||||
| func (s *seqConversationCacheRedis) batchGetMaxSeq(ctx context.Context, keys []string, keyConversationID map[string]string, seqs map[string]int64) error { | ||||
| 	result := make([]*redis.StringCmd, len(keys)) | ||||
| 	pipe := s.rdb.Pipeline() | ||||
| @ -88,6 +97,46 @@ func (s *seqConversationCacheRedis) batchGetMaxSeq(ctx context.Context, keys []s | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func (s *seqConversationCacheRedis) batchGetMaxSeqWithTime(ctx context.Context, keys []string, keyConversationID map[string]string, seqs map[string]database.SeqTime) error { | ||||
| 	result := make([]*redis.SliceCmd, len(keys)) | ||||
| 	pipe := s.rdb.Pipeline() | ||||
| 	for i, key := range keys { | ||||
| 		result[i] = pipe.HMGet(ctx, key, "CURR", "TIME") | ||||
| 	} | ||||
| 	if _, err := pipe.Exec(ctx); err != nil && !errors.Is(err, redis.Nil) { | ||||
| 		return errs.Wrap(err) | ||||
| 	} | ||||
| 	var notFoundKey []string | ||||
| 	for i, r := range result { | ||||
| 		val, err := r.Result() | ||||
| 		if len(val) != 2 { | ||||
| 			return errs.WrapMsg(err, "batchGetMaxSeqWithTime invalid result", "key", keys[i], "res", val) | ||||
| 		} | ||||
| 		if val[0] == nil { | ||||
| 			notFoundKey = append(notFoundKey, keys[i]) | ||||
| 			continue | ||||
| 		} | ||||
| 		seq, err := s.parseInt64(val[0]) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		mill, err := s.parseInt64(val[1]) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		seqs[keyConversationID[keys[i]]] = database.SeqTime{Seq: seq, Time: mill} | ||||
| 	} | ||||
| 	for _, key := range notFoundKey { | ||||
| 		conversationID := keyConversationID[key] | ||||
| 		seq, err := s.GetMaxSeqWithTime(ctx, conversationID) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		seqs[conversationID] = seq | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func (s *seqConversationCacheRedis) GetMaxSeqs(ctx context.Context, conversationIDs []string) (map[string]int64, error) { | ||||
| 	switch len(conversationIDs) { | ||||
| 	case 0: | ||||
| @ -121,11 +170,44 @@ func (s *seqConversationCacheRedis) GetMaxSeqs(ctx context.Context, conversation | ||||
| 	return seqs, nil | ||||
| } | ||||
| 
 | ||||
| func (s *seqConversationCacheRedis) GetMaxSeqsWithTime(ctx context.Context, conversationIDs []string) (map[string]database.SeqTime, error) { | ||||
| 	switch len(conversationIDs) { | ||||
| 	case 0: | ||||
| 		return map[string]database.SeqTime{}, nil | ||||
| 	case 1: | ||||
| 		return s.getSingleMaxSeqWithTime(ctx, conversationIDs[0]) | ||||
| 	} | ||||
| 	keys := make([]string, 0, len(conversationIDs)) | ||||
| 	keyConversationID := make(map[string]string, len(conversationIDs)) | ||||
| 	for _, conversationID := range conversationIDs { | ||||
| 		key := s.getSeqMallocKey(conversationID) | ||||
| 		if _, ok := keyConversationID[key]; ok { | ||||
| 			continue | ||||
| 		} | ||||
| 		keys = append(keys, key) | ||||
| 		keyConversationID[key] = conversationID | ||||
| 	} | ||||
| 	if len(keys) == 1 { | ||||
| 		return s.getSingleMaxSeqWithTime(ctx, conversationIDs[0]) | ||||
| 	} | ||||
| 	slotKeys, err := groupKeysBySlot(ctx, s.rdb, keys) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	seqs := make(map[string]database.SeqTime, len(conversationIDs)) | ||||
| 	for _, keys := range slotKeys { | ||||
| 		if err := s.batchGetMaxSeqWithTime(ctx, keys, keyConversationID, seqs); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	} | ||||
| 	return seqs, nil | ||||
| } | ||||
| 
 | ||||
| func (s *seqConversationCacheRedis) getSeqMallocKey(conversationID string) string { | ||||
| 	return cachekey.GetMallocSeqKey(conversationID) | ||||
| } | ||||
| 
 | ||||
| func (s *seqConversationCacheRedis) setSeq(ctx context.Context, key string, owner int64, currSeq int64, lastSeq int64) (int64, error) { | ||||
| func (s *seqConversationCacheRedis) setSeq(ctx context.Context, key string, owner int64, currSeq int64, lastSeq int64, mill int64) (int64, error) { | ||||
| 	if lastSeq < currSeq { | ||||
| 		return 0, errs.New("lastSeq must be greater than currSeq") | ||||
| 	} | ||||
| @ -138,8 +220,9 @@ local lockValue = ARGV[1] | ||||
| local dataSecond = ARGV[2] | ||||
| local curr_seq = tonumber(ARGV[3]) | ||||
| local last_seq = tonumber(ARGV[4]) | ||||
| local mallocTime = ARGV[5] | ||||
| if redis.call("EXISTS", key) == 0 then | ||||
| 	redis.call("HSET", key, "CURR", curr_seq, "LAST", last_seq) | ||||
| 	redis.call("HSET", key, "CURR", curr_seq, "LAST", last_seq, "TIME", mallocTime) | ||||
| 	redis.call("EXPIRE", key, dataSecond) | ||||
| 	return 1 | ||||
| end | ||||
| @ -147,11 +230,11 @@ if redis.call("HGET", key, "LOCK") ~= lockValue then | ||||
| 	return 2 | ||||
| end | ||||
| redis.call("HDEL", key, "LOCK") | ||||
| redis.call("HSET", key, "CURR", curr_seq, "LAST", last_seq) | ||||
| redis.call("HSET", key, "CURR", curr_seq, "LAST", last_seq, "TIME", mallocTime) | ||||
| redis.call("EXPIRE", key, dataSecond) | ||||
| return 0 | ||||
| ` | ||||
| 	result, err := s.rdb.Eval(ctx, script, []string{key}, owner, int64(s.dataTime/time.Second), currSeq, lastSeq).Int64() | ||||
| 	result, err := s.rdb.Eval(ctx, script, []string{key}, owner, int64(s.dataTime/time.Second), currSeq, lastSeq, mill).Int64() | ||||
| 	if err != nil { | ||||
| 		return 0, errs.Wrap(err) | ||||
| 	} | ||||
| @ -169,6 +252,7 @@ local key = KEYS[1] | ||||
| local size = tonumber(ARGV[1]) | ||||
| local lockSecond = ARGV[2] | ||||
| local dataSecond = ARGV[3] | ||||
| local mallocTime = ARGV[4] | ||||
| local result = {} | ||||
| if redis.call("EXISTS", key) == 0 then | ||||
| 	local lockValue = math.random(0, 999999999) | ||||
| @ -176,6 +260,7 @@ if redis.call("EXISTS", key) == 0 then | ||||
| 	redis.call("EXPIRE", key, lockSecond) | ||||
| 	table.insert(result, 1) | ||||
| 	table.insert(result, lockValue) | ||||
| 	table.insert(result, mallocTime) | ||||
| 	return result | ||||
| end | ||||
| if redis.call("HEXISTS", key, "LOCK") == 1 then | ||||
| @ -189,6 +274,12 @@ if size == 0 then | ||||
| 	table.insert(result, 0) | ||||
| 	table.insert(result, curr_seq) | ||||
| 	table.insert(result, last_seq) | ||||
| 	local setTime = redis.call("HGET", key, "TIME") | ||||
| 	if setTime then | ||||
| 		table.insert(result, setTime)	 | ||||
| 	else | ||||
| 		table.insert(result, 0) | ||||
| 	end | ||||
| 	return result | ||||
| end | ||||
| local max_seq = curr_seq + size | ||||
| @ -196,21 +287,25 @@ if max_seq > last_seq then | ||||
| 	local lockValue = math.random(0, 999999999) | ||||
| 	redis.call("HSET", key, "LOCK", lockValue) | ||||
| 	redis.call("HSET", key, "CURR", last_seq) | ||||
| 	redis.call("HSET", key, "TIME", mallocTime) | ||||
| 	redis.call("EXPIRE", key, lockSecond) | ||||
| 	table.insert(result, 3) | ||||
| 	table.insert(result, curr_seq) | ||||
| 	table.insert(result, last_seq) | ||||
| 	table.insert(result, lockValue) | ||||
| 	table.insert(result, mallocTime) | ||||
| 	return result | ||||
| end | ||||
| redis.call("HSET", key, "CURR", max_seq) | ||||
| redis.call("HSET", key, "TIME", ARGV[4]) | ||||
| redis.call("EXPIRE", key, dataSecond) | ||||
| table.insert(result, 0) | ||||
| table.insert(result, curr_seq) | ||||
| table.insert(result, last_seq) | ||||
| table.insert(result, mallocTime) | ||||
| return result | ||||
| ` | ||||
| 	result, err := s.rdb.Eval(ctx, script, []string{key}, size, int64(s.lockTime/time.Second), int64(s.dataTime/time.Second)).Int64Slice() | ||||
| 	result, err := s.rdb.Eval(ctx, script, []string{key}, size, int64(s.lockTime/time.Second), int64(s.dataTime/time.Second), time.Now().UnixMilli()).Int64Slice() | ||||
| 	if err != nil { | ||||
| 		return nil, errs.Wrap(err) | ||||
| 	} | ||||
| @ -228,9 +323,9 @@ func (s *seqConversationCacheRedis) wait(ctx context.Context) error { | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func (s *seqConversationCacheRedis) setSeqRetry(ctx context.Context, key string, owner int64, currSeq int64, lastSeq int64) { | ||||
| func (s *seqConversationCacheRedis) setSeqRetry(ctx context.Context, key string, owner int64, currSeq int64, lastSeq int64, mill int64) { | ||||
| 	for i := 0; i < 10; i++ { | ||||
| 		state, err := s.setSeq(ctx, key, owner, currSeq, lastSeq) | ||||
| 		state, err := s.setSeq(ctx, key, owner, currSeq, lastSeq, mill) | ||||
| 		if err != nil { | ||||
| 			log.ZError(ctx, "set seq cache failed", err, "key", key, "owner", owner, "currSeq", currSeq, "lastSeq", lastSeq, "count", i+1) | ||||
| 			if err := s.wait(ctx); err != nil { | ||||
| @ -267,60 +362,74 @@ func (s *seqConversationCacheRedis) getMallocSize(conversationID string, size in | ||||
| } | ||||
| 
 | ||||
| func (s *seqConversationCacheRedis) Malloc(ctx context.Context, conversationID string, size int64) (int64, error) { | ||||
| 	seq, _, err := s.mallocTime(ctx, conversationID, size) | ||||
| 	return seq, err | ||||
| } | ||||
| 
 | ||||
| func (s *seqConversationCacheRedis) mallocTime(ctx context.Context, conversationID string, size int64) (int64, int64, error) { | ||||
| 	if size < 0 { | ||||
| 		return 0, errs.New("size must be greater than 0") | ||||
| 		return 0, 0, errs.New("size must be greater than 0") | ||||
| 	} | ||||
| 	key := s.getSeqMallocKey(conversationID) | ||||
| 	for i := 0; i < 10; i++ { | ||||
| 		states, err := s.malloc(ctx, key, size) | ||||
| 		if err != nil { | ||||
| 			return 0, err | ||||
| 			return 0, 0, err | ||||
| 		} | ||||
| 		switch states[0] { | ||||
| 		case 0: // success | ||||
| 			return states[1], nil | ||||
| 			return states[1], states[3], nil | ||||
| 		case 1: // not found | ||||
| 			mallocSize := s.getMallocSize(conversationID, size) | ||||
| 			seq, err := s.mgo.Malloc(ctx, conversationID, mallocSize) | ||||
| 			if err != nil { | ||||
| 				return 0, err | ||||
| 				return 0, 0, err | ||||
| 			} | ||||
| 			s.setSeqRetry(ctx, key, states[1], seq+size, seq+mallocSize) | ||||
| 			return seq, nil | ||||
| 			s.setSeqRetry(ctx, key, states[1], seq+size, seq+mallocSize, states[2]) | ||||
| 			return seq, 0, nil | ||||
| 		case 2: // locked | ||||
| 			if err := s.wait(ctx); err != nil { | ||||
| 				return 0, err | ||||
| 				return 0, 0, err | ||||
| 			} | ||||
| 			continue | ||||
| 		case 3: // exceeded cache max value | ||||
| 			currSeq := states[1] | ||||
| 			lastSeq := states[2] | ||||
| 			mill := states[4] | ||||
| 			mallocSize := s.getMallocSize(conversationID, size) | ||||
| 			seq, err := s.mgo.Malloc(ctx, conversationID, mallocSize) | ||||
| 			if err != nil { | ||||
| 				return 0, err | ||||
| 				return 0, 0, err | ||||
| 			} | ||||
| 			if lastSeq == seq { | ||||
| 				s.setSeqRetry(ctx, key, states[3], currSeq+size, seq+mallocSize) | ||||
| 				return currSeq, nil | ||||
| 				s.setSeqRetry(ctx, key, states[3], currSeq+size, seq+mallocSize, mill) | ||||
| 				return currSeq, states[4], nil | ||||
| 			} else { | ||||
| 				log.ZWarn(ctx, "malloc seq not equal cache last seq", nil, "conversationID", conversationID, "currSeq", currSeq, "lastSeq", lastSeq, "mallocSeq", seq) | ||||
| 				s.setSeqRetry(ctx, key, states[3], seq+size, seq+mallocSize) | ||||
| 				return seq, nil | ||||
| 				s.setSeqRetry(ctx, key, states[3], seq+size, seq+mallocSize, mill) | ||||
| 				return seq, mill, nil | ||||
| 			} | ||||
| 		default: | ||||
| 			log.ZError(ctx, "malloc seq unknown state", nil, "state", states[0], "conversationID", conversationID, "size", size) | ||||
| 			return 0, errs.New(fmt.Sprintf("unknown state: %d", states[0])) | ||||
| 			return 0, 0, errs.New(fmt.Sprintf("unknown state: %d", states[0])) | ||||
| 		} | ||||
| 	} | ||||
| 	log.ZError(ctx, "malloc seq retrying still failed", nil, "conversationID", conversationID, "size", size) | ||||
| 	return 0, errs.New("malloc seq waiting for lock timeout", "conversationID", conversationID, "size", size) | ||||
| 	return 0, 0, errs.New("malloc seq waiting for lock timeout", "conversationID", conversationID, "size", size) | ||||
| } | ||||
| 
 | ||||
| func (s *seqConversationCacheRedis) GetMaxSeq(ctx context.Context, conversationID string) (int64, error) { | ||||
| 	return s.Malloc(ctx, conversationID, 0) | ||||
| } | ||||
| 
 | ||||
| func (s *seqConversationCacheRedis) GetMaxSeqWithTime(ctx context.Context, conversationID string) (database.SeqTime, error) { | ||||
| 	seq, mill, err := s.mallocTime(ctx, conversationID, 0) | ||||
| 	if err != nil { | ||||
| 		return database.SeqTime{}, err | ||||
| 	} | ||||
| 	return database.SeqTime{Seq: seq, Time: mill}, nil | ||||
| } | ||||
| 
 | ||||
| func (s *seqConversationCacheRedis) SetMinSeqs(ctx context.Context, seqs map[string]int64) error { | ||||
| 	keys := make([]string, 0, len(seqs)) | ||||
| 	for conversationID, seq := range seqs { | ||||
| @ -331,3 +440,80 @@ func (s *seqConversationCacheRedis) SetMinSeqs(ctx context.Context, seqs map[str | ||||
| 	} | ||||
| 	return DeleteCacheBySlot(ctx, s.rocks, keys) | ||||
| } | ||||
| 
 | ||||
| // GetCacheMaxSeqWithTime only get the existing cache, if there is no cache, no cache will be generated | ||||
| func (s *seqConversationCacheRedis) GetCacheMaxSeqWithTime(ctx context.Context, conversationIDs []string) (map[string]database.SeqTime, error) { | ||||
| 	if len(conversationIDs) == 0 { | ||||
| 		return map[string]database.SeqTime{}, nil | ||||
| 	} | ||||
| 	key2conversationID := make(map[string]string) | ||||
| 	keys := make([]string, 0, len(conversationIDs)) | ||||
| 	for _, conversationID := range conversationIDs { | ||||
| 		key := s.getSeqMallocKey(conversationID) | ||||
| 		if _, ok := key2conversationID[key]; ok { | ||||
| 			continue | ||||
| 		} | ||||
| 		key2conversationID[key] = conversationID | ||||
| 		keys = append(keys, key) | ||||
| 	} | ||||
| 	slotKeys, err := groupKeysBySlot(ctx, s.rdb, keys) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	res := make(map[string]database.SeqTime) | ||||
| 	for _, keys := range slotKeys { | ||||
| 		if len(keys) == 0 { | ||||
| 			continue | ||||
| 		} | ||||
| 		pipe := s.rdb.Pipeline() | ||||
| 		cmds := make([]*redis.SliceCmd, 0, len(keys)) | ||||
| 		for _, key := range keys { | ||||
| 			cmds = append(cmds, pipe.HMGet(ctx, key, "CURR", "TIME")) | ||||
| 		} | ||||
| 		if _, err := pipe.Exec(ctx); err != nil { | ||||
| 			return nil, errs.Wrap(err) | ||||
| 		} | ||||
| 		for i, cmd := range cmds { | ||||
| 			val, err := cmd.Result() | ||||
| 			if err != nil { | ||||
| 				return nil, err | ||||
| 			} | ||||
| 			if len(val) != 2 { | ||||
| 				return nil, errs.WrapMsg(err, "GetCacheMaxSeqWithTime invalid result", "key", keys[i], "res", val) | ||||
| 			} | ||||
| 			if val[0] == nil { | ||||
| 				continue | ||||
| 			} | ||||
| 			seq, err := s.parseInt64(val[0]) | ||||
| 			if err != nil { | ||||
| 				return nil, err | ||||
| 			} | ||||
| 			mill, err := s.parseInt64(val[1]) | ||||
| 			if err != nil { | ||||
| 				return nil, err | ||||
| 			} | ||||
| 			conversationID := key2conversationID[keys[i]] | ||||
| 			res[conversationID] = database.SeqTime{Seq: seq, Time: mill} | ||||
| 		} | ||||
| 	} | ||||
| 	return res, nil | ||||
| } | ||||
| 
 | ||||
| func (s *seqConversationCacheRedis) parseInt64(val any) (int64, error) { | ||||
| 	switch v := val.(type) { | ||||
| 	case nil: | ||||
| 		return 0, nil | ||||
| 	case int: | ||||
| 		return int64(v), nil | ||||
| 	case int64: | ||||
| 		return v, nil | ||||
| 	case string: | ||||
| 		res, err := strconv.ParseInt(v, 10, 64) | ||||
| 		if err != nil { | ||||
| 			return 0, errs.WrapMsg(err, "invalid string not int64", "value", v) | ||||
| 		} | ||||
| 		return res, nil | ||||
| 	default: | ||||
| 		return 0, errs.New("invalid result not int64", "resType", fmt.Sprintf("%T", v), "value", v) | ||||
| 	} | ||||
| } | ||||
|  | ||||
Some files were not shown because too many files have changed in this diff Show More
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user