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 | ALERTMANAGER_IMAGE=prom/alertmanager:v0.27.0 | ||||||
| GRAFANA_IMAGE=grafana/grafana:11.0.1 | GRAFANA_IMAGE=grafana/grafana:11.0.1 | ||||||
| 
 | 
 | ||||||
| OPENIM_WEB_FRONT_IMAGE=openim/openim-web-front:release-v3.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.0 | OPENIM_ADMIN_FRONT_IMAGE=openim/openim-admin-front:release-v1.8.2 | ||||||
| 
 | 
 | ||||||
| #FRONT_IMAGE: use aliyun images | #FRONT_IMAGE: use aliyun images | ||||||
| #OPENIM_WEB_FRONT_IMAGE=registry.cn-hangzhou.aliyuncs.com/openimsdk/openim-web-front:release-v3.5.1 | #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.7 | #OPENIM_ADMIN_FRONT_IMAGE=registry.cn-hangzhou.aliyuncs.com/openimsdk/openim-admin-front:release-v1.8.2 | ||||||
| 
 | 
 | ||||||
| DATA_DIR=./ | 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 start | ||||||
|           mage check |           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: |   dockerfile-test: | ||||||
|     name: Build and Test Dockerfile |     name: Build and Test Dockerfile | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|  | |||||||
							
								
								
									
										83
									
								
								.github/workflows/publish-docker-image.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										83
									
								
								.github/workflows/publish-docker-image.yml
									
									
									
									
										vendored
									
									
								
							| @ -4,6 +4,8 @@ on: | |||||||
|   push: |   push: | ||||||
|     branches: |     branches: | ||||||
|       - release-* |       - release-* | ||||||
|  |     # tags: | ||||||
|  |     #   - 'v*' | ||||||
| 
 | 
 | ||||||
|   release: |   release: | ||||||
|     types: [published] |     types: [published] | ||||||
| @ -15,11 +17,8 @@ on: | |||||||
|         required: true |         required: true | ||||||
|         default: "v3.8.0" |         default: "v3.8.0" | ||||||
| 
 | 
 | ||||||
| # env: |  | ||||||
| #   GO_VERSION: "1.21" |  | ||||||
| 
 |  | ||||||
| jobs: | jobs: | ||||||
|   publish-docker-images: |   build-and-test: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v4 | ||||||
| @ -28,16 +27,22 @@ jobs: | |||||||
| 
 | 
 | ||||||
|       - name: Set up QEMU |       - name: Set up QEMU | ||||||
|         uses: docker/setup-qemu-action@v3 |         uses: docker/setup-qemu-action@v3 | ||||||
|          | 
 | ||||||
|       - name: Set up Docker Buildx |       - name: Set up Docker Buildx | ||||||
|         uses: docker/setup-buildx-action@v3 |         uses: docker/setup-buildx-action@v3 | ||||||
| 
 | 
 | ||||||
|       - name: Build and push Docker image |       - name: Build Docker image | ||||||
|  |         id: build | ||||||
|         uses: docker/build-push-action@v5 |         uses: docker/build-push-action@v5 | ||||||
|         with: |         with: | ||||||
|           context: ./main-repo |           context: ./main-repo | ||||||
|           load: true |           load: true | ||||||
|           tags: "openim/openim-server:local" |           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 |       - name: Checkout compose repository | ||||||
|         uses: actions/checkout@v4 |         uses: actions/checkout@v4 | ||||||
| @ -63,38 +68,39 @@ jobs: | |||||||
|           docker compose up -d |           docker compose up -d | ||||||
|           sleep 60 |           sleep 60 | ||||||
| 
 | 
 | ||||||
|       - name: Check openim-server health |       # - name: Check openim-server health | ||||||
|         run: | |       #   run: | | ||||||
|           timeout=300 |       #     timeout=300 | ||||||
|           interval=30 |       #     interval=30 | ||||||
|           elapsed=0 |       #     elapsed=0 | ||||||
|           while [[ $elapsed -le $timeout ]]; do |       #     while [[ $elapsed -le $timeout ]]; do | ||||||
|             if ! docker exec openim-server mage check; then |       #       if ! docker exec openim-server mage check; then | ||||||
|               echo "openim-server is not ready, waiting..." |       #         echo "openim-server is not ready, waiting..." | ||||||
|               sleep $interval |       #         sleep $interval | ||||||
|               elapsed=$(($elapsed + $interval)) |       #         elapsed=$(($elapsed + $interval)) | ||||||
|             else |       #       else | ||||||
|               echo "Health check successful" |       #         echo "Health check successful" | ||||||
|               exit 0 |       #         exit 0 | ||||||
|             fi |       #       fi | ||||||
|           done |       #     done | ||||||
|           echo "Health check failed after 5 minutes" |       #     echo "Health check failed after 5 minutes" | ||||||
|           exit 1 |       #     exit 1 | ||||||
| 
 | 
 | ||||||
|       - name: Check openim-chat health |       # - name: Check openim-chat health | ||||||
|         if: success() |       #   if: success() | ||||||
|         run: | |       #   run: | | ||||||
|           if ! docker exec openim-chat mage check; then |       #     if ! docker exec openim-chat mage check; then | ||||||
|               echo "openim-chat check failed" |       #         echo "openim-chat check failed" | ||||||
|               exit 1 |       #         exit 1 | ||||||
|             else |       #       else | ||||||
|               echo "Health check successful" |       #         echo "Health check successful" | ||||||
|               exit 0 |       #         exit 0 | ||||||
|             fi |       #       fi | ||||||
| 
 | 
 | ||||||
|  |       - name: Load Docker image from file | ||||||
|  |         run: docker load -i image.tar | ||||||
| 
 | 
 | ||||||
|       - name: Extract metadata for Docker #  (tags, labels)  |       - name: Extract metadata for Docker (tags, labels) | ||||||
|         if: success() |  | ||||||
|         id: meta |         id: meta | ||||||
|         uses: docker/metadata-action@v5.5.1 |         uses: docker/metadata-action@v5.5.1 | ||||||
|         with: |         with: | ||||||
| @ -102,18 +108,17 @@ jobs: | |||||||
|             openim/openim-server |             openim/openim-server | ||||||
|             ghcr.io/openimsdk/openim-server |             ghcr.io/openimsdk/openim-server | ||||||
|             registry.cn-hangzhou.aliyuncs.com/openimsdk/openim-server |             registry.cn-hangzhou.aliyuncs.com/openimsdk/openim-server | ||||||
| 
 |  | ||||||
|           # generate Docker tags based on the following events/attributes |  | ||||||
|           tags: | |           tags: | | ||||||
|             type=ref,event=tag |             type=ref,event=tag | ||||||
|             type=schedule |             type=schedule | ||||||
|             type=ref,event=branch |             type=ref,event=branch | ||||||
|             type=ref,event=pr |  | ||||||
|             type=semver,pattern={{version}} |             type=semver,pattern={{version}} | ||||||
|             type=semver,pattern=v{{version}} |             type=semver,pattern=v{{version}} | ||||||
|             type=semver,pattern={{major}}.{{minor}} |             type=semver,pattern={{major}}.{{minor}} | ||||||
|             type=semver,pattern={{major}} |             type=semver,pattern={{major}} | ||||||
|  |             type=semver,pattern=release-{{raw}} | ||||||
|             type=sha |             type=sha | ||||||
|  |             type=raw,value=${{ github.event.inputs.tag }} | ||||||
| 
 | 
 | ||||||
|       - name: Log in to Docker Hub |       - name: Log in to Docker Hub | ||||||
|         uses: docker/login-action@v2 |         uses: docker/login-action@v2 | ||||||
| @ -135,7 +140,7 @@ jobs: | |||||||
|           username: ${{ secrets.ALIREGISTRY_USERNAME }} |           username: ${{ secrets.ALIREGISTRY_USERNAME }} | ||||||
|           password: ${{ secrets.ALIREGISTRY_TOKEN }} |           password: ${{ secrets.ALIREGISTRY_TOKEN }} | ||||||
| 
 | 
 | ||||||
|       - name: Build and push Docker images |       - name: Push Docker images | ||||||
|         uses: docker/build-push-action@v5 |         uses: docker/build-push-action@v5 | ||||||
|         with: |         with: | ||||||
|           context: ./main-repo |           context: ./main-repo | ||||||
|  | |||||||
| @ -1,5 +1,5 @@ | |||||||
| # Use Go 1.21 Alpine as the base image for building the application | # 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 | # Define the base directory for the application as an environment variable | ||||||
| ENV SERVER_DIR=/openim-server | ENV SERVER_DIR=/openim-server | ||||||
|  | |||||||
| @ -15,10 +15,9 @@ | |||||||
| package main | package main | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	_ "net/http/pprof" |  | ||||||
| 
 |  | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/cmd" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/cmd" | ||||||
| 	"github.com/openimsdk/tools/system/program" | 	"github.com/openimsdk/tools/system/program" | ||||||
|  | 	_ "net/http/pprof" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func main() { | func main() { | ||||||
|  | |||||||
| @ -5,9 +5,4 @@ etcd: | |||||||
|   username: '' |   username: '' | ||||||
|   password: '' |   password: '' | ||||||
| 
 | 
 | ||||||
| zookeeper: |  | ||||||
|   schema: openim |  | ||||||
|   address: [ localhost:12181 ] |  | ||||||
|   username: '' |  | ||||||
|   password: '' |  | ||||||
| 
 | 
 | ||||||
|  | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @ -14,12 +14,16 @@ toRedisTopic: toRedis | |||||||
| toMongoTopic: toMongo | toMongoTopic: toMongo | ||||||
| # Kafka topic for push notifications | # Kafka topic for push notifications | ||||||
| toPushTopic: toPush | toPushTopic: toPush | ||||||
|  | # Kafka topic for offline push notifications | ||||||
|  | toOfflinePushTopic: toOfflinePush | ||||||
| # Consumer group ID for Redis topic | # Consumer group ID for Redis topic | ||||||
| toRedisGroupID: redis | toRedisGroupID: redis | ||||||
| # Consumer group ID for MongoDB topic | # Consumer group ID for MongoDB topic | ||||||
| toMongoGroupID: mongo | toMongoGroupID: mongo | ||||||
| # Consumer group ID for push notifications topic | # Consumer group ID for push notifications topic | ||||||
| toPushGroupID: push | toPushGroupID: push | ||||||
|  | # Consumer group ID for offline push notifications topic | ||||||
|  | toOfflinePushGroupID: offlinePush | ||||||
| # TLS (Transport Layer Security) configuration | # TLS (Transport Layer Security) configuration | ||||||
| tls: | tls: | ||||||
|   # Enable or disable TLS |   # Enable or disable TLS | ||||||
|  | |||||||
| @ -3,11 +3,14 @@ api: | |||||||
|   listenIP: 0.0.0.0 |   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 |   # Listening ports; if multiple are configured, multiple instances will be launched, must be consistent with the number of prometheus.ports | ||||||
|   ports: [ 10002 ] |   ports: [ 10002 ] | ||||||
|  |   # API compression level; 0: default compression, 1: best compression, 2: best speed, -1: no compression | ||||||
|  |   compressionLevel: 0 | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| prometheus: | prometheus: | ||||||
|   # Whether to enable prometheus |   # Whether to enable prometheus | ||||||
|   enable: true |   enable: true | ||||||
|   # Prometheus listening ports, must match the number of api.ports |   # Prometheus listening ports, must match the number of api.ports | ||||||
|   ports: [ 20502 ] |   ports: [ 12002 ] | ||||||
|   # This address can be accessed via a browser |   # This address can be accessed via a browser | ||||||
|   grafanaURL: http://127.0.0.1:13000/ |   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 |   # The IP address where this RPC service registers itself; if left blank, it defaults to the internal network IP | ||||||
|   registerIP:  |   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 |   # 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: | prometheus: | ||||||
|   # Enable or disable Prometheus monitoring |   # Enable or disable Prometheus monitoring | ||||||
|   enable: true |   enable: true | ||||||
|   # List of ports that Prometheus listens on; these must match the number of rpc.ports to ensure correct monitoring setup |   # 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 | # 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 | listenIP: 0.0.0.0 | ||||||
| @ -23,8 +23,4 @@ longConnSvr: | |||||||
|   # WebSocket connection handshake timeout in seconds |   # WebSocket connection handshake timeout in seconds | ||||||
|   websocketTimeout: 10 |   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 |   enable: true | ||||||
|   # List of ports that Prometheus listens on; each port corresponds to an instance of monitoring. Ensure these are managed accordingly |   # 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 |   # 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 ] | ||||||
|  | |||||||
| @ -1,16 +1,16 @@ | |||||||
| rpc: | rpc: | ||||||
|   # The IP address where this RPC service registers itself; if left blank, it defaults to the internal network IP |   # The IP address where this RPC service registers itself; if left blank, it defaults to the internal network IP | ||||||
|   registerIP:  |   registerIP: | ||||||
|   # 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 |   # 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 |   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 |   # 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: | prometheus: | ||||||
|   # Enable or disable Prometheus monitoring |   # Enable or disable Prometheus monitoring | ||||||
|   enable: true |   enable: true | ||||||
|   # List of ports that Prometheus listens on; these must match the number of rpc.ports to ensure correct monitoring setup |   # 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 | maxConcurrentWorkers: 3 | ||||||
| #Use geTui for offline push notifications, or choose fcm or jpns; corresponding configuration settings must be specified. | #Use geTui for offline push notifications, or choose fcm or jpns; corresponding configuration settings must be specified. | ||||||
| @ -38,9 +38,4 @@ iosPush: | |||||||
|       badgeCount: true |       badgeCount: true | ||||||
|       production: false |       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 |   # 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 |   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 |   # 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: | prometheus: | ||||||
|   # Enable or disable Prometheus monitoring |   # Enable or disable Prometheus monitoring | ||||||
|   enable: true |   enable: true | ||||||
|   # List of ports that Prometheus listens on; these must match the number of rpc.ports to ensure correct monitoring setup |   # 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: | tokenPolicy: | ||||||
|   # Token validity period, in days |   # Token validity period, in days | ||||||
|   expire: 90 |   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 |   # 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 |   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 |   # 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: | prometheus: | ||||||
|   # Enable or disable Prometheus monitoring |   # Enable or disable Prometheus monitoring | ||||||
|   enable: true |   enable: true | ||||||
|   # List of ports that Prometheus listens on; these must match the number of rpc.ports to ensure correct monitoring setup |   # 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 |   # 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 |   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 |   # 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: | prometheus: | ||||||
|   # Enable or disable Prometheus monitoring |   # Enable or disable Prometheus monitoring | ||||||
|   enable: true |   enable: true | ||||||
|   # List of ports that Prometheus listens on; these must match the number of rpc.ports to ensure correct monitoring setup |   # 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 |   # 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 |   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 |   # 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: | prometheus: | ||||||
|   # Enable or disable Prometheus monitoring |   # Enable or disable Prometheus monitoring | ||||||
|   enable: true |   enable: true | ||||||
|   # List of ports that Prometheus listens on; these must match the number of rpc.ports to ensure correct monitoring setup |   # 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 | 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 |   # 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 |   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 |   # 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: | prometheus: | ||||||
|   # Enable or disable Prometheus monitoring |   # Enable or disable Prometheus monitoring | ||||||
|   enable: true |   enable: true | ||||||
|   # List of ports that Prometheus listens on; these must match the number of rpc.ports to ensure correct monitoring setup |   # 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 | # Does sending messages require friend verification | ||||||
| friendVerify: false | 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 |   # 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 |   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 |   # 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: | prometheus: | ||||||
|   # Enable or disable Prometheus monitoring |   # Enable or disable Prometheus monitoring | ||||||
|   enable: true |   enable: true | ||||||
|   # List of ports that Prometheus listens on; these must match the number of rpc.ports to ensure correct monitoring setup |   # 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: | object: | ||||||
| @ -37,4 +37,4 @@ object: | |||||||
|     accessKeyID:  |     accessKeyID:  | ||||||
|     accessKeySecret:  |     accessKeySecret:  | ||||||
|     sessionToken:  |     sessionToken:  | ||||||
|     publicRead: false |     publicRead: false | ||||||
|  | |||||||
| @ -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 |   # 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 |   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 |   # 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: | prometheus: | ||||||
|   # Whether to enable prometheus |   # Whether to enable prometheus | ||||||
|   enable: true |   enable: true | ||||||
|   # Prometheus listening ports, must be consistent with the number of rpc.ports |   # 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 ] |       - targets: [ internal_ip:20500 ] | ||||||
|   - job_name: openimserver-openim-api |   - job_name: openimserver-openim-api | ||||||
|     static_configs: |     static_configs: | ||||||
|       - targets: [ internal_ip:20502 ] |       - targets: [ internal_ip:12002 ] | ||||||
|         labels: |         labels: | ||||||
|           namespace: default |           namespace: default | ||||||
|   - job_name: openimserver-openim-msggateway |   - job_name: openimserver-openim-msggateway | ||||||
|     static_configs: |     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: |         labels: | ||||||
|           namespace: default |           namespace: default | ||||||
|   - job_name: openimserver-openim-msgtransfer |   - job_name: openimserver-openim-msgtransfer | ||||||
|     static_configs: |     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: |         labels: | ||||||
|           namespace: default |           namespace: default | ||||||
|   - job_name: openimserver-openim-push |   - job_name: openimserver-openim-push | ||||||
|     static_configs: |     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: |         labels: | ||||||
|           namespace: default |           namespace: default | ||||||
|   - job_name: openimserver-openim-rpc-auth |   - job_name: openimserver-openim-rpc-auth | ||||||
|     static_configs: |     static_configs: | ||||||
|       - targets: [ internal_ip:20600 ] |       - targets: [ internal_ip:12200 ] | ||||||
|         labels: |         labels: | ||||||
|           namespace: default |           namespace: default | ||||||
|   - job_name: openimserver-openim-rpc-conversation |   - job_name: openimserver-openim-rpc-conversation | ||||||
|     static_configs: |     static_configs: | ||||||
|       - targets: [ internal_ip:20680 ] |       - targets: [ internal_ip:12220 ] | ||||||
|         labels: |         labels: | ||||||
|           namespace: default |           namespace: default | ||||||
|   - job_name: openimserver-openim-rpc-friend |   - job_name: openimserver-openim-rpc-friend | ||||||
|     static_configs: |     static_configs: | ||||||
|       - targets: [ internal_ip:20620 ] |       - targets: [ internal_ip:12240 ] | ||||||
|         labels: |         labels: | ||||||
|           namespace: default |           namespace: default | ||||||
|   - job_name: openimserver-openim-rpc-group |   - job_name: openimserver-openim-rpc-group | ||||||
|     static_configs: |     static_configs: | ||||||
|       - targets: [ internal_ip:20650 ] |       - targets: [ internal_ip:12260 ] | ||||||
|         labels: |         labels: | ||||||
|           namespace: default |           namespace: default | ||||||
|   - job_name: openimserver-openim-rpc-msg |   - job_name: openimserver-openim-rpc-msg | ||||||
|     static_configs: |     static_configs: | ||||||
|       - targets: [ internal_ip:20630 ] |       - targets: [ internal_ip:12280 ] | ||||||
|         labels: |         labels: | ||||||
|           namespace: default |           namespace: default | ||||||
|   - job_name: openimserver-openim-rpc-third |   - job_name: openimserver-openim-rpc-third | ||||||
|     static_configs: |     static_configs: | ||||||
|       - targets: [ internal_ip:20690 ] |       - targets: [ internal_ip:12300 ] | ||||||
|         labels: |         labels: | ||||||
|           namespace: default |           namespace: default | ||||||
|   - job_name: openimserver-openim-rpc-user |   - job_name: openimserver-openim-rpc-user | ||||||
|     static_configs: |     static_configs: | ||||||
|       - targets: [ internal_ip:20610 ] |       - targets: [ internal_ip:12320 ] | ||||||
|         labels: |         labels: | ||||||
|           namespace: default |           namespace: default | ||||||
| @ -4,3 +4,4 @@ password: openIM123 | |||||||
| clusterMode: false | clusterMode: false | ||||||
| db: 0 | db: 0 | ||||||
| maxRetry: 10 | maxRetry: 10 | ||||||
|  | poolSize: 100 | ||||||
|  | |||||||
| @ -12,3 +12,18 @@ rpcRegisterName: | |||||||
| 
 | 
 | ||||||
| imAdminUserID: [ imAdmin ] | 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: | beforeSendSingleMsg: | ||||||
|   enable: false |   enable: false | ||||||
|   timeout: 5 |   timeout: 5 | ||||||
|   failedContinue: true |   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: | beforeUpdateUserInfoEx: | ||||||
|   enable:  false |   enable:  false | ||||||
|   timeout: 5 |   timeout: 5 | ||||||
| @ -16,17 +26,29 @@ afterSendSingleMsg: | |||||||
|   # Only the senID/recvID specified in attentionIds will send the callback |   # Only the senID/recvID specified in attentionIds will send the callback | ||||||
|   # if not set, all user messages will be callback |   # if not set, all user messages will be callback | ||||||
|   attentionIds: [] |   attentionIds: [] | ||||||
|  |   # See beforeSendSingleMsg comment. | ||||||
|  |   allowedTypes: [] | ||||||
|  |   deniedTypes: [] | ||||||
| beforeSendGroupMsg: | beforeSendGroupMsg: | ||||||
|   enable: false |   enable: false | ||||||
|   timeout: 5 |   timeout: 5 | ||||||
|   failedContinue: true |   failedContinue: true | ||||||
|  |   # See beforeSendSingleMsg comment. | ||||||
|  |   allowedTypes: [] | ||||||
|  |   deniedTypes: [] | ||||||
| beforeMsgModify: | beforeMsgModify: | ||||||
|   enable: false |   enable: false | ||||||
|   timeout: 5 |   timeout: 5 | ||||||
|   failedContinue: true |   failedContinue: true | ||||||
|  |   # See beforeSendSingleMsg comment. | ||||||
|  |   allowedTypes: [] | ||||||
|  |   deniedTypes: [] | ||||||
| afterSendGroupMsg: | afterSendGroupMsg: | ||||||
|   enable: false |   enable: false | ||||||
|   timeout: 5 |   timeout: 5 | ||||||
|  |   # See beforeSendSingleMsg comment. | ||||||
|  |   allowedTypes: [] | ||||||
|  |   deniedTypes: [] | ||||||
| afterUserOnline: | afterUserOnline: | ||||||
|   enable: false |   enable: false | ||||||
|   timeout: 5 |   timeout: 5 | ||||||
| @ -130,10 +152,10 @@ beforeSetGroupInfo: | |||||||
|   enable: false |   enable: false | ||||||
|   timeout: 5 |   timeout: 5 | ||||||
|   failedContinue: true |   failedContinue: true | ||||||
| afterSetGroupInfoEX: | afterSetGroupInfoEx: | ||||||
|   enable: false |   enable: false | ||||||
|   timeout: 5 |   timeout: 5 | ||||||
| beforeSetGroupInfoEX: | beforeSetGroupInfoEx: | ||||||
|   enable: false |   enable: false | ||||||
|   timeout: 5 |   timeout: 5 | ||||||
|   failedContinue: true |   failedContinue: true | ||||||
| @ -151,6 +173,9 @@ beforeAddFriendAgree: | |||||||
|   enable: false |   enable: false | ||||||
|   timeout: 5 |   timeout: 5 | ||||||
|   failedContinue: true |   failedContinue: true | ||||||
|  | afterAddFriendAgree: | ||||||
|  |   enable: false | ||||||
|  |   timeout: 5 | ||||||
| afterDeleteFriend: | afterDeleteFriend: | ||||||
|   enable: false |   enable: false | ||||||
|   timeout: 5 |   timeout: 5 | ||||||
|  | |||||||
| @ -1,5 +1,3 @@ | |||||||
| version: '3' |  | ||||||
| 
 |  | ||||||
| networks: | networks: | ||||||
|   openim: |   openim: | ||||||
|     driver: bridge |     driver: bridge | ||||||
| @ -45,19 +43,6 @@ services: | |||||||
|     networks: |     networks: | ||||||
|       - openim |       - 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: |   etcd: | ||||||
|     image: "${ETCD_IMAGE}" |     image: "${ETCD_IMAGE}" | ||||||
|     container_name: etcd |     container_name: etcd | ||||||
| @ -144,6 +129,7 @@ services: | |||||||
| #    image: ${PROMETHEUS_IMAGE} | #    image: ${PROMETHEUS_IMAGE} | ||||||
| #    container_name: prometheus | #    container_name: prometheus | ||||||
| #    restart: always | #    restart: always | ||||||
|  | #    user: root | ||||||
| #    volumes: | #    volumes: | ||||||
| #      - ./config/prometheus.yml:/etc/prometheus/prometheus.yml | #      - ./config/prometheus.yml:/etc/prometheus/prometheus.yml | ||||||
| #      - ./config/instance-down-rules.yml:/etc/prometheus/instance-down-rules.yml | #      - ./config/instance-down-rules.yml:/etc/prometheus/instance-down-rules.yml | ||||||
| @ -186,4 +172,3 @@ services: | |||||||
| #    networks: | #    networks: | ||||||
| #      - openim | #      - openim | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|  | |||||||
							
								
								
									
										79
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										79
									
								
								go.mod
									
									
									
									
									
								
							| @ -3,24 +3,24 @@ module github.com/openimsdk/open-im-server/v3 | |||||||
| go 1.21.2 | go 1.21.2 | ||||||
| 
 | 
 | ||||||
| require ( | 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/dtm-labs/rockscache v0.1.1 | ||||||
| 	github.com/gin-gonic/gin v1.9.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/gogo/protobuf v1.3.2 // indirect | ||||||
| 	github.com/golang-jwt/jwt/v4 v4.5.0 | 	github.com/golang-jwt/jwt/v4 v4.5.0 | ||||||
| 	github.com/gorilla/websocket v1.5.1 | 	github.com/gorilla/websocket v1.5.1 | ||||||
| 	github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0 | 	github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0 | ||||||
| 	github.com/mitchellh/mapstructure v1.5.0 | 	github.com/mitchellh/mapstructure v1.5.0 | ||||||
| 	github.com/openimsdk/protocol v0.0.72-alpha.9 | 	github.com/openimsdk/protocol v0.0.72-alpha.46 | ||||||
| 	github.com/openimsdk/tools v0.0.49-alpha.55 | 	github.com/openimsdk/tools v0.0.50-alpha.16 | ||||||
| 	github.com/pkg/errors v0.9.1 // indirect | 	github.com/pkg/errors v0.9.1 // indirect | ||||||
| 	github.com/prometheus/client_golang v1.18.0 | 	github.com/prometheus/client_golang v1.18.0 | ||||||
| 	github.com/stretchr/testify v1.9.0 | 	github.com/stretchr/testify v1.9.0 | ||||||
| 	go.mongodb.org/mongo-driver v1.14.0 | 	go.mongodb.org/mongo-driver v1.14.0 | ||||||
| 	google.golang.org/api v0.165.0 | 	google.golang.org/api v0.170.0 | ||||||
| 	google.golang.org/grpc v1.62.1 | 	google.golang.org/grpc v1.66.2 | ||||||
| 	google.golang.org/protobuf v1.33.0 | 	google.golang.org/protobuf v1.34.2 | ||||||
| 	gopkg.in/yaml.v3 v3.0.1 | 	gopkg.in/yaml.v3 v3.0.1 | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| @ -29,6 +29,7 @@ require github.com/google/uuid v1.6.0 | |||||||
| require ( | require ( | ||||||
| 	github.com/IBM/sarama v1.43.0 | 	github.com/IBM/sarama v1.43.0 | ||||||
| 	github.com/fatih/color v1.14.1 | 	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/redis v6.15.9+incompatible | ||||||
| 	github.com/go-redis/redismock/v9 v9.2.0 | 	github.com/go-redis/redismock/v9 v9.2.0 | ||||||
| 	github.com/hashicorp/golang-lru/v2 v2.0.7 | 	github.com/hashicorp/golang-lru/v2 v2.0.7 | ||||||
| @ -42,17 +43,17 @@ require ( | |||||||
| 	github.com/stathat/consistent v1.0.0 | 	github.com/stathat/consistent v1.0.0 | ||||||
| 	go.uber.org/automaxprocs v1.5.3 | 	go.uber.org/automaxprocs v1.5.3 | ||||||
| 	golang.org/x/exp v0.0.0-20230905200255-921286631fa9 | 	golang.org/x/exp v0.0.0-20230905200255-921286631fa9 | ||||||
| 	golang.org/x/sync v0.6.0 | 	golang.org/x/sync v0.8.0 | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| require ( | require ( | ||||||
| 	cloud.google.com/go v0.112.0 // indirect | 	cloud.google.com/go v0.112.1 // indirect | ||||||
| 	cloud.google.com/go/compute v1.23.3 // indirect | 	cloud.google.com/go/compute/metadata v0.3.0 // indirect | ||||||
| 	cloud.google.com/go/compute/metadata v0.2.3 // indirect | 	cloud.google.com/go/firestore v1.15.0 // indirect | ||||||
| 	cloud.google.com/go/firestore v1.14.0 // indirect | 	cloud.google.com/go/iam v1.1.7 // indirect | ||||||
| 	cloud.google.com/go/iam v1.1.5 // indirect | 	cloud.google.com/go/longrunning v0.5.5 // indirect | ||||||
| 	cloud.google.com/go/longrunning v0.5.4 // indirect | 	cloud.google.com/go/storage v1.40.0 // indirect | ||||||
| 	cloud.google.com/go/storage v1.36.0 // indirect | 	github.com/MicahParks/keyfunc v1.9.0 // indirect | ||||||
| 	github.com/aliyun/aliyun-oss-go-sdk v3.0.2+incompatible // 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 v1.23.1 // indirect | ||||||
| 	github.com/aws/aws-sdk-go-v2/aws/protocol/eventstream v1.5.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/aws-sdk-go-v2/service/sts v1.25.4 // indirect | ||||||
| 	github.com/aws/smithy-go v1.17.0 // indirect | 	github.com/aws/smithy-go v1.17.0 // indirect | ||||||
| 	github.com/beorn7/perks v1.0.1 // indirect | 	github.com/beorn7/perks v1.0.1 // indirect | ||||||
| 	github.com/bytedance/sonic v1.9.1 // indirect | 	github.com/bytedance/sonic v1.11.6 // indirect | ||||||
| 	github.com/cespare/xxhash/v2 v2.2.0 // indirect | 	github.com/bytedance/sonic/loader v0.1.1 // indirect | ||||||
| 	github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 // indirect | 	github.com/cespare/xxhash/v2 v2.3.0 // indirect | ||||||
| 	github.com/clbanning/mxj v1.8.4 // 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-semver v0.3.0 // indirect | ||||||
| 	github.com/coreos/go-systemd/v22 v22.3.2 // indirect | 	github.com/coreos/go-systemd/v22 v22.3.2 // indirect | ||||||
| 	github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc // 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/go-querystring v1.1.0 // indirect | ||||||
| 	github.com/google/s2a-go v0.1.7 // indirect | 	github.com/google/s2a-go v0.1.7 // indirect | ||||||
| 	github.com/googleapis/enterprise-certificate-proxy v0.3.2 // 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/errwrap v1.1.0 // indirect | ||||||
| 	github.com/hashicorp/go-multierror v1.1.1 // indirect | 	github.com/hashicorp/go-multierror v1.1.1 // indirect | ||||||
| 	github.com/hashicorp/go-uuid v1.0.3 // 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/json-iterator/go v1.1.12 // indirect | ||||||
| 	github.com/kelindar/simd v1.1.2 // indirect | 	github.com/kelindar/simd v1.1.2 // indirect | ||||||
| 	github.com/klauspost/compress v1.17.7 // 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/leodido/go-urn v1.4.0 // indirect | ||||||
| 	github.com/lestrrat-go/strftime v1.0.6 // indirect | 	github.com/lestrrat-go/strftime v1.0.6 // indirect | ||||||
| 	github.com/lithammer/shortuuid v3.0.0+incompatible // 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/modern-go/reflect2 v1.0.2 // indirect | ||||||
| 	github.com/montanaflynn/stats v0.0.0-20171201202039-1bf9dbcd8cbe // indirect | 	github.com/montanaflynn/stats v0.0.0-20171201202039-1bf9dbcd8cbe // indirect | ||||||
| 	github.com/mozillazg/go-httpheader v0.4.0 // 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/pierrec/lz4/v4 v4.1.21 // indirect | ||||||
| 	github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 // indirect | 	github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 // indirect | ||||||
| 	github.com/prometheus/client_model v0.5.0 // 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/pkg/v3 v3.5.13 // indirect | ||||||
| 	go.etcd.io/etcd/client/v3 v3.5.13 // indirect | 	go.etcd.io/etcd/client/v3 v3.5.13 // indirect | ||||||
| 	go.opencensus.io v0.24.0 // 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/google.golang.org/grpc/otelgrpc v0.49.0 // indirect | ||||||
| 	go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.47.0 // indirect | 	go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.49.0 // indirect | ||||||
| 	go.opentelemetry.io/otel v1.23.0 // indirect | 	go.opentelemetry.io/otel v1.24.0 // indirect | ||||||
| 	go.opentelemetry.io/otel/metric v1.23.0 // indirect | 	go.opentelemetry.io/otel/metric v1.24.0 // indirect | ||||||
| 	go.opentelemetry.io/otel/trace v1.23.0 // indirect | 	go.opentelemetry.io/otel/trace v1.24.0 // indirect | ||||||
| 	go.uber.org/atomic v1.9.0 // indirect | 	go.uber.org/atomic v1.9.0 // indirect | ||||||
| 	go.uber.org/multierr v1.11.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/image v0.15.0 // indirect | ||||||
| 	golang.org/x/net v0.22.0 // indirect | 	golang.org/x/net v0.29.0 // indirect | ||||||
| 	golang.org/x/oauth2 v0.17.0 // indirect | 	golang.org/x/oauth2 v0.21.0 // indirect | ||||||
| 	golang.org/x/sys v0.19.0 // indirect | 	golang.org/x/sys v0.25.0 // indirect | ||||||
| 	golang.org/x/text v0.14.0 // indirect | 	golang.org/x/text v0.18.0 // indirect | ||||||
| 	golang.org/x/time v0.5.0 // indirect | 	golang.org/x/time v0.5.0 // indirect | ||||||
| 	google.golang.org/appengine v1.6.8 // indirect | 	google.golang.org/appengine/v2 v2.0.2 // indirect | ||||||
| 	google.golang.org/genproto v0.0.0-20240125205218-1f4bbc51befe // indirect | 	google.golang.org/genproto v0.0.0-20240213162025-012b6fc9bca9 // indirect | ||||||
| 	google.golang.org/genproto/googleapis/api v0.0.0-20240123012728-ef4313101c80 // indirect | 	google.golang.org/genproto/googleapis/api v0.0.0-20240604185151-ef581f913117 // indirect | ||||||
| 	google.golang.org/genproto/googleapis/rpc v0.0.0-20240318140521-94a12d6c2237 // indirect | 	google.golang.org/genproto/googleapis/rpc v0.0.0-20240903143218-8af14fe29dc1 // indirect | ||||||
| 	gorm.io/gorm v1.25.8 // indirect | 	gorm.io/gorm v1.25.8 // indirect | ||||||
| 	stathat.com/c/consistent v1.0.0 // indirect | 	stathat.com/c/consistent v1.0.0 // indirect | ||||||
| ) | ) | ||||||
| @ -187,10 +190,10 @@ require ( | |||||||
| require ( | require ( | ||||||
| 	github.com/go-playground/locales v0.14.1 // indirect | 	github.com/go-playground/locales v0.14.1 // indirect | ||||||
| 	github.com/goccy/go-json v0.10.2 // 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/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 | 	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 | 	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.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.1 h1:uJSeirPke5UNZHIb4SxfZklVSiWWVqW4oXlETwZziwM= | ||||||
| cloud.google.com/go v0.112.0/go.mod h1:3jEEVwZ/MHU4djK5t5RHuKOA/GbLddgTdVubX1qnPD4= | cloud.google.com/go v0.112.1/go.mod h1:+Vbu+Y1UU+I1rjmzeMOb/8RfkKJK2Gyxi1X6jJCZLo4= | ||||||
| cloud.google.com/go/compute v1.23.3 h1:6sVlXXBmbd7jNX0Ipq0trII3e4n1/MsADLK6a+aiVlk= | cloud.google.com/go/compute/metadata v0.3.0 h1:Tz+eQXMEqDIKRsmY3cHTL6FVaynIjX2QxYC4trgAKZc= | ||||||
| cloud.google.com/go/compute v1.23.3/go.mod h1:VCgBUoMnIVIR0CscqQiPJLAG25E3ZRZMzcFZeQ+h8CI= | cloud.google.com/go/compute/metadata v0.3.0/go.mod h1:zFmK7XCadkQkj6TtorcaGlCW1hT1fIilQDwofLpJ20k= | ||||||
| cloud.google.com/go/compute/metadata v0.2.3 h1:mg4jlk7mCAj6xXp9UJ4fjI9VUI5rubuGBW5aJ7UnBMY= | cloud.google.com/go/firestore v1.15.0 h1:/k8ppuWOtNuDHt2tsRV42yI21uaGnKDEQnRFeBpbFF8= | ||||||
| cloud.google.com/go/compute/metadata v0.2.3/go.mod h1:VAV5nSsACxMJvgaAuX6Pk2AawlZn8kiOGuCv6gTkwuA= | cloud.google.com/go/firestore v1.15.0/go.mod h1:GWOxFXcv8GZUtYpWHw/w6IuYNux/BtmeVTMmjrm4yhk= | ||||||
| cloud.google.com/go/firestore v1.14.0 h1:8aLcKnMPoldYU3YHgu4t2exrKhLQkqaXAGqT0ljrFVw= | cloud.google.com/go/iam v1.1.7 h1:z4VHOhwKLF/+UYXAJDFwGtNF0b6gjsW1Pk9Ml0U/IoM= | ||||||
| cloud.google.com/go/firestore v1.14.0/go.mod h1:96MVaHLsEhbvkBEdZgfN+AS/GIkco1LRpH9Xp9YZfzQ= | cloud.google.com/go/iam v1.1.7/go.mod h1:J4PMPg8TtyurAUvSmPj8FF3EDgY1SPRZxcUGrn7WXGA= | ||||||
| cloud.google.com/go/iam v1.1.5 h1:1jTsCu4bcsNsE4iiqNT5SHwrDRCfRmIaaaVFhRveTJI= | cloud.google.com/go/longrunning v0.5.5 h1:GOE6pZFdSrTb4KAiKnXsJBtlE6mEyaW44oKyMILWnOg= | ||||||
| cloud.google.com/go/iam v1.1.5/go.mod h1:rB6P/Ic3mykPbFio+vo7403drjlgvoWfYpJhMXEbzv8= | cloud.google.com/go/longrunning v0.5.5/go.mod h1:WV2LAxD8/rg5Z1cNW6FJ/ZpX4E4VnDnoTk0yawPBB7s= | ||||||
| cloud.google.com/go/longrunning v0.5.4 h1:w8xEcbZodnA2BbW6sVirkkoC+1gP8wS57EUUgGS0GVg= | cloud.google.com/go/storage v1.40.0 h1:VEpDQV5CJxFmJ6ueWNsKxcr1QAYOXEgxDa+sBbJahPw= | ||||||
| cloud.google.com/go/longrunning v0.5.4/go.mod h1:zqNVncI0BOP8ST6XQD1+VcvuShMmq7+xFSzOL++V0dI= | cloud.google.com/go/storage v1.40.0/go.mod h1:Rrj7/hKlG87BLqDJYtwR0fbPld8uJPbQ2ucUMY7Ir0g= | ||||||
| cloud.google.com/go/storage v1.36.0 h1:P0mOkAcaJxhCTvAkMhxMfrTKiNcub4YmmPBtlhAyTr8= | firebase.google.com/go/v4 v4.14.1 h1:4qiUETaFRWoFGE1XP5VbcEdtPX93Qs+8B/7KvP2825g= | ||||||
| cloud.google.com/go/storage v1.36.0/go.mod h1:M6M/3V/D3KpzMTJyPOR/HU6n2Si5QdaXYEsng2xgOs8= | firebase.google.com/go/v4 v4.14.1/go.mod h1:fgk2XshgNDEKaioKco+AouiegSI9oTWVqRaBdTTGBoM= | ||||||
| firebase.google.com/go v3.13.0+incompatible h1:3TdYC3DDi6aHn20qoRkxwGqNgdjtblwVAyRLQwGn/+4= |  | ||||||
| firebase.google.com/go v3.13.0+incompatible/go.mod h1:xlah6XbEyW6tbfSklcfe5FHJIwjt8toICdV5Wh9ptHs= |  | ||||||
| github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= | 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 h1:YFFDn8mMI2QL0wOrG0J2sFoVIAFl7hS9JQi2YZsXtJc= | ||||||
| github.com/IBM/sarama v1.43.0/go.mod h1:zlE6HEbC/SMQ9mhEYaF7nNLYOUyrs0obySKCckWP9BM= | 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/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 h1:8psS8a+wKfiLt1iVDX79F7Y6wUM49Lcha2FMXt4UM8g= | ||||||
| github.com/aliyun/aliyun-oss-go-sdk v3.0.2+incompatible/go.mod h1:T/Aws4fEfogEE9v+HPhhw+CntffsBHJ8nXQCwKr0/g8= | 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/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 h1:yeMWxP2pV2fG3FgAODIY8EiRE3dy0aeFYt4l7wh6yKA= | ||||||
| github.com/bsm/gomega v1.27.10/go.mod h1:JyEr/xRbxbtgWNi8tIEVPUYZ5Dzef52k01W3YH0H+O0= | 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.11.6 h1:oUp34TzMlL+OY1OUWxHqsdkgC/Zfc85zGqw9siXjrc0= | ||||||
| github.com/bytedance/sonic v1.9.1 h1:6iJ6NqdoxCDr6mbY8h18oSO+cShGSMRGCEo7F2h0x8s= | github.com/bytedance/sonic v1.11.6/go.mod h1:LysEHSvpvDySVdC2f87zGWf6CIKJcAvqab1ZaiQtds4= | ||||||
| github.com/bytedance/sonic v1.9.1/go.mod h1:i736AoUSYt75HyZLoJW9ERYxcy6eaN6h4BZXU064P/U= | 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/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.3.0 h1:UL815xU9SqsFlibzuggzjXhog7bL6oX9BbNZnL2UFvs= | ||||||
| github.com/cespare/xxhash/v2 v2.2.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | github.com/cespare/xxhash/v2 v2.3.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/clbanning/mxj v1.8.4 h1:HuhwZtbyvyOw+3Z1AowPkU87JkJUSv751ELWaiTpj8I= | 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/clbanning/mxj v1.8.4/go.mod h1:BVjHeAH+rl9rs6f+QIpeRl0tfu10SXn1pUSa5PVGJng= | ||||||
| github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw= | 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/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 h1:wkHLiw0WNATZnSG7epLsujiMCgPAc9xhjJ4tgnAxmfM= | ||||||
| github.com/coreos/go-semver v0.3.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk= | 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= | 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.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/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 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 h1:qfhVLaG5s+nCROl1zJsZRxFeYrHLqWroPOQ8BWiNb4w= | ||||||
| github.com/fatih/color v1.14.1/go.mod h1:2oHN61fhTpgcxD3TSWCgKDiH1+x4OiDVVGH8WlgGZGg= | github.com/fatih/color v1.14.1/go.mod h1:2oHN61fhTpgcxD3TSWCgKDiH1+x4OiDVVGH8WlgGZGg= | ||||||
| github.com/felixge/httpsnoop v1.0.4 h1:NFTV2Zj1bL4mc9sqWACXbQFVBBg2W3GPvqp8/ESS2Wg= | 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/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 h1:in2uUcidCuFcDKtdcBxlR0rJ1+fsokWf+uqxgUFjbI0= | ||||||
| github.com/gabriel-vasile/mimetype v1.4.3/go.mod h1:d8uq/6HKRL6CGdk+aubisF/M5GcPfT7nKyLpA0lbSSk= | 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 h1:Y/yl/+YNO8GZSjAhjMsSuLt29uWRFHdHYUb5lYOV9qE= | ||||||
| github.com/gin-contrib/sse v0.1.0/go.mod h1:RHrZQHXnP2xjPF+u1gW/2HnVO7nvIa9PG3Gm+fLHvGI= | 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= | 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 h1:Bcnm0ZwsGyWbCzImXv+pAJnYK9S473LQFuzCbDbfSFY= | ||||||
| github.com/go-playground/universal-translator v0.18.1/go.mod h1:xekY+UJKNuX9WP91TpwSH2VMlDf28Uj24BCp08ZFTUY= | 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.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.20.0 h1:K9ISHbSaI0lyB2eWMPJo+kOS/FBExVwjEviJTixqxL8= | ||||||
| github.com/go-playground/validator/v10 v10.18.0/go.mod h1:dbuPbCMFw/DrkbEynArYaCwl3amGuJotoKCe95atGMM= | 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 h1:K0pv1D7EQUjfyoMql+r/jZqCLizCGKFlFgcHWWmHQjg= | ||||||
| github.com/go-redis/redis v6.15.9+incompatible/go.mod h1:NAIEuMOZ/fxfXJIrKDQDz8wamY7mA7PouImQ2Jvg6kA= | 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= | 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/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 h1:Ov1cvc58UF3b5XjBnZv7+opcTcQFZebYjWzi34vdm4Q= | ||||||
| github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q= | 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 h1:7cYmW1XlMY7h7ii7UhUyChSgS5wUJEnm9uZVTGqOWzg= | ||||||
| github.com/golang-jwt/jwt/v4 v4.5.0/go.mod h1:m21LjoU+eqJr34lmDMbreY2eSTRJ1cv77w39/MY0Ch0= | 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= | 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/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/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.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.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/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8= | ||||||
| github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA= | 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.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0= | ||||||
| github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8= | 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.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 h1:i7eJL8qZTpSEXOPTxNKhASYpMn+8e5Q6AdndVa1dWek= | ||||||
| github.com/golang/protobuf v1.5.4/go.mod h1:lnTiLA8Wa4RWRcIUkrtSVa5nRhsEGBg48fD6rSs7xps= | github.com/golang/protobuf v1.5.4/go.mod h1:lnTiLA8Wa4RWRcIUkrtSVa5nRhsEGBg48fD6rSs7xps= | ||||||
| github.com/golang/snappy v0.0.4 h1:yAGX7huGHXlcLOEtBnF4w7FQwA26wojNCwOYAEhLjQM= | 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.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.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.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 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= | ||||||
| github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= | 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= | 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/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 h1:Vie5ybvEvT75RniqhfFxPRy3Bf7vr3h0cechB90XaQs= | ||||||
| github.com/googleapis/enterprise-certificate-proxy v0.3.2/go.mod h1:VLSiSSBs/ksPL8kq3OBOQ6WRI2QnaFynd1DCjZ62+V0= | 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.3 h1:5/zPPDvw8Q1SuXjrqrZslrqT7dL/uJT2CQii/cLCKqA= | ||||||
| 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/go.mod h1:AKloxT6GtNbaLm8QTNSidHUVsHYcBHwWRvkNFJUQcS4= | ||||||
| github.com/gorilla/securecookie v1.1.1/go.mod h1:ra0sb63/xPlUeL+yeDciTfxMRAA+MP+HVt/4epWDjd4= | 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/sessions v1.2.1/go.mod h1:dk2InVEVJ0sfLlnXv9EAgkf6ecYs/i80K/zI+bUmuGM= | ||||||
| github.com/gorilla/websocket v1.5.1 h1:gmztn0JnHVt9JZquRuzLw3g4wouNVzKL15iLr/zn/QY= | 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/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.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.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.7 h1:ZWSB3igEs+d0qvnxR/ZBzXVmxkgt8DdzP6m9pfuVLDM= | ||||||
| github.com/klauspost/cpuid/v2 v2.2.6/go.mod h1:Lcz8mBdAVJIBVzewtcLocK12l3Y+JytZYpaMropDUws= | 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.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= | ||||||
| github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= | github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= | ||||||
| github.com/kr/pretty v0.3.0/go.mod h1:640gp4NfQd8pI5XOwp5fnNeVWj67G7CFk/SaSQn7NBk= | 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 h1:fFA4WZxdEF4tXPZVKMLwD8oUnCTTo08duU7wxecdEvA= | ||||||
| github.com/mattn/go-colorable v0.1.13/go.mod h1:7S9/ev0klgBDR4GtXTXX8a3vIGJpMovkB8vQcUbaXHg= | 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.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.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY= | ||||||
| github.com/mattn/go-isatty v0.0.19/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y= | 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 h1:jWpvCLoY8Z/e3VKvlsiIGKtc+UG6U5vzxaoagmhXfyg= | ||||||
| github.com/matttproud/golang_protobuf_extensions/v2 v2.0.0/go.mod h1:QUyp042oQthUoa9bqDv0ER0wrtXnBruoNd7aNjkbP+k= | 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= | 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/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 h1:VY9c5x515lTfmdhhPjMvR3BBRrRquAUCFsz7t7vbv7Y= | ||||||
| github.com/openimsdk/gomake v0.0.14-alpha.5/go.mod h1:PndCozNc2IsQIciyn9mvEblYWZwJmAI+06z94EY+csI= | 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.46 h1:1LZlfEHLzw1F4afFmqBczmXKJWm5rUQ+yr8rJ4oyEAc= | ||||||
| github.com/openimsdk/protocol v0.0.72-alpha.9/go.mod h1:OZQA9FR55lseYoN2Ql1XAHYKHJGu7OMNkUbuekrKCM8= | github.com/openimsdk/protocol v0.0.72-alpha.46/go.mod h1:OZQA9FR55lseYoN2Ql1XAHYKHJGu7OMNkUbuekrKCM8= | ||||||
| github.com/openimsdk/tools v0.0.49-alpha.55 h1:KPgC53oqiwZYssLKljhtXbWXifMlTj2SSQEusj4Uf4k= | github.com/openimsdk/tools v0.0.50-alpha.16 h1:bC1AQvJMuOHtZm8LZRvN8L5mH1Ws2VYdL+TLTs1iGSc= | ||||||
| github.com/openimsdk/tools v0.0.49-alpha.55/go.mod h1:h1cYmfyaVtgFbKmb1Cfsl8XwUOMTt8ubVUQrdGtsUh4= | github.com/openimsdk/tools v0.0.50-alpha.16/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.2.2 h1:aYUidT7k73Pcl9nb2gScu7NSrKCSHIDE89b3+6Wq+LM= | ||||||
| github.com/pelletier/go-toml/v2 v2.1.0/go.mod h1:tJU2Z3ZkXwnxa4DPO899bsyIoywizdUvyaeZurnPPDc= | 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 h1:yOVMLb6qSIDP67pl/5F7RepeKYu/VmTyEXvuMI5d9mQ= | ||||||
| github.com/pierrec/lz4/v4 v4.1.21/go.mod h1:gZWDp/Ze/IJXGXf23ltt2EXimqmTUXEy0GFuRQyBid4= | 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= | 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/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 h1:SU5vSMR7hnwNxj24w34ZyCi/FmDZTkS4MhqMhdFk5YI= | ||||||
| github.com/twitchyliquid64/golang-asm v0.15.1/go.mod h1:a1lVb/DtPvCB8fslRZhAngC2+aY1QWCk3Cedj/Gdt08= | 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.12 h1:9LC83zGrHhuUA9l16C9AHXAqEV/2wBQ4nkvumAE65EE= | ||||||
| github.com/ugorji/go/codec v1.2.11/go.mod h1:UNopzCgEMSXjBc6AOMqYvWC1ktqTAfzJZUZgYf6w6lg= | 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 h1:Su7DPu48wXMwC3bs7MCNG+z4FhcyEuz5dlvchbq0B0c= | ||||||
| github.com/xdg-go/pbkdf2 v1.0.0/go.mod h1:jrpuAogTd400dnrH08LKmI/xc1MbPOebTwRqcT5RDeI= | github.com/xdg-go/pbkdf2 v1.0.0/go.mod h1:jrpuAogTd400dnrH08LKmI/xc1MbPOebTwRqcT5RDeI= | ||||||
| github.com/xdg-go/scram v1.1.2 h1:FHX5I5B4i4hKRVRBCFRxq1iQRej7WO3hhBuJf+UUySY= | 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.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 h1:y73uSU6J157QMP2kn2r30vwW1A2W2WFwSCGnAVxeaD0= | ||||||
| go.opencensus.io v0.24.0/go.mod h1:vNK8G9p7aAivkbmorf4v+7Hgx+Zs0yY+0fOtgBfjQKo= | 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.49.0 h1:4Pp6oUg3+e/6M4C0A/3kJ2VYa++dsWVTtGgLVj5xtHg= | ||||||
| go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.47.0/go.mod h1:r9vWsPS/3AQItv3OSlEJ/E4mbrhUbbw18meOjArPtKQ= | 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.47.0 h1:sv9kVfal0MK0wBMCOGr+HeJm9v803BkJxGrk2au7j08= | 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.47.0/go.mod h1:SK2UL73Zy1quvRPonmOmRDiWk1KBV3LyIeeIxcEApWw= | go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.49.0/go.mod h1:p8pYQP+m5XfbZm9fxtSKAbM6oIllS7s2AfxrChvc7iw= | ||||||
| go.opentelemetry.io/otel v1.23.0 h1:Df0pqjqExIywbMCMTxkAwzjLZtRf+bBKLbUcpxO2C9E= | go.opentelemetry.io/otel v1.24.0 h1:0LAOdjNmQeSTzGBzduGe/rU4tZhMwL5rWgtp9Ku5Jfo= | ||||||
| go.opentelemetry.io/otel v1.23.0/go.mod h1:YCycw9ZeKhcJFrb34iVSkyT0iczq/zYDtZYFufObyB0= | go.opentelemetry.io/otel v1.24.0/go.mod h1:W7b9Ozg4nkF5tWI5zsXkaKKDjdVjpD4oAt9Qi/MArHo= | ||||||
| go.opentelemetry.io/otel/metric v1.23.0 h1:pazkx7ss4LFVVYSxYew7L5I6qvLXHA0Ap2pwV+9Cnpo= | go.opentelemetry.io/otel/metric v1.24.0 h1:6EhoGWWK28x1fbpA4tYTOWBkPefTDQnb8WSGXlc88kI= | ||||||
| go.opentelemetry.io/otel/metric v1.23.0/go.mod h1:MqUW2X2a6Q8RN96E2/nqNoT+z9BSms20Jb7Bbp+HiTo= | go.opentelemetry.io/otel/metric v1.24.0/go.mod h1:VYhLe1rFfxuTXLgj4CBiyz+9WYBA8pNGJgDcSFRKBco= | ||||||
| go.opentelemetry.io/otel/sdk v1.21.0 h1:FTt8qirL1EysG6sTQRZ5TokkU8d0ugCj8htOgThZXQ8= | go.opentelemetry.io/otel/sdk v1.22.0 h1:6coWHw9xw7EfClIC/+O31R8IY3/+EiRFHevmHafB2Gw= | ||||||
| go.opentelemetry.io/otel/sdk v1.21.0/go.mod h1:Nna6Yv7PWTdgJHVRD9hIYywQBRx7pbox6nwBnZIxl/E= | go.opentelemetry.io/otel/sdk v1.22.0/go.mod h1:iu7luyVGYovrRpe2fmj3CVKouQNdTOkxtLzPvPz1DOc= | ||||||
| go.opentelemetry.io/otel/trace v1.23.0 h1:37Ik5Ib7xfYVb4V1UtnT97T1jI+AoIYkJyPkuL4iJgI= | go.opentelemetry.io/otel/trace v1.24.0 h1:CsKnnL4dUAr/0llH9FKuc698G04IrpWV0MQA/Y1YELI= | ||||||
| go.opentelemetry.io/otel/trace v1.23.0/go.mod h1:GSGTbIClEsuZrGIzoEHqsVfxgn5UkggkflQwDScNUsk= | 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 h1:ECmE8Bn/WFTYwEW/bpKD3M8VtR/zQVbavAoalC1PYyE= | ||||||
| go.uber.org/atomic v1.9.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc= | go.uber.org/atomic v1.9.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc= | ||||||
| go.uber.org/automaxprocs v1.5.3 h1:kWazyxZUrS3Gs4qUpbwo5kEIMGe/DAvi5Z4tl2NW4j8= | 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 h1:FiJd5l1UOLj0wCgbSE0rwwXHzEdAZS6hiiSnxJN/D60= | ||||||
| go.uber.org/zap v1.24.0/go.mod h1:2kMP+WWQ8aoFoedH3T2sq6iJ2yDWpHbP0f6MQbS9Gkg= | 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.0.0-20210923205945-b76863e36670/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8= | ||||||
| golang.org/x/arch v0.3.0 h1:02VY4/ZcO/gBOH6PUaoiptASxtXU10jazRCP865E97k= | golang.org/x/arch v0.7.0 h1:pskyeJh/3AmoQ8CPE95vxHLqp1G1GfGNXTmcl9NEKTc= | ||||||
| golang.org/x/arch v0.3.0/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8= | 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-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-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||||
| golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | 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.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.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.6.0/go.mod h1:OFC/31mSvZgRz0V1QTNCzfAI1aIRzbiufJtkMIlEp58= | ||||||
| golang.org/x/crypto v0.21.0 h1:X31++rzVUdKhX5sWmSOFZxx8UW/ldWx55cbf08iNAMA= | golang.org/x/crypto v0.27.0 h1:GXm2NjJrPaiv/h1tb2UH8QfgC/hOf/+z0p6PT8o1w7A= | ||||||
| golang.org/x/crypto v0.21.0/go.mod h1:0BP7YvVV9gBbVKyeTG0Gyn+gZm94bibOW5BjDEYAOMs= | 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-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 h1:GoHiUyI/Tp2nVkLI2mCxVkOjsbSXD66ic0XW0js0R9g= | ||||||
| golang.org/x/exp v0.0.0-20230905200255-921286631fa9/go.mod h1:S2oDrQGGwySpoQPVqRShND87VCbxmc6bL1Yd2oYrm6k= | 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-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-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-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.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.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.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.7.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= | ||||||
| golang.org/x/net v0.22.0 h1:9sGLhx7iRIHEiX0oAJ3MRZMUCElJgy7Br1nO+AMN3Tc= | golang.org/x/net v0.29.0 h1:5ORfpBpCs4HzDYoodCDBbwHzdR5UrLBZ3sOnUJmFoHo= | ||||||
| golang.org/x/net v0.22.0/go.mod h1:JKghWKKOSdJwpW2GEx0Ja7fmaKnMsbu+MWVZTokSYmg= | 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.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.21.0 h1:tsimM75w1tF/uws5rbeHzIWxEqElMehnc+iW793zsZs= | ||||||
| golang.org/x/oauth2 v0.17.0/go.mod h1:OzPDGQiuQMguemayvdylqddI7qcD9lnSDb+1FiwQ5HA= | 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-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-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-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-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-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.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.8.0 h1:3NFvSEYkUoMifnESzZl15y791HH1qU2xm6eCJU5ZPXQ= | ||||||
| golang.org/x/sync v0.6.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= | 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-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-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||||
| golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | 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.1.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/sys v0.5.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.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/sys v0.19.0 h1:q5f1RH2jigJ1MoAWp2KTp3gm5zAGFUTarQZ5U386+4o= | golang.org/x/sys v0.25.0 h1:r+8e+loiHxRqhXVl6ML1nO3l1+oFoWbnlu2Ehimmi34= | ||||||
| golang.org/x/sys v0.19.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | 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-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.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= | ||||||
| golang.org/x/term v0.1.0/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.3.8/go.mod h1:E6s5w1FMmriuDzIBO73fBruAKo1PCIq6d2Q6DHfQ8WQ= | ||||||
| golang.org/x/text v0.4.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= | 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.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.18.0 h1:XvMDiNzPAl0jr17s6W9lcaIhGUfUORdGCNsuLmPG224= | ||||||
| golang.org/x/text v0.14.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU= | 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 h1:o7cqy6amK/52YcAKIPlM3a+Fpj35zvRj2TP+e1xFSfk= | ||||||
| golang.org/x/time v0.5.0/go.mod h1:3BpzKBy/shNhVucY/MWOyx10tF3SFh9QdLuxbVysPQM= | 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= | 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-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-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-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-20231012003039-104605ab7028 h1:+cNy6SZtPcJQH3LJVLOSmiC7MMxXNOb3PU/VUEz+EhU= | ||||||
| golang.org/x/xerrors v0.0.0-20220907171357-04be3eba64a2/go.mod h1:K8+ghG5WaK9qNqU5K3HdILfMLy1f3aNYFI/wnl100a8= | golang.org/x/xerrors v0.0.0-20231012003039-104605ab7028/go.mod h1:NDW/Ps6MPRej6fsCIbMTohpP40sJ/P/vI1MoTEGwX90= | ||||||
| google.golang.org/api v0.165.0 h1:zd5d4JIIIaYYsfVy1HzoXYZ9rWCSBxxAglbczzo7Bgc= | google.golang.org/api v0.170.0 h1:zMaruDePM88zxZBG+NG8+reALO2rfLhe/JShitLyT48= | ||||||
| google.golang.org/api v0.165.0/go.mod h1:2OatzO7ZDQsoS7IFf3rvsE17/TldiU3F/zxFHeqUB5o= | 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.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.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= | ||||||
| google.golang.org/appengine v1.6.8 h1:IhEN5q69dyKagZPYMSdIjS2HqprW324FRQZJcGqPAsM= | google.golang.org/appengine/v2 v2.0.2 h1:MSqyWy2shDLwG7chbwBJ5uMyw6SNqJzhJHNDwYB0Akk= | ||||||
| google.golang.org/appengine v1.6.8/go.mod h1:1jJ3jBArFh5pcgW8gCtRJnepW8FzD1V44FJffLiz/Ds= | 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-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-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-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-20240213162025-012b6fc9bca9 h1:9+tzLLstTlPTRyJTh+ah5wIMsBW5c4tQwGTN3thOW9Y= | ||||||
| google.golang.org/genproto v0.0.0-20240125205218-1f4bbc51befe/go.mod h1:cc8bqMqtv9gMOr0zHg2Vzff5ULhhL2IXP4sbcn32Dro= | google.golang.org/genproto v0.0.0-20240213162025-012b6fc9bca9/go.mod h1:mqHbVIp48Muh7Ywss/AD6I5kNVKZMmAa/QEW58Gxp2s= | ||||||
| google.golang.org/genproto/googleapis/api v0.0.0-20240123012728-ef4313101c80 h1:Lj5rbfG876hIAYFjqiJnPHfhXbv+nzTWfm04Fg/XSVU= | google.golang.org/genproto/googleapis/api v0.0.0-20240604185151-ef581f913117 h1:+rdxYoE3E5htTEWIe15GlN6IfvbURM//Jt0mmkmm6ZU= | ||||||
| google.golang.org/genproto/googleapis/api v0.0.0-20240123012728-ef4313101c80/go.mod h1:4jWUdICTdgc3Ibxmr8nAJiiLHwQBY0UI0XZcEMaFKaA= | 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-20240318140521-94a12d6c2237 h1:NnYq6UN9ReLM9/Y01KWNOWyI5xQ9kbIms5GGJVwS/Yc= | google.golang.org/genproto/googleapis/rpc v0.0.0-20240903143218-8af14fe29dc1 h1:pPJltXNxVzT4pK9yD8vR9X75DaWYYmLGMsEvBfFQZzQ= | ||||||
| google.golang.org/genproto/googleapis/rpc v0.0.0-20240318140521-94a12d6c2237/go.mod h1:WtryC6hu0hhx87FDGxWCDptyssuo68sk10vYjF+T9fY= | 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.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c= | ||||||
| google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= | 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.25.1/go.mod h1:c3i+UQWmh7LiEpx4sFZnkU36qjEYZ0imhYfXVyQciAY= | ||||||
| google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= | 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.33.2/go.mod h1:JMHMWHQWaTccqQQlmk3MJZS+GWXOdAesneDmEnv2fbc= | ||||||
| google.golang.org/grpc v1.62.1 h1:B4n+nfKzOICUXMgyrNd19h/I9oH0L1pizfk1d4zSgTk= | google.golang.org/grpc v1.66.2 h1:3QdXkuq3Bkh7w+ywLdLvM56cmGvQHUMZpiCzt6Rqaoo= | ||||||
| google.golang.org/grpc v1.62.1/go.mod h1:IWTG0VlJLCh1SkC58F7np9ka9mx/WNkjl4PGJaiq+QE= | 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-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-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= | ||||||
| google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM= | 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.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.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.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.34.2 h1:6xV6lTsCfpGD21XK49h7MhtcApnLqkfYgPcdHftf6hg= | ||||||
| google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= | google.golang.org/protobuf v1.34.2/go.mod h1:qYOHts0dSfpeUzUFpOMr/WGzszTmLH+DiWniOlNbLDw= | ||||||
| google.golang.org/protobuf v1.33.0 h1:uNO2rsAINq/JlFpSdYEKIZ0uKD/R9cpdv0T+yoGwGmI= |  | ||||||
| google.golang.org/protobuf v1.33.0/go.mod h1:c6P6GXX6sHbq/GpV6MGZEdwhWPcYBgnhAHhKbcUYpos= |  | ||||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | 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-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||||
| gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= | 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= | 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-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= | ||||||
| honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/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= | 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 h1:ezyc51EGcRPJUxfHGSgJjWzJdj3NiMU9pNfLNGiXV0c= | ||||||
| stathat.com/c/consistent v1.0.0/go.mod h1:QkzMWzcbB+yQBL2AttO6sgsQS/JSTapcDISJalmCDS0= | 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) | 	return AuthApi(client) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (o *AuthApi) UserToken(c *gin.Context) { | func (o *AuthApi) GetAdminToken(c *gin.Context) { | ||||||
| 	a2r.Call(auth.AuthClient.UserToken, o.Client, c) | 	a2r.Call(auth.AuthClient.GetAdminToken, o.Client, c) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (o *AuthApi) GetUserToken(c *gin.Context) { | 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) { | func (o *ConversationApi) GetOwnerConversation(c *gin.Context) { | ||||||
| 	a2r.Call(conversation.ConversationClient.GetOwnerConversation, o.Client, c) | 	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) | 	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) { | func (o *FriendApi) RemoveBlack(c *gin.Context) { | ||||||
| 	a2r.Call(relation.FriendClient.RemoveBlack, o.Client, c) | 	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) | 	a2r.Call(group.GroupClient.SetGroupInfo, o.Client, c) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (o *GroupApi) SetGroupInfoEX(c *gin.Context) { | func (o *GroupApi) SetGroupInfoEx(c *gin.Context) { | ||||||
| 	a2r.Call(group.GroupClient.SetGroupInfoEX, o.Client, c) | 	a2r.Call(group.GroupClient.SetGroupInfoEx, o.Client, c) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (o *GroupApi) JoinGroup(c *gin.Context) { | 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) | 	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) { | func (o *GroupApi) GetGroupsInfo(c *gin.Context) { | ||||||
| 	a2r.Call(group.GroupClient.GetGroupsInfo, o.Client, c) | 	a2r.Call(group.GroupClient.GetGroupsInfo, o.Client, c) | ||||||
| 	//a2r.Call(group.GroupClient.GetGroupsInfo, o.Client, c, a2r.NewNilReplaceOption(group.GroupClient.GetGroupsInfo)) | 	//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 ( | import ( | ||||||
| 	"fmt" | 	"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" | ||||||
| 	"github.com/gin-gonic/gin/binding" | 	"github.com/gin-gonic/gin/binding" | ||||||
| 	"github.com/go-playground/validator/v10" | 	"github.com/go-playground/validator/v10" | ||||||
| @ -22,6 +26,13 @@ import ( | |||||||
| 	"github.com/openimsdk/tools/mw" | 	"github.com/openimsdk/tools/mw" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | const ( | ||||||
|  | 	NoCompression      = -1 | ||||||
|  | 	DefaultCompression = 0 | ||||||
|  | 	BestCompression    = 1 | ||||||
|  | 	BestSpeed          = 2 | ||||||
|  | ) | ||||||
|  | 
 | ||||||
| func prommetricsGin() gin.HandlerFunc { | func prommetricsGin() gin.HandlerFunc { | ||||||
| 	return func(c *gin.Context) { | 	return func(c *gin.Context) { | ||||||
| 		c.Next() | 		c.Next() | ||||||
| @ -54,10 +65,19 @@ func newGinRouter(disCov discovery.SvcDiscoveryRegistry, config *Config) *gin.En | |||||||
| 	conversationRpc := rpcclient.NewConversation(disCov, config.Share.RpcRegisterName.Conversation) | 	conversationRpc := rpcclient.NewConversation(disCov, config.Share.RpcRegisterName.Conversation) | ||||||
| 	authRpc := rpcclient.NewAuth(disCov, config.Share.RpcRegisterName.Auth) | 	authRpc := rpcclient.NewAuth(disCov, config.Share.RpcRegisterName.Auth) | ||||||
| 	thirdRpc := rpcclient.NewThird(disCov, config.Share.RpcRegisterName.Third, config.API.Prometheus.GrafanaURL) | 	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)) | 	r.Use(prommetricsGin(), gin.Recovery(), mw.CorsHandler(), mw.GinParseOperationID(), GinParseToken(authRpc)) | ||||||
| 	u := NewUserApi(*userRpc) | 	u := NewUserApi(*userRpc) | ||||||
| 	m := NewMessageApi(messageRpc, userRpc, config.Share.IMAdminUserID) | 	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 := r.Group("/user") | ||||||
| 	{ | 	{ | ||||||
| 		userRouterGroup.POST("/user_register", u.UserRegister) | 		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("/set_friend_remark", f.SetFriendRemark) | ||||||
| 		friendRouterGroup.POST("/add_black", f.AddBlack) | 		friendRouterGroup.POST("/add_black", f.AddBlack) | ||||||
| 		friendRouterGroup.POST("/get_black_list", f.GetPaginationBlacks) | 		friendRouterGroup.POST("/get_black_list", f.GetPaginationBlacks) | ||||||
|  | 		friendRouterGroup.POST("/get_specified_blacks", f.GetSpecifiedBlacks) | ||||||
| 		friendRouterGroup.POST("/remove_black", f.RemoveBlack) | 		friendRouterGroup.POST("/remove_black", f.RemoveBlack) | ||||||
| 		friendRouterGroup.POST("/get_incremental_blacks", f.GetIncrementalBlacks) | 		friendRouterGroup.POST("/get_incremental_blacks", f.GetIncrementalBlacks) | ||||||
| 		friendRouterGroup.POST("/import_friend", f.ImportFriends) | 		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("/create_group", g.CreateGroup) | ||||||
| 		groupRouterGroup.POST("/set_group_info", g.SetGroupInfo) | 		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("/join_group", g.JoinGroup) | ||||||
| 		groupRouterGroup.POST("/quit_group", g.QuitGroup) | 		groupRouterGroup.POST("/quit_group", g.QuitGroup) | ||||||
| 		groupRouterGroup.POST("/group_application_response", g.ApplicationGroupResponse) | 		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_recv_group_applicationList", g.GetRecvGroupApplicationList) | ||||||
| 		groupRouterGroup.POST("/get_user_req_group_applicationList", g.GetUserReqGroupApplicationList) | 		groupRouterGroup.POST("/get_user_req_group_applicationList", g.GetUserReqGroupApplicationList) | ||||||
| 		groupRouterGroup.POST("/get_group_users_req_application_list", g.GetGroupUsersReqApplicationList) | 		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("/get_groups_info", g.GetGroupsInfo) | ||||||
| 		groupRouterGroup.POST("/kick_group", g.KickGroupMember) | 		groupRouterGroup.POST("/kick_group", g.KickGroupMember) | ||||||
| 		groupRouterGroup.POST("/get_group_members_info", g.GetGroupMembersInfo) | 		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") | 	authRouterGroup := r.Group("/auth") | ||||||
| 	{ | 	{ | ||||||
| 		a := NewAuthApi(*authRpc) | 		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("/get_user_token", a.GetUserToken) | ||||||
| 		authRouterGroup.POST("/parse_token", a.ParseToken) | 		authRouterGroup.POST("/parse_token", a.ParseToken) | ||||||
| 		authRouterGroup.POST("/force_logout", a.ForceLogout) | 		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_full_conversation_ids", c.GetFullOwnerConversationIDs) | ||||||
| 		conversationGroup.POST("/get_incremental_conversations", c.GetIncrementalConversation) | 		conversationGroup.POST("/get_incremental_conversations", c.GetIncrementalConversation) | ||||||
| 		conversationGroup.POST("/get_owner_conversation", c.GetOwnerConversation) | 		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") | 	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/create", g.GroupCreateCount) | ||||||
| 		statisticsGroup.POST("/group/active", m.GetActiveGroup) | 		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 | 	return r | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -259,7 +288,6 @@ func GinParseToken(authRPC *rpcclient.Auth) gin.HandlerFunc { | |||||||
| 
 | 
 | ||||||
| // Whitelist api not parse token | // Whitelist api not parse token | ||||||
| var Whitelist = []string{ | var Whitelist = []string{ | ||||||
| 	"/user/user_register", | 	"/auth/get_admin_token", | ||||||
| 	"/auth/user_token", |  | ||||||
| 	"/auth/parse_token", | 	"/auth/parse_token", | ||||||
| } | } | ||||||
|  | |||||||
| @ -107,14 +107,14 @@ func (u *UserApi) GetUsersOnlineStatus(c *gin.Context) { | |||||||
| 			if v2.UserID == v1 { | 			if v2.UserID == v1 { | ||||||
| 				flag = true | 				flag = true | ||||||
| 				res.UserID = v1 | 				res.UserID = v1 | ||||||
| 				res.Status = constant.OnlineStatus | 				res.Status = constant.Online | ||||||
| 				res.DetailPlatformStatus = append(res.DetailPlatformStatus, v2.DetailPlatformStatus...) | 				res.DetailPlatformStatus = append(res.DetailPlatformStatus, v2.DetailPlatformStatus...) | ||||||
| 				break | 				break | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		if !flag { | 		if !flag { | ||||||
| 			res.UserID = v1 | 			res.UserID = v1 | ||||||
| 			res.Status = constant.OfflineStatus | 			res.Status = constant.Offline | ||||||
| 		} | 		} | ||||||
| 		respResult = append(respResult, res) | 		respResult = append(respResult, res) | ||||||
| 	} | 	} | ||||||
| @ -153,26 +153,26 @@ func (u *UserApi) GetUsersOnlineTokenDetail(c *gin.Context) { | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	for _, v1 := range req.UserIDs { | 	for _, v1 := range req.UserIDs { | ||||||
| 		m := make(map[string][]string, 10) | 		m := make(map[int32][]string, 10) | ||||||
| 		flag = false | 		flag = false | ||||||
| 		temp := new(msggateway.SingleDetail) | 		temp := new(msggateway.SingleDetail) | ||||||
| 		for _, v2 := range wsResult { | 		for _, v2 := range wsResult { | ||||||
| 			if v2.UserID == v1 { | 			if v2.UserID == v1 { | ||||||
| 				flag = true | 				flag = true | ||||||
| 				temp.UserID = v1 | 				temp.UserID = v1 | ||||||
| 				temp.Status = constant.OnlineStatus | 				temp.Status = constant.Online | ||||||
| 				for _, status := range v2.DetailPlatformStatus { | 				for _, status := range v2.DetailPlatformStatus { | ||||||
| 					if v, ok := m[status.Platform]; ok { | 					if v, ok := m[status.PlatformID]; ok { | ||||||
| 						m[status.Platform] = append(v, status.Token) | 						m[status.PlatformID] = append(v, status.Token) | ||||||
| 					} else { | 					} else { | ||||||
| 						m[status.Platform] = []string{status.Token} | 						m[status.PlatformID] = []string{status.Token} | ||||||
| 					} | 					} | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		for p, tokens := range m { | 		for p, tokens := range m { | ||||||
| 			t := new(msggateway.SinglePlatformToken) | 			t := new(msggateway.SinglePlatformToken) | ||||||
| 			t.Platform = p | 			t.PlatformID = p | ||||||
| 			t.Token = tokens | 			t.Token = tokens | ||||||
| 			t.Total = int32(len(tokens)) | 			t.Total = int32(len(tokens)) | ||||||
| 			temp.SinglePlatformToken = append(temp.SinglePlatformToken, t) | 			temp.SinglePlatformToken = append(temp.SinglePlatformToken, t) | ||||||
|  | |||||||
| @ -22,6 +22,8 @@ import ( | |||||||
| 	"sync/atomic" | 	"sync/atomic" | ||||||
| 	"time" | 	"time" | ||||||
| 
 | 
 | ||||||
|  | 	"google.golang.org/protobuf/proto" | ||||||
|  | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/msgprocessor" | 	"github.com/openimsdk/open-im-server/v3/pkg/msgprocessor" | ||||||
| 	"github.com/openimsdk/protocol/constant" | 	"github.com/openimsdk/protocol/constant" | ||||||
| 	"github.com/openimsdk/protocol/sdkws" | 	"github.com/openimsdk/protocol/sdkws" | ||||||
| @ -30,7 +32,6 @@ import ( | |||||||
| 	"github.com/openimsdk/tools/log" | 	"github.com/openimsdk/tools/log" | ||||||
| 	"github.com/openimsdk/tools/mcontext" | 	"github.com/openimsdk/tools/mcontext" | ||||||
| 	"github.com/openimsdk/tools/utils/stringutil" | 	"github.com/openimsdk/tools/utils/stringutil" | ||||||
| 	"google.golang.org/protobuf/proto" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| var ( | var ( | ||||||
| @ -220,6 +221,10 @@ func (c *Client) handleMessage(message []byte) error { | |||||||
| 		resp, messageErr = c.longConnServer.SendSignalMessage(ctx, binaryReq) | 		resp, messageErr = c.longConnServer.SendSignalMessage(ctx, binaryReq) | ||||||
| 	case WSPullMsgBySeqList: | 	case WSPullMsgBySeqList: | ||||||
| 		resp, messageErr = c.longConnServer.PullMessageBySeqList(ctx, binaryReq) | 		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: | 	case WsLogoutMsg: | ||||||
| 		resp, messageErr = c.longConnServer.UserLogout(ctx, binaryReq) | 		resp, messageErr = c.longConnServer.UserLogout(ctx, binaryReq) | ||||||
| 	case WsSetBackgroundStatus: | 	case WsSetBackgroundStatus: | ||||||
|  | |||||||
| @ -39,6 +39,8 @@ const ( | |||||||
| 	WSPullMsgBySeqList    = 1002 | 	WSPullMsgBySeqList    = 1002 | ||||||
| 	WSSendMsg             = 1003 | 	WSSendMsg             = 1003 | ||||||
| 	WSSendSignalMsg       = 1004 | 	WSSendSignalMsg       = 1004 | ||||||
|  | 	WSPullMsg             = 1005 | ||||||
|  | 	WSGetConvMaxReadSeq   = 1006 | ||||||
| 	WSPushMsg             = 2001 | 	WSPushMsg             = 2001 | ||||||
| 	WSKickOnlineMsg       = 2002 | 	WSKickOnlineMsg       = 2002 | ||||||
| 	WsLogoutMsg           = 2003 | 	WsLogoutMsg           = 2003 | ||||||
|  | |||||||
| @ -66,12 +66,16 @@ func (c *UserConnContext) Value(key any) any { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func newContext(respWriter http.ResponseWriter, req *http.Request) *UserConnContext { | 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{ | 	return &UserConnContext{ | ||||||
| 		RespWriter: respWriter, | 		RespWriter: respWriter, | ||||||
| 		Req:        req, | 		Req:        req, | ||||||
| 		Path:       req.URL.Path, | 		Path:       req.URL.Path, | ||||||
| 		Method:     req.Method, | 		Method:     req.Method, | ||||||
| 		RemoteAddr: req.RemoteAddr, | 		RemoteAddr: remoteAddr, | ||||||
| 		ConnID:     encrypt.Md5(req.RemoteAddr + "_" + strconv.Itoa(int(timeutil.GetCurrentTimestampByMill()))), | 		ConnID:     encrypt.Md5(req.RemoteAddr + "_" + strconv.Itoa(int(timeutil.GetCurrentTimestampByMill()))), | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  | |||||||
| @ -16,6 +16,8 @@ package msggateway | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 	"sync/atomic" | ||||||
|  | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | 	"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/servererrs" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/startrpc" | 	"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/mq/memamq" | ||||||
| 	"github.com/openimsdk/tools/utils/datautil" | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
| 	"google.golang.org/grpc" | 	"google.golang.org/grpc" | ||||||
| 	"sync/atomic" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func (s *Server) InitServer(ctx context.Context, config *Config, disCov discovery.SvcDiscoveryRegistry, server *grpc.Server) error { | 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 := new(msggateway.GetUsersOnlineStatusResp_SuccessDetail) | ||||||
| 			ps.Platform = constant.PlatformIDToName(client.PlatformID) | 			ps.PlatformID = int32(client.PlatformID) | ||||||
| 			ps.Status = constant.OnlineStatus |  | ||||||
| 			ps.ConnID = client.ctx.GetConnID() | 			ps.ConnID = client.ctx.GetConnID() | ||||||
| 			ps.Token = client.token | 			ps.Token = client.token | ||||||
| 			ps.IsBackground = client.IsBackground | 			ps.IsBackground = client.IsBackground | ||||||
| 			uresp.Status = constant.OnlineStatus | 			uresp.Status = constant.Online | ||||||
| 			uresp.DetailPlatformStatus = append(uresp.DetailPlatformStatus, ps) | 			uresp.DetailPlatformStatus = append(uresp.DetailPlatformStatus, ps) | ||||||
| 		} | 		} | ||||||
| 		if uresp.Status == constant.OnlineStatus { | 		if uresp.Status == constant.Online { | ||||||
| 			resp.SuccessResult = append(resp.SuccessResult, uresp) | 			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 { | 	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 | 		return nil | ||||||
| 	}) | 	}) | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -19,6 +19,8 @@ import ( | |||||||
| 	"sync" | 	"sync" | ||||||
| 
 | 
 | ||||||
| 	"github.com/go-playground/validator/v10" | 	"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/common/config" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcclient" | 	"github.com/openimsdk/open-im-server/v3/pkg/rpcclient" | ||||||
| 	"github.com/openimsdk/protocol/msg" | 	"github.com/openimsdk/protocol/msg" | ||||||
| @ -27,7 +29,6 @@ import ( | |||||||
| 	"github.com/openimsdk/tools/discovery" | 	"github.com/openimsdk/tools/discovery" | ||||||
| 	"github.com/openimsdk/tools/errs" | 	"github.com/openimsdk/tools/errs" | ||||||
| 	"github.com/openimsdk/tools/utils/jsonutil" | 	"github.com/openimsdk/tools/utils/jsonutil" | ||||||
| 	"google.golang.org/protobuf/proto" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| type Req struct { | type Req struct { | ||||||
| @ -94,6 +95,8 @@ type MessageHandler interface { | |||||||
| 	SendMessage(context context.Context, data *Req) ([]byte, error) | 	SendMessage(context context.Context, data *Req) ([]byte, error) | ||||||
| 	SendSignalMessage(context context.Context, data *Req) ([]byte, error) | 	SendSignalMessage(context context.Context, data *Req) ([]byte, error) | ||||||
| 	PullMessageBySeqList(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) | 	UserLogout(context context.Context, data *Req) ([]byte, error) | ||||||
| 	SetUserDeviceBackground(context context.Context, data *Req) ([]byte, bool, 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) { | func (g GrpcHandler) PullMessageBySeqList(context context.Context, data *Req) ([]byte, error) { | ||||||
| 	req := sdkws.PullMessageBySeqsReq{} | 	req := sdkws.PullMessageBySeqsReq{} | ||||||
| 	if err := proto.Unmarshal(data.Data, &req); err != nil { | 	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 { | 	if err := g.validate.Struct(data); err != nil { | ||||||
| 		return nil, errs.WrapMsg(err, "validation failed", "action", "validate", "dataType", "PullMessageBySeqsReq") | 		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 | 	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) { | func (g GrpcHandler) UserLogout(context context.Context, data *Req) ([]byte, error) { | ||||||
| 	req := push.DelUserPushTokenReq{} | 	req := push.DelUserPushTokenReq{} | ||||||
| 	if err := proto.Unmarshal(data.Data, &req); err != nil { | 	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 { | 		if _, err := ws.userClient.Client.SetUserOnlineStatus(ctx, req); err != nil { | ||||||
| 			log.ZError(ctx, "update user online status", err) | 			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++ { | 	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 | package msggateway | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| @ -212,7 +198,6 @@ func (ws *WsServer) sendUserOnlineInfoToOtherNode(ctx context.Context, client *C | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 
 |  | ||||||
| 	wg := errgroup.Group{} | 	wg := errgroup.Group{} | ||||||
| 	wg.SetLimit(concurrentRequest) | 	wg.SetLimit(concurrentRequest) | ||||||
| 
 | 
 | ||||||
| @ -265,7 +250,7 @@ func (ws *WsServer) registerClient(client *Client) { | |||||||
| 		if clientOK { | 		if clientOK { | ||||||
| 			ws.clients.Set(client.UserID, client) | 			ws.clients.Set(client.UserID, client) | ||||||
| 			// There is already a connection to the platform | 			// 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)) | 				client.PlatformID, "old remote addr", getRemoteAdders(oldClients)) | ||||||
| 			ws.onlineUserConnNum.Add(1) | 			ws.onlineUserConnNum.Add(1) | ||||||
| 		} else { | 		} else { | ||||||
| @ -293,7 +278,7 @@ func (ws *WsServer) registerClient(client *Client) { | |||||||
| 
 | 
 | ||||||
| 	wg.Wait() | 	wg.Wait() | ||||||
| 
 | 
 | ||||||
| 	log.ZInfo( | 	log.ZDebug( | ||||||
| 		client.ctx, | 		client.ctx, | ||||||
| 		"user online", | 		"user online", | ||||||
| 		"online user Num", | 		"online user Num", | ||||||
| @ -321,8 +306,32 @@ func (ws *WsServer) KickUserConn(client *Client) error { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (ws *WsServer) multiTerminalLoginChecker(clientOK bool, oldClients []*Client, newClient *Client) { | 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.DefalutNotKick: | ||||||
|  | 	case constant.WebAndOther: | ||||||
|  | 		if constant.PlatformIDToClass(newClient.PlatformID) == constant.WebPlatformStr { | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 		fallthrough | ||||||
| 	case constant.PCAndOther: | 	case constant.PCAndOther: | ||||||
| 		if constant.PlatformIDToClass(newClient.PlatformID) == constant.TerminalPC { | 		if constant.PlatformIDToClass(newClient.PlatformID) == constant.TerminalPC { | ||||||
| 			return | 			return | ||||||
| @ -347,6 +356,35 @@ func (ws *WsServer) multiTerminalLoginChecker(clientOK bool, oldClients []*Clien | |||||||
| 			log.ZWarn(newClient.ctx, "InvalidateToken err", err, "userID", newClient.UserID, | 			log.ZWarn(newClient.ctx, "InvalidateToken err", err, "userID", newClient.UserID, | ||||||
| 				"platformID", newClient.PlatformID) | 				"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.onlineUserConnNum.Add(-1) | ||||||
| 	ws.subscription.DelClient(client) | 	ws.subscription.DelClient(client) | ||||||
| 	//ws.SetUserOnlineStatus(client.ctx, client, constant.Offline) | 	//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.onlineUserNum.Load(), "online user conn Num", | ||||||
| 		ws.onlineUserConnNum.Load(), | 		ws.onlineUserConnNum.Load(), | ||||||
| 	) | 	) | ||||||
| @ -425,6 +463,7 @@ func (ws *WsServer) wsHandler(w http.ResponseWriter, r *http.Request) { | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	log.ZDebug(connContext, "new conn", "token", connContext.GetToken()) | ||||||
| 	// Create a WebSocket long connection object | 	// Create a WebSocket long connection object | ||||||
| 	wsLongConn := newGWebSocket(WebSocket, ws.handshakeTimeout, ws.writeBufferSize) | 	wsLongConn := newGWebSocket(WebSocket, ws.handshakeTimeout, ws.writeBufferSize) | ||||||
| 	if err := wsLongConn.GenerateLongConn(w, r); err != nil { | 	if err := wsLongConn.GenerateLongConn(w, r); err != nil { | ||||||
|  | |||||||
| @ -18,19 +18,20 @@ import ( | |||||||
| 	"context" | 	"context" | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"fmt" | 	"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/prommetrics" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache/redis" | 	"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/open-im-server/v3/pkg/common/storage/database/mgo" | ||||||
| 	"github.com/openimsdk/tools/db/mongoutil" | 	"github.com/openimsdk/tools/db/mongoutil" | ||||||
| 	"github.com/openimsdk/tools/db/redisutil" | 	"github.com/openimsdk/tools/db/redisutil" | ||||||
| 	"github.com/openimsdk/tools/utils/datautil" | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
| 	"net/http" |  | ||||||
| 	"os" |  | ||||||
| 	"os/signal" |  | ||||||
| 	"syscall" |  | ||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | 	"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/common/storage/controller" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcclient" | 	"github.com/openimsdk/open-im-server/v3/pkg/rpcclient" | ||||||
| 	"github.com/openimsdk/tools/errs" | 	"github.com/openimsdk/tools/errs" | ||||||
| @ -65,6 +66,7 @@ type Config struct { | |||||||
| func Start(ctx context.Context, index int, config *Config) error { | func Start(ctx context.Context, index int, config *Config) error { | ||||||
| 	log.CInfo(ctx, "MSG-TRANSFER server is initializing", "prometheusPorts", | 	log.CInfo(ctx, "MSG-TRANSFER server is initializing", "prometheusPorts", | ||||||
| 		config.MsgTransfer.Prometheus.Ports, "index", index) | 		config.MsgTransfer.Prometheus.Ports, "index", index) | ||||||
|  | 
 | ||||||
| 	mgocli, err := mongoutil.NewMongoDB(ctx, config.MongodbConfig.Build()) | 	mgocli, err := mongoutil.NewMongoDB(ctx, config.MongodbConfig.Build()) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| @ -73,12 +75,13 @@ func Start(ctx context.Context, index int, config *Config) error { | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	client, err := kdisc.NewDiscoveryRegister(&config.Discovery, &config.Share) | 	client, err := discRegister.NewDiscoveryRegister(&config.Discovery, &config.Share) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	client.AddOption(mw.GrpcClient(), grpc.WithTransportCredentials(insecure.NewCredentials()), | 	client.AddOption(mw.GrpcClient(), grpc.WithTransportCredentials(insecure.NewCredentials()), | ||||||
| 		grpc.WithDefaultServiceConfig(fmt.Sprintf(`{"LoadBalancingPolicy": "%s"}`, "round_robin"))) | 		grpc.WithDefaultServiceConfig(fmt.Sprintf(`{"LoadBalancingPolicy": "%s"}`, "round_robin"))) | ||||||
|  | 
 | ||||||
| 	msgModel := redis.NewMsgCache(rdb) | 	msgModel := redis.NewMsgCache(rdb) | ||||||
| 	msgDocModel, err := mgo.NewMsgMongo(mgocli.GetDB()) | 	msgDocModel, err := mgo.NewMsgMongo(mgocli.GetDB()) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -94,20 +97,21 @@ func Start(ctx context.Context, index int, config *Config) error { | |||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	seqUserCache := redis.NewSeqUserCacheRedis(rdb, seqUser) | 	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 { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	conversationRpcClient := rpcclient.NewConversationRpcClient(client, config.Share.RpcRegisterName.Conversation) | 	conversationRpcClient := rpcclient.NewConversationRpcClient(client, config.Share.RpcRegisterName.Conversation) | ||||||
| 	groupRpcClient := rpcclient.NewGroupRpcClient(client, config.Share.RpcRegisterName.Group) | 	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 { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	historyMongoCH, err := NewOnlineHistoryMongoConsumerHandler(&config.KafkaConfig, msgDatabase) | 	historyMongoCH, err := NewOnlineHistoryMongoConsumerHandler(&config.KafkaConfig, msgTransferDatabase) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	msgTransfer := &MsgTransfer{ | 	msgTransfer := &MsgTransfer{ | ||||||
| 		historyCH:      historyCH, | 		historyCH:      historyCH, | ||||||
| 		historyMongoCH: historyMongoCH, | 		historyMongoCH: historyMongoCH, | ||||||
|  | |||||||
| @ -18,6 +18,10 @@ import ( | |||||||
| 	"context" | 	"context" | ||||||
| 	"encoding/json" | 	"encoding/json" | ||||||
| 	"errors" | 	"errors" | ||||||
|  | 	"strconv" | ||||||
|  | 	"strings" | ||||||
|  | 	"time" | ||||||
|  | 
 | ||||||
| 	"github.com/IBM/sarama" | 	"github.com/IBM/sarama" | ||||||
| 	"github.com/go-redis/redis" | 	"github.com/go-redis/redis" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | 	"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/mq/kafka" | ||||||
| 	"github.com/openimsdk/tools/utils/stringutil" | 	"github.com/openimsdk/tools/utils/stringutil" | ||||||
| 	"google.golang.org/protobuf/proto" | 	"google.golang.org/protobuf/proto" | ||||||
| 	"strconv" |  | ||||||
| 	"strings" |  | ||||||
| 	"time" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| const ( | const ( | ||||||
| @ -56,19 +57,19 @@ type OnlineHistoryRedisConsumerHandler struct { | |||||||
| 
 | 
 | ||||||
| 	redisMessageBatches *batcher.Batcher[sarama.ConsumerMessage] | 	redisMessageBatches *batcher.Batcher[sarama.ConsumerMessage] | ||||||
| 
 | 
 | ||||||
| 	msgDatabase           controller.CommonMsgDatabase | 	msgTransferDatabase   controller.MsgTransferDatabase | ||||||
| 	conversationRpcClient *rpcclient.ConversationRpcClient | 	conversationRpcClient *rpcclient.ConversationRpcClient | ||||||
| 	groupRpcClient        *rpcclient.GroupRpcClient | 	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) { | 	conversationRpcClient *rpcclient.ConversationRpcClient, groupRpcClient *rpcclient.GroupRpcClient) (*OnlineHistoryRedisConsumerHandler, error) { | ||||||
| 	historyConsumerGroup, err := kafka.NewMConsumerGroup(kafkaConf.Build(), kafkaConf.ToRedisGroupID, []string{kafkaConf.ToRedisTopic}, false) | 	historyConsumerGroup, err := kafka.NewMConsumerGroup(kafkaConf.Build(), kafkaConf.ToRedisGroupID, []string{kafkaConf.ToRedisTopic}, false) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	var och OnlineHistoryRedisConsumerHandler | 	var och OnlineHistoryRedisConsumerHandler | ||||||
| 	och.msgDatabase = database | 	och.msgTransferDatabase = database | ||||||
| 
 | 
 | ||||||
| 	b := batcher.New[sarama.ConsumerMessage]( | 	b := batcher.New[sarama.ConsumerMessage]( | ||||||
| 		batcher.WithSize(size), | 		batcher.WithSize(size), | ||||||
| @ -161,7 +162,7 @@ func (och *OnlineHistoryRedisConsumerHandler) doSetReadSeq(ctx context.Context, | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	for key, seq := range readSeq { | 	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) | 			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) { | 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) | 	och.toPushTopic(ctx, key, conversationID, notStorageList) | ||||||
| 	var storageMessageList []*sdkws.MsgData | 	var storageMessageList []*sdkws.MsgData | ||||||
| 	for _, msg := range storageList { | 	for _, msg := range storageList { | ||||||
| @ -244,21 +250,25 @@ func (och *OnlineHistoryRedisConsumerHandler) handleMsg(ctx context.Context, key | |||||||
| 	} | 	} | ||||||
| 	if len(storageMessageList) > 0 { | 	if len(storageMessageList) > 0 { | ||||||
| 		msg := 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) { | 		if err != nil && !errors.Is(errs.Unwrap(err), redis.Nil) { | ||||||
| 			log.ZError(ctx, "batch data insert to redis err", err, "storageMsgList", storageMessageList) | 			log.ZError(ctx, "batch data insert to redis err", err, "storageMsgList", storageMessageList) | ||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
|  | 		log.ZInfo(ctx, "BatchInsertChat2Cache end") | ||||||
|  | 
 | ||||||
| 		if isNewConversation { | 		if isNewConversation { | ||||||
| 			switch msg.SessionType { | 			switch msg.SessionType { | ||||||
| 			case constant.ReadGroupChatType: | 			case constant.ReadGroupChatType: | ||||||
| 				log.ZInfo(ctx, "group chat first create conversation", "conversationID", | 				log.ZDebug(ctx, "group chat first create conversation", "conversationID", | ||||||
| 					conversationID) | 					conversationID) | ||||||
| 				userIDs, err := och.groupRpcClient.GetGroupMemberIDs(ctx, msg.GroupID) | 				userIDs, err := och.groupRpcClient.GetGroupMemberIDs(ctx, msg.GroupID) | ||||||
| 				if err != nil { | 				if err != nil { | ||||||
| 					log.ZWarn(ctx, "get group member ids error", err, "conversationID", | 					log.ZWarn(ctx, "get group member ids error", err, "conversationID", | ||||||
| 						conversationID) | 						conversationID) | ||||||
| 				} else { | 				} else { | ||||||
|  | 					log.ZInfo(ctx, "GetGroupMemberIDs end") | ||||||
|  | 
 | ||||||
| 					if err := och.conversationRpcClient.GroupChatFirstCreateConversation(ctx, | 					if err := och.conversationRpcClient.GroupChatFirstCreateConversation(ctx, | ||||||
| 						msg.GroupID, userIDs); err != nil { | 						msg.GroupID, userIDs); err != nil { | ||||||
| 						log.ZWarn(ctx, "single chat first create conversation error", err, | 						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") | 		log.ZInfo(ctx, "success incr to next topic") | ||||||
| 		err = och.msgDatabase.MsgToMongoMQ(ctx, key, conversationID, storageMessageList, lastSeq) | 		err = och.msgTransferDatabase.MsgToMongoMQ(ctx, key, conversationID, storageMessageList, lastSeq) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.ZError(ctx, "Msg To MongoDB MQ error", err, "conversationID", | 			log.ZError(ctx, "Msg To MongoDB MQ error", err, "conversationID", | ||||||
| 				conversationID, "storageList", storageMessageList, "lastSeq", lastSeq) | 				conversationID, "storageList", storageMessageList, "lastSeq", lastSeq) | ||||||
| 		} | 		} | ||||||
|  | 		log.ZInfo(ctx, "MsgToMongoMQ end") | ||||||
|  | 
 | ||||||
| 		och.toPushTopic(ctx, key, conversationID, storageList) | 		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) | 		storageMessageList = append(storageMessageList, msg.message) | ||||||
| 	} | 	} | ||||||
| 	if len(storageMessageList) > 0 { | 	if len(storageMessageList) > 0 { | ||||||
| 		lastSeq, _, err := och.msgDatabase.BatchInsertChat2Cache(ctx, conversationID, storageMessageList) | 		lastSeq, _, err := och.msgTransferDatabase.BatchInsertChat2Cache(ctx, conversationID, storageMessageList) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.ZError(ctx, "notification batch insert to redis error", err, "conversationID", conversationID, | 			log.ZError(ctx, "notification batch insert to redis error", err, "conversationID", conversationID, | ||||||
| 				"storageList", storageMessageList) | 				"storageList", storageMessageList) | ||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
| 		log.ZDebug(ctx, "success to next topic", "conversationID", conversationID) | 		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 { | 		if err != nil { | ||||||
| 			log.ZError(ctx, "Msg To MongoDB MQ error", err, "conversationID", | 			log.ZError(ctx, "Msg To MongoDB MQ error", err, "conversationID", | ||||||
| 				conversationID, "storageList", storageMessageList, "lastSeq", lastSeq) | 				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 { | 	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, | func (och *OnlineHistoryRedisConsumerHandler) ConsumeClaim(session sarama.ConsumerGroupSession, | ||||||
| 	claim sarama.ConsumerGroupClaim) error { // a instance in the consumer group | 	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()) | 		claim.HighWaterMarkOffset(), "topic", claim.Topic(), "partition", claim.Partition()) | ||||||
| 	och.redisMessageBatches.OnComplete = func(lastMessage *sarama.ConsumerMessage, totalCount int) { | 	och.redisMessageBatches.OnComplete = func(lastMessage *sarama.ConsumerMessage, totalCount int) { | ||||||
| 		session.MarkMessage(lastMessage, "") | 		session.MarkMessage(lastMessage, "") | ||||||
|  | |||||||
| @ -16,6 +16,7 @@ package msgtransfer | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 
 | ||||||
| 	"github.com/IBM/sarama" | 	"github.com/IBM/sarama" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/prommetrics" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/prommetrics" | ||||||
| @ -28,10 +29,10 @@ import ( | |||||||
| 
 | 
 | ||||||
| type OnlineHistoryMongoConsumerHandler struct { | type OnlineHistoryMongoConsumerHandler struct { | ||||||
| 	historyConsumerGroup *kafka.MConsumerGroup | 	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) | 	historyConsumerGroup, err := kafka.NewMConsumerGroup(kafkaConf.Build(), kafkaConf.ToMongoGroupID, []string{kafkaConf.ToMongoTopic}, true) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -39,7 +40,7 @@ func NewOnlineHistoryMongoConsumerHandler(kafkaConf *config.Kafka, database cont | |||||||
| 
 | 
 | ||||||
| 	mc := &OnlineHistoryMongoConsumerHandler{ | 	mc := &OnlineHistoryMongoConsumerHandler{ | ||||||
| 		historyConsumerGroup: historyConsumerGroup, | 		historyConsumerGroup: historyConsumerGroup, | ||||||
| 		msgDatabase:          database, | 		msgTransferDatabase:  database, | ||||||
| 	} | 	} | ||||||
| 	return mc, nil | 	return mc, nil | ||||||
| } | } | ||||||
| @ -56,8 +57,8 @@ func (mc *OnlineHistoryMongoConsumerHandler) handleChatWs2Mongo(ctx context.Cont | |||||||
| 		log.ZError(ctx, "msgFromMQ.MsgData is empty", nil, "cMsg", cMsg) | 		log.ZError(ctx, "msgFromMQ.MsgData is empty", nil, "cMsg", cMsg) | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	log.ZInfo(ctx, "mongo consumer recv msg", "msgs", msgFromMQ.String()) | 	log.ZDebug(ctx, "mongo consumer recv msg", "msgs", msgFromMQ.String()) | ||||||
| 	err = mc.msgDatabase.BatchInsertChat2DB(ctx, msgFromMQ.ConversationID, msgFromMQ.MsgData, msgFromMQ.LastSeq) | 	err = mc.msgTransferDatabase.BatchInsertChat2DB(ctx, msgFromMQ.ConversationID, msgFromMQ.MsgData, msgFromMQ.LastSeq) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.ZError( | 		log.ZError( | ||||||
| 			ctx, | 			ctx, | ||||||
| @ -76,7 +77,7 @@ func (mc *OnlineHistoryMongoConsumerHandler) handleChatWs2Mongo(ctx context.Cont | |||||||
| 	for _, msg := range msgFromMQ.MsgData { | 	for _, msg := range msgFromMQ.MsgData { | ||||||
| 		seqs = append(seqs, msg.Seq) | 		seqs = append(seqs, msg.Seq) | ||||||
| 	} | 	} | ||||||
| 	err = mc.msgDatabase.DeleteMessagesFromCache(ctx, msgFromMQ.ConversationID, seqs) | 	err = mc.msgTransferDatabase.DeleteMessagesFromCache(ctx, msgFromMQ.ConversationID, seqs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.ZError( | 		log.ZError( | ||||||
| 			ctx, | 			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 ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/internal/push/offlinepush/options" | 	"github.com/openimsdk/open-im-server/v3/internal/push/offlinepush/options" | ||||||
|  | 	"github.com/openimsdk/tools/log" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func NewClient() *Dummy { | 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 { | func (d *Dummy) Push(ctx context.Context, userIDs []string, title, content string, opts *options.Opts) error { | ||||||
|  | 	log.ZDebug(ctx, "dummy push") | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  | |||||||
| @ -22,8 +22,8 @@ import ( | |||||||
| 	"path/filepath" | 	"path/filepath" | ||||||
| 	"strings" | 	"strings" | ||||||
| 
 | 
 | ||||||
| 	firebase "firebase.google.com/go" | 	firebase "firebase.google.com/go/v4" | ||||||
| 	"firebase.google.com/go/messaging" | 	"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/config" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache" | ||||||
| 	"github.com/openimsdk/protocol/constant" | 	"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}}} | 		apns := &messaging.APNSConfig{Payload: &messaging.APNSPayload{Aps: &messaging.Aps{Sound: opts.IOSPushSound}}} | ||||||
| 		messageCount := len(messages) | 		messageCount := len(messages) | ||||||
| 		if messageCount >= SinglePushCountLimit { | 		if messageCount >= SinglePushCountLimit { | ||||||
| 			response, err := f.fcmMsgCli.SendAll(ctx, messages) | 			response, err := f.fcmMsgCli.SendEach(ctx, messages) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				Fail = Fail + messageCount | 				Fail = Fail + messageCount | ||||||
| 				// Record push error | 				// Record push error | ||||||
| @ -154,7 +154,7 @@ func (f *Fcm) Push(ctx context.Context, userIDs []string, title, content string, | |||||||
| 	} | 	} | ||||||
| 	messageCount := len(messages) | 	messageCount := len(messages) | ||||||
| 	if messageCount > 0 { | 	if messageCount > 0 { | ||||||
| 		response, err := f.fcmMsgCli.SendAll(ctx, messages) | 		response, err := f.fcmMsgCli.SendEach(ctx, messages) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			Fail = Fail + messageCount | 			Fail = Fail + messageCount | ||||||
| 		} else { | 		} else { | ||||||
|  | |||||||
| @ -18,11 +18,11 @@ import ( | |||||||
| 	"context" | 	"context" | ||||||
| 	"crypto/sha256" | 	"crypto/sha256" | ||||||
| 	"encoding/hex" | 	"encoding/hex" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/internal/push/offlinepush/options" |  | ||||||
| 	"strconv" | 	"strconv" | ||||||
| 	"sync" | 	"sync" | ||||||
| 	"time" | 	"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/config" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache" | ||||||
| 	"github.com/openimsdk/tools/errs" | 	"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() { | 			for i, v := range s.GetSplitResult() { | ||||||
| 				go func(index int, userIDs []string) { | 				go func(index int, userIDs []string) { | ||||||
| 					defer wg.Done() | 					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 { | 					if err = g.batchPush(ctx, token, userIDs, pushReq); err != nil { | ||||||
| 						log.ZError(ctx, "batchPush failed", err, "index", index, "token", token, "req", pushReq) | 						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/internal/push/offlinepush/options" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache" | ||||||
|  | 	"github.com/openimsdk/tools/log" | ||||||
|  | 	"github.com/openimsdk/tools/mcontext" | ||||||
|  | 	"strings" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| const ( | const ( | ||||||
| 	geTUI    = "geTui" | 	geTUI    = "getui" | ||||||
| 	firebase = "fcm" | 	firebase = "fcm" | ||||||
| 	jPush    = "jpush" | 	jPush    = "jpush" | ||||||
| ) | ) | ||||||
| @ -38,6 +41,7 @@ type OfflinePusher interface { | |||||||
| 
 | 
 | ||||||
| func NewOfflinePusher(pushConf *config.Push, cache cache.ThirdCache, fcmConfigPath string) (OfflinePusher, error) { | func NewOfflinePusher(pushConf *config.Push, cache cache.ThirdCache, fcmConfigPath string) (OfflinePusher, error) { | ||||||
| 	var offlinePusher OfflinePusher | 	var offlinePusher OfflinePusher | ||||||
|  | 	pushConf.Enable = strings.ToLower(pushConf.Enable) | ||||||
| 	switch pushConf.Enable { | 	switch pushConf.Enable { | ||||||
| 	case geTUI: | 	case geTUI: | ||||||
| 		offlinePusher = getui.NewClient(pushConf, cache) | 		offlinePusher = getui.NewClient(pushConf, cache) | ||||||
| @ -47,6 +51,7 @@ func NewOfflinePusher(pushConf *config.Push, cache cache.ThirdCache, fcmConfigPa | |||||||
| 		offlinePusher = jpush.NewClient(pushConf) | 		offlinePusher = jpush.NewClient(pushConf) | ||||||
| 	default: | 	default: | ||||||
| 		offlinePusher = dummy.NewClient() | 		offlinePusher = dummy.NewClient() | ||||||
|  | 		log.ZWarn(mcontext.WithMustInfoCtx([]string{"push start", "admin", "admin", ""}), "Unknown push config", nil) | ||||||
| 	} | 	} | ||||||
| 	return offlinePusher, 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, | func (emptyOnlinePusher) GetConnsAndOnlinePush(ctx context.Context, msg *sdkws.MsgData, | ||||||
| 	pushToUserIDs []string) (wsResults []*msggateway.SingleMsgToUserResults, err error) { | 	pushToUserIDs []string) (wsResults []*msggateway.SingleMsgToUserResults, err error) { | ||||||
| 	log.ZWarn(ctx, "emptyOnlinePusher GetConnsAndOnlinePush", nil) | 	log.ZInfo(ctx, "emptyOnlinePusher GetConnsAndOnlinePush", nil) | ||||||
| 	return nil, nil | 	return nil, nil | ||||||
| } | } | ||||||
| func (u emptyOnlinePusher) GetOnlinePushFailedUserIDs(ctx context.Context, msg *sdkws.MsgData, | func (u emptyOnlinePusher) GetOnlinePushFailedUserIDs(ctx context.Context, msg *sdkws.MsgData, | ||||||
| 	wsResults []*msggateway.SingleMsgToUserResults, pushToUserIDs *[]string) []string { | 	wsResults []*msggateway.SingleMsgToUserResults, pushToUserIDs *[]string) []string { | ||||||
| 	log.ZWarn(ctx, "emptyOnlinePusher GetOnlinePushFailedUserIDs", nil) | 	log.ZInfo(ctx, "emptyOnlinePusher GetOnlinePushFailedUserIDs", nil) | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -2,6 +2,7 @@ package push | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/internal/push/offlinepush" | 	"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/config" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache/redis" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache/redis" | ||||||
| @ -17,12 +18,12 @@ type pushServer struct { | |||||||
| 	disCov        discovery.SvcDiscoveryRegistry | 	disCov        discovery.SvcDiscoveryRegistry | ||||||
| 	offlinePusher offlinepush.OfflinePusher | 	offlinePusher offlinepush.OfflinePusher | ||||||
| 	pushCh        *ConsumerHandler | 	pushCh        *ConsumerHandler | ||||||
|  | 	offlinePushCh *OfflinePushConsumerHandler | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| type Config struct { | type Config struct { | ||||||
| 	RpcConfig          config.Push | 	RpcConfig          config.Push | ||||||
| 	RedisConfig        config.Redis | 	RedisConfig        config.Redis | ||||||
| 	MongodbConfig      config.Mongo |  | ||||||
| 	KafkaConfig        config.Kafka | 	KafkaConfig        config.Kafka | ||||||
| 	NotificationConfig config.Notification | 	NotificationConfig config.Notification | ||||||
| 	Share              config.Share | 	Share              config.Share | ||||||
| @ -55,18 +56,30 @@ func Start(ctx context.Context, config *Config, client discovery.SvcDiscoveryReg | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		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 { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
|  | 	offlinePushConsumer, err := NewOfflinePushConsumerHandler(config, offlinePusher) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	pbpush.RegisterPushMsgServiceServer(server, &pushServer{ | 	pbpush.RegisterPushMsgServiceServer(server, &pushServer{ | ||||||
| 		database:      database, | 		database:      database, | ||||||
| 		disCov:        client, | 		disCov:        client, | ||||||
| 		offlinePusher: offlinePusher, | 		offlinePusher: offlinePusher, | ||||||
| 		pushCh:        consumer, | 		pushCh:        consumer, | ||||||
|  | 		offlinePushCh: offlinePushConsumer, | ||||||
| 	}) | 	}) | ||||||
|  | 
 | ||||||
| 	go consumer.pushConsumerGroup.RegisterHandleAndConsumer(ctx, consumer) | 	go consumer.pushConsumerGroup.RegisterHandleAndConsumer(ctx, consumer) | ||||||
|  | 
 | ||||||
|  | 	go offlinePushConsumer.OfflinePushConsumerGroup.RegisterHandleAndConsumer(ctx, offlinePushConsumer) | ||||||
|  | 
 | ||||||
| 	return nil | 	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 | package push | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
| 	"encoding/json" | 	"encoding/json" | ||||||
|  | 
 | ||||||
| 	"github.com/IBM/sarama" | 	"github.com/IBM/sarama" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/internal/push/offlinepush" | 	"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/internal/push/offlinepush/options" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/prommetrics" | 	"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/common/webhook" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/msgprocessor" | 	"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/rpccache" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcclient" | 	"github.com/openimsdk/open-im-server/v3/pkg/rpcclient" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/util/conversationutil" | 	"github.com/openimsdk/open-im-server/v3/pkg/util/conversationutil" | ||||||
| 	"github.com/openimsdk/protocol/constant" | 	"github.com/openimsdk/protocol/constant" | ||||||
| 	pbchat "github.com/openimsdk/protocol/msg" |  | ||||||
| 	"github.com/openimsdk/protocol/msggateway" | 	"github.com/openimsdk/protocol/msggateway" | ||||||
| 	pbpush "github.com/openimsdk/protocol/push" | 	pbpush "github.com/openimsdk/protocol/push" | ||||||
| 	"github.com/openimsdk/protocol/sdkws" | 	"github.com/openimsdk/protocol/sdkws" | ||||||
| @ -40,12 +27,16 @@ import ( | |||||||
| 	"github.com/openimsdk/tools/utils/timeutil" | 	"github.com/openimsdk/tools/utils/timeutil" | ||||||
| 	"github.com/redis/go-redis/v9" | 	"github.com/redis/go-redis/v9" | ||||||
| 	"google.golang.org/protobuf/proto" | 	"google.golang.org/protobuf/proto" | ||||||
|  | 	"math/rand" | ||||||
|  | 	"strconv" | ||||||
|  | 	"time" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| type ConsumerHandler struct { | type ConsumerHandler struct { | ||||||
| 	pushConsumerGroup      *kafka.MConsumerGroup | 	pushConsumerGroup      *kafka.MConsumerGroup | ||||||
| 	offlinePusher          offlinepush.OfflinePusher | 	offlinePusher          offlinepush.OfflinePusher | ||||||
| 	onlinePusher           OnlinePusher | 	onlinePusher           OnlinePusher | ||||||
|  | 	pushDatabase           controller.PushDatabase | ||||||
| 	onlineCache            *rpccache.OnlineCache | 	onlineCache            *rpccache.OnlineCache | ||||||
| 	groupLocalCache        *rpccache.GroupLocalCache | 	groupLocalCache        *rpccache.GroupLocalCache | ||||||
| 	conversationLocalCache *rpccache.ConversationLocalCache | 	conversationLocalCache *rpccache.ConversationLocalCache | ||||||
| @ -56,7 +47,7 @@ type ConsumerHandler struct { | |||||||
| 	config                 *Config | 	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) { | 	client discovery.SvcDiscoveryRegistry) (*ConsumerHandler, error) { | ||||||
| 	var consumerHandler ConsumerHandler | 	var consumerHandler ConsumerHandler | ||||||
| 	var err error | 	var err error | ||||||
| @ -65,7 +56,9 @@ func NewConsumerHandler(config *Config, offlinePusher offlinepush.OfflinePusher, | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	userRpcClient := rpcclient.NewUserRpcClient(client, config.Share.RpcRegisterName.User, config.Share.IMAdminUserID) | 	userRpcClient := rpcclient.NewUserRpcClient(client, config.Share.RpcRegisterName.User, config.Share.IMAdminUserID) | ||||||
|  | 
 | ||||||
| 	consumerHandler.offlinePusher = offlinePusher | 	consumerHandler.offlinePusher = offlinePusher | ||||||
| 	consumerHandler.onlinePusher = NewOnlinePusher(client, config) | 	consumerHandler.onlinePusher = NewOnlinePusher(client, config) | ||||||
| 	consumerHandler.groupRpcClient = rpcclient.NewGroupRpcClient(client, config.Share.RpcRegisterName.Group) | 	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.conversationLocalCache = rpccache.NewConversationLocalCache(consumerHandler.conversationRpcClient, &config.LocalCacheConfig, rdb) | ||||||
| 	consumerHandler.webhookClient = webhook.NewWebhookClient(config.WebhooksConfig.URL) | 	consumerHandler.webhookClient = webhook.NewWebhookClient(config.WebhooksConfig.URL) | ||||||
| 	consumerHandler.config = config | 	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 | 	return &consumerHandler, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *ConsumerHandler) handleMs2PsChat(ctx context.Context, msg []byte) { | func (c *ConsumerHandler) handleMs2PsChat(ctx context.Context, msg []byte) { | ||||||
| 	msgFromMQ := pbchat.PushMsgDataToMQ{} | 	msgFromMQ := pbpush.PushMsgReq{} | ||||||
| 	if err := proto.Unmarshal(msg, &msgFromMQ); err != nil { | 	if err := proto.Unmarshal(msg, &msgFromMQ); err != nil { | ||||||
| 		log.ZError(ctx, "push Unmarshal msg err", err, "msg", string(msg)) | 		log.ZError(ctx, "push Unmarshal msg err", err, "msg", string(msg)) | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	pbData := &pbpush.PushMsgReq{ | 
 | ||||||
| 		MsgData:        msgFromMQ.MsgData, |  | ||||||
| 		ConversationID: msgFromMQ.ConversationID, |  | ||||||
| 	} |  | ||||||
| 	sec := msgFromMQ.MsgData.SendTime / 1000 | 	sec := msgFromMQ.MsgData.SendTime / 1000 | ||||||
| 	nowSec := timeutil.GetCurrentTimestampBySecond() | 	nowSec := timeutil.GetCurrentTimestampBySecond() | ||||||
| 
 | 
 | ||||||
| 	if nowSec-sec > 10 { | 	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 | 	var err error | ||||||
|  | 
 | ||||||
| 	switch msgFromMQ.MsgData.SessionType { | 	switch msgFromMQ.MsgData.SessionType { | ||||||
| 	case constant.ReadGroupChatType: | 	case constant.ReadGroupChatType: | ||||||
| 		err = c.Push2Group(ctx, pbData.MsgData.GroupID, pbData.MsgData) | 		err = c.Push2Group(ctx, msgFromMQ.MsgData.GroupID, msgFromMQ.MsgData) | ||||||
| 	default: | 	default: | ||||||
| 		var pushUserIDList []string | 		var pushUserIDList []string | ||||||
| 		isSenderSync := datautil.GetSwitchFromOptions(pbData.MsgData.Options, constant.IsSenderSync) | 		isSenderSync := datautil.GetSwitchFromOptions(msgFromMQ.MsgData.Options, constant.IsSenderSync) | ||||||
| 		if !isSenderSync || pbData.MsgData.SendID == pbData.MsgData.RecvID { | 		if !isSenderSync || msgFromMQ.MsgData.SendID == msgFromMQ.MsgData.RecvID { | ||||||
| 			pushUserIDList = append(pushUserIDList, pbData.MsgData.RecvID) | 			pushUserIDList = append(pushUserIDList, msgFromMQ.MsgData.RecvID) | ||||||
| 		} else { | 		} 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 { | 	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 (*ConsumerHandler) Cleanup(sarama.ConsumerGroupSession) error { return nil } | ||||||
| 
 | 
 | ||||||
| func (c *ConsumerHandler) ConsumeClaim(sess sarama.ConsumerGroupSession, claim sarama.ConsumerGroupClaim) error { | 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() { | 	for msg := range claim.Messages() { | ||||||
| 		ctx := c.pushConsumerGroup.GetContextFromMsg(msg) | 		ctx := c.pushConsumerGroup.GetContextFromMsg(msg) | ||||||
| 		c.handleMs2PsChat(ctx, msg.Value) | 		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. | // 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) { | 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 { | 	if err := c.webhookBeforeOnlinePush(ctx, &c.config.WebhooksConfig.BeforeOnlinePush, userIDs, msg); err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  | 	log.ZInfo(ctx, "webhookBeforeOnlinePush end") | ||||||
|  | 
 | ||||||
| 	wsResults, err := c.GetConnsAndOnlinePush(ctx, msg, userIDs) | 	wsResults, err := c.GetConnsAndOnlinePush(ctx, msg, userIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		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) { | 	if !c.shouldPushOffline(ctx, msg) { | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
|  | 	log.ZInfo(ctx, "shouldPushOffline end") | ||||||
| 
 | 
 | ||||||
| 	for _, v := range wsResults { | 	for _, v := range wsResults { | ||||||
| 		//message sender do not need offline push | 		//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 { | 		offlinePushUserID, msg, nil); err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 
 | 	log.ZInfo(ctx, "webhookBeforeOfflinePush end") | ||||||
| 	err = c.offlinePushMsg(ctx, msg, offlinePushUserID) | 	err = c.offlinePushMsg(ctx, msg, offlinePushUserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.ZWarn(ctx, "offlinePushMsg failed", err, "offlinePushUserID", offlinePushUserID, "msg", msg) | 		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) { | func (c *ConsumerHandler) GetConnsAndOnlinePush(ctx context.Context, msg *sdkws.MsgData, pushToUserIDs []string) ([]*msggateway.SingleMsgToUserResults, error) { | ||||||
| 	var ( | 	if msg != nil && msg.Status == constant.MsgStatusSending { | ||||||
| 		onlineUserIDs  []string | 		msg.Status = constant.MsgStatusSendSuccess | ||||||
| 		offlineUserIDs []string |  | ||||||
| 	) |  | ||||||
| 	for _, userID := range pushToUserIDs { |  | ||||||
| 		online, err := c.onlineCache.GetUserOnline(ctx, userID) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return nil, err |  | ||||||
| 		} |  | ||||||
| 		if online { |  | ||||||
| 			onlineUserIDs = append(onlineUserIDs, userID) |  | ||||||
| 		} else { |  | ||||||
| 			offlineUserIDs = append(offlineUserIDs, userID) |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
|  | 	onlineUserIDs, offlineUserIDs, err := c.onlineCache.GetUsersOnline(ctx, pushToUserIDs) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	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) | 	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 | 	var result []*msggateway.SingleMsgToUserResults | ||||||
| 	if len(onlineUserIDs) > 0 { | 	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) { | 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 | 	var pushToUserIDs []string | ||||||
| 	if err = c.webhookBeforeGroupOnlinePush(ctx, &c.config.WebhooksConfig.BeforeGroupOnlinePush, groupID, msg, | 	if err = c.webhookBeforeGroupOnlinePush(ctx, &c.config.WebhooksConfig.BeforeGroupOnlinePush, groupID, msg, | ||||||
| 		&pushToUserIDs); err != nil { | 		&pushToUserIDs); err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  | 	log.ZInfo(ctx, "webhookBeforeGroupOnlinePush end") | ||||||
| 
 | 
 | ||||||
| 	err = c.groupMessagesHandler(ctx, groupID, &pushToUserIDs, msg) | 	err = c.groupMessagesHandler(ctx, groupID, &pushToUserIDs, msg) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  | 	log.ZInfo(ctx, "groupMessagesHandler end") | ||||||
| 
 | 
 | ||||||
| 	wsResults, err := c.GetConnsAndOnlinePush(ctx, msg, pushToUserIDs) | 	wsResults, err := c.GetConnsAndOnlinePush(ctx, msg, pushToUserIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		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) { | 	if !c.shouldPushOffline(ctx, msg) { | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| 	needOfflinePushUserIDs := c.onlinePusher.GetOnlinePushFailedUserIDs(ctx, msg, wsResults, &pushToUserIDs) | 	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. | 	//filter some user, like don not disturb or don't need offline push etc. | ||||||
| 	needOfflinePushUserIDs, err = c.filterGroupMessageOfflinePush(ctx, groupID, msg, needOfflinePushUserIDs) | 	needOfflinePushUserIDs, err = c.filterGroupMessageOfflinePush(ctx, groupID, msg, needOfflinePushUserIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  | 	log.ZInfo(ctx, "filterGroupMessageOfflinePush end") | ||||||
|  | 
 | ||||||
| 	// Use offline push messaging | 	// Use offline push messaging | ||||||
| 	if len(needOfflinePushUserIDs) > 0 { | 	if len(needOfflinePushUserIDs) > 0 { | ||||||
| 		var offlinePushUserIDs []string | 		c.asyncOfflinePush(ctx, needOfflinePushUserIDs, msg) | ||||||
| 		err = c.webhookBeforeOfflinePush(ctx, &c.config.WebhooksConfig.BeforeOfflinePush, needOfflinePushUserIDs, msg, &offlinePushUserIDs) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		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 |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	return nil | 	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) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.ZWarn(ctx, "webhookBeforeOfflinePush failed", err, "msg", msg) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if len(offlinePushUserIDs) > 0 { | ||||||
|  | 		needOfflinePushUserIDs = offlinePushUserIDs | ||||||
|  | 	} | ||||||
|  | 	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 | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func (c *ConsumerHandler) groupMessagesHandler(ctx context.Context, groupID string, pushToUserIDs *[]string, msg *sdkws.MsgData) (err error) { | func (c *ConsumerHandler) groupMessagesHandler(ctx context.Context, groupID string, pushToUserIDs *[]string, msg *sdkws.MsgData) (err error) { | ||||||
| 	if len(*pushToUserIDs) == 0 { | 	if len(*pushToUserIDs) == 0 { | ||||||
| 		*pushToUserIDs, err = c.groupLocalCache.GetGroupMemberIDs(ctx, groupID) | 		*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 { | 				if unmarshalNotificationElem(msg.Content, &tips) != nil { | ||||||
| 					return err | 					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 { | 				if len(c.config.Share.IMAdminUserID) > 0 { | ||||||
| 					ctx = mcontext.WithOpUserIDContext(ctx, 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 | 	return | ||||||
| } | } | ||||||
|  | 
 | ||||||
| func (c *ConsumerHandler) DeleteMemberAndSetConversationSeq(ctx context.Context, groupID string, userIDs []string) error { | func (c *ConsumerHandler) DeleteMemberAndSetConversationSeq(ctx context.Context, groupID string, userIDs []string) error { | ||||||
| 	conversationID := msgprocessor.GetConversationIDBySessionType(constant.ReadGroupChatType, groupID) | 	conversationID := msgprocessor.GetConversationIDBySessionType(constant.ReadGroupChatType, groupID) | ||||||
| 	maxSeq, err := c.msgRpcClient.GetConversationMaxSeq(ctx, conversationID) | 	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) | 	return c.conversationRpcClient.SetConversationMaxSeq(ctx, userIDs, conversationID, maxSeq) | ||||||
| } | } | ||||||
|  | 
 | ||||||
| func unmarshalNotificationElem(bytes []byte, t any) error { | func unmarshalNotificationElem(bytes []byte, t any) error { | ||||||
| 	var notification sdkws.NotificationElem | 	var notification sdkws.NotificationElem | ||||||
| 	if err := json.Unmarshal(bytes, ¬ification); err != nil { | 	if err := json.Unmarshal(bytes, ¬ification); err != nil { | ||||||
|  | |||||||
| @ -20,6 +20,7 @@ import ( | |||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||||
| 	redis2 "github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache/redis" | 	redis2 "github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache/redis" | ||||||
| 	"github.com/openimsdk/tools/db/redisutil" | 	"github.com/openimsdk/tools/db/redisutil" | ||||||
|  | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
| 	"github.com/redis/go-redis/v9" | 	"github.com/redis/go-redis/v9" | ||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | 	"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), | 			redis2.NewTokenCacheModel(rdb, config.RpcConfig.TokenPolicy.Expire), | ||||||
| 			config.Share.Secret, | 			config.Share.Secret, | ||||||
| 			config.RpcConfig.TokenPolicy.Expire, | 			config.RpcConfig.TokenPolicy.Expire, | ||||||
|  | 			config.Share.MultiLogin, | ||||||
| 		), | 		), | ||||||
| 		config: config, | 		config: config, | ||||||
| 	}) | 	}) | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *authServer) UserToken(ctx context.Context, req *pbauth.UserTokenReq) (*pbauth.UserTokenResp, error) { | func (s *authServer) GetAdminToken(ctx context.Context, req *pbauth.GetAdminTokenReq) (*pbauth.GetAdminTokenResp, error) { | ||||||
| 	resp := pbauth.UserTokenResp{} | 	resp := pbauth.GetAdminTokenResp{} | ||||||
| 	if req.Secret != s.config.Share.Secret { | 	if req.Secret != s.config.Share.Secret { | ||||||
| 		return nil, errs.ErrNoPermission.WrapMsg("secret invalid") | 		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 { | 	if _, err := s.userRpcClient.GetUserInfo(ctx, req.UserID); err != nil { | ||||||
| 		return nil, err | 		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 { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	prommetrics.UserLoginCounter.Inc() | 	prommetrics.UserLoginCounter.Inc() | ||||||
| 	resp.Token = token | 	resp.Token = token | ||||||
| 	resp.ExpireTimeSeconds = s.config.RpcConfig.TokenPolicy.Expire * 24 * 60 * 60 | 	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 { | 	if err := authverify.CheckAdmin(ctx, s.config.Share.IMAdminUserID); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
|  | 	if req.PlatformID == constant.AdminPlatformID { | ||||||
|  | 		return nil, errs.ErrNoPermission.WrapMsg("platformID invalid. platformID must not be adminPlatformID") | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	resp := pbauth.GetUserTokenResp{} | 	resp := pbauth.GetUserTokenResp{} | ||||||
| 
 | 
 | ||||||
| 	if authverify.IsManagerUserID(req.UserID, s.config.Share.IMAdminUserID) { | 	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 | 	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 { | 		if req.Conversation.MsgDestructTime != nil { | ||||||
| 			m["msg_destruct_time"] = req.Conversation.MsgDestructTime.Value | 			m["msg_destruct_time"] = req.Conversation.MsgDestructTime.Value | ||||||
| 		} | 		} | ||||||
| 		if req.Conversation.MsgDestructTime != nil { | 		if req.Conversation.IsMsgDestruct != nil { | ||||||
| 			m["msg_destruct_time"] = req.Conversation.MsgDestructTime.Value | 			m["is_msg_destruct"] = req.Conversation.IsMsgDestruct.Value | ||||||
| 		} | 		} | ||||||
| 		if req.Conversation.BurnDuration != nil { | 		if req.Conversation.BurnDuration != nil { | ||||||
| 			m["burn_duration"] = req.Conversation.BurnDuration.Value | 			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 | 	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, | 		CallbackCommand: callbackstruct.CallbackAfterKickGroupCommand, | ||||||
| 		GroupID:         req.GroupID, | 		GroupID:         req.GroupID, | ||||||
| 		KickedUserIDs:   req.KickedUserIDs, | 		KickedUserIDs:   req.KickedUserIDs, | ||||||
|  | 		Reason:          req.Reason, | ||||||
| 	} | 	} | ||||||
| 	s.webhookClient.AsyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, &callbackstruct.CallbackKillGroupMemberResp{}, after) | 	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) | 	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 { | 	return webhook.WithCondition(ctx, before, func(ctx context.Context) error { | ||||||
| 		cbReq := &callbackstruct.CallbackBeforeSetGroupInfoEXReq{ | 		cbReq := &callbackstruct.CallbackBeforeSetGroupInfoExReq{ | ||||||
| 			CallbackCommand: callbackstruct.CallbackBeforeSetGroupInfoCommand, | 			CallbackCommand: callbackstruct.CallbackBeforeSetGroupInfoExCommand, | ||||||
| 			GroupID:         req.GroupInfoForSetEX.GroupID, | 			GroupID:         req.GroupID, | ||||||
| 			GroupName:       req.GroupInfoForSetEX.GroupName, | 			GroupName:       req.GroupName, | ||||||
| 			Notification:    req.GroupInfoForSetEX.Notification, | 			Notification:    req.Notification, | ||||||
| 			Introduction:    req.GroupInfoForSetEX.Introduction, | 			Introduction:    req.Introduction, | ||||||
| 			FaceURL:         req.GroupInfoForSetEX.FaceURL, | 			FaceURL:         req.FaceURL, | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		if req.GroupInfoForSetEX.Ex != nil { | 		if req.Ex != nil { | ||||||
| 			cbReq.Ex = req.GroupInfoForSetEX.Ex | 			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 { | 		if req.NeedVerification != nil { | ||||||
| 			cbReq.NeedVerification = req.GroupInfoForSetEX.NeedVerification | 			cbReq.NeedVerification = req.NeedVerification | ||||||
| 		} | 		} | ||||||
| 		if req.GroupInfoForSetEX.LookMemberInfo != nil { | 		if req.LookMemberInfo != nil { | ||||||
| 			cbReq.LookMemberInfo = req.GroupInfoForSetEX.LookMemberInfo | 			cbReq.LookMemberInfo = req.LookMemberInfo | ||||||
| 		} | 		} | ||||||
| 		if req.GroupInfoForSetEX.ApplyMemberFriend != nil { | 		if req.ApplyMemberFriend != nil { | ||||||
| 			cbReq.ApplyMemberFriend = req.GroupInfoForSetEX.ApplyMemberFriend | 			cbReq.ApplyMemberFriend = req.ApplyMemberFriend | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		resp := &callbackstruct.CallbackBeforeSetGroupInfoEXResp{} | 		resp := &callbackstruct.CallbackBeforeSetGroupInfoExResp{} | ||||||
| 
 | 
 | ||||||
| 		if err := s.webhookClient.SyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, resp, before); err != nil { | 		if err := s.webhookClient.SyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, resp, before); err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		datautil.NotNilReplace(&req.GroupInfoForSetEX.GroupID, &resp.GroupID) | 		datautil.NotNilReplace(&req.GroupID, &resp.GroupID) | ||||||
| 		datautil.NotNilReplace(&req.GroupInfoForSetEX.GroupName, &resp.GroupName) | 		datautil.NotNilReplace(&req.GroupName, &resp.GroupName) | ||||||
| 		datautil.NotNilReplace(&req.GroupInfoForSetEX.FaceURL, &resp.FaceURL) | 		datautil.NotNilReplace(&req.FaceURL, &resp.FaceURL) | ||||||
| 		datautil.NotNilReplace(&req.GroupInfoForSetEX.Introduction, &resp.Introduction) | 		datautil.NotNilReplace(&req.Introduction, &resp.Introduction) | ||||||
| 		datautil.NotNilReplace(&req.GroupInfoForSetEX.Ex, &resp.Ex) | 		datautil.NotNilReplace(&req.Ex, &resp.Ex) | ||||||
| 		datautil.NotNilReplace(&req.GroupInfoForSetEX.NeedVerification, &resp.NeedVerification) | 		datautil.NotNilReplace(&req.NeedVerification, &resp.NeedVerification) | ||||||
| 		datautil.NotNilReplace(&req.GroupInfoForSetEX.LookMemberInfo, &resp.LookMemberInfo) | 		datautil.NotNilReplace(&req.LookMemberInfo, &resp.LookMemberInfo) | ||||||
| 		datautil.NotNilReplace(&req.GroupInfoForSetEX.ApplyMemberFriend, &resp.ApplyMemberFriend) | 		datautil.NotNilReplace(&req.ApplyMemberFriend, &resp.ApplyMemberFriend) | ||||||
| 
 | 
 | ||||||
| 		return nil | 		return nil | ||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *groupServer) webhookAfterSetGroupInfoEX(ctx context.Context, after *config.AfterConfig, req *group.SetGroupInfoEXReq) { | func (s *groupServer) webhookAfterSetGroupInfoEx(ctx context.Context, after *config.AfterConfig, req *group.SetGroupInfoExReq) { | ||||||
| 	cbReq := &callbackstruct.CallbackAfterSetGroupInfoEXReq{ | 	cbReq := &callbackstruct.CallbackAfterSetGroupInfoExReq{ | ||||||
| 		CallbackCommand: callbackstruct.CallbackAfterSetGroupInfoCommand, | 		CallbackCommand: callbackstruct.CallbackAfterSetGroupInfoExCommand, | ||||||
| 		GroupID:         req.GroupInfoForSetEX.GroupID, | 		GroupID:         req.GroupID, | ||||||
| 		GroupName:       req.GroupInfoForSetEX.GroupName, | 		GroupName:       req.GroupName, | ||||||
| 		Notification:    req.GroupInfoForSetEX.Notification, | 		Notification:    req.Notification, | ||||||
| 		Introduction:    req.GroupInfoForSetEX.Introduction, | 		Introduction:    req.Introduction, | ||||||
| 		FaceURL:         req.GroupInfoForSetEX.FaceURL, | 		FaceURL:         req.FaceURL, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if req.GroupInfoForSetEX.Ex != nil { | 	if req.Ex != nil { | ||||||
| 		cbReq.Ex = req.GroupInfoForSetEX.Ex | 		cbReq.Ex = req.Ex | ||||||
| 	} | 	} | ||||||
| 	if req.GroupInfoForSetEX.NeedVerification != nil { | 	if req.NeedVerification != nil { | ||||||
| 		cbReq.NeedVerification = req.GroupInfoForSetEX.NeedVerification | 		cbReq.NeedVerification = req.NeedVerification | ||||||
| 	} | 	} | ||||||
| 	if req.GroupInfoForSetEX.LookMemberInfo != nil { | 	if req.LookMemberInfo != nil { | ||||||
| 		cbReq.LookMemberInfo = req.GroupInfoForSetEX.LookMemberInfo | 		cbReq.LookMemberInfo = req.LookMemberInfo | ||||||
| 	} | 	} | ||||||
| 	if req.GroupInfoForSetEX.ApplyMemberFriend != nil { | 	if req.ApplyMemberFriend != nil { | ||||||
| 		cbReq.ApplyMemberFriend = req.GroupInfoForSetEX.ApplyMemberFriend | 		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" | 	pbgroup "github.com/openimsdk/protocol/group" | ||||||
| 	"github.com/openimsdk/protocol/sdkws" | 	"github.com/openimsdk/protocol/sdkws" | ||||||
|  | 	"github.com/openimsdk/tools/errs" | ||||||
| 	"github.com/openimsdk/tools/mcontext" | 	"github.com/openimsdk/tools/mcontext" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| @ -54,11 +55,15 @@ func UpdateGroupInfoMap(ctx context.Context, group *sdkws.GroupInfoForSet) map[s | |||||||
| 	return m | 	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) | 	m := make(map[string]any) | ||||||
| 
 | 
 | ||||||
| 	if group.GroupName != "" { | 	if group.GroupName != nil { | ||||||
| 		m["group_name"] = group.GroupName | 		if group.GroupName.Value != "" { | ||||||
|  | 			m["group_name"] = group.GroupName.Value | ||||||
|  | 		} else { | ||||||
|  | 			return nil, errs.ErrArgs.WrapMsg("group name is empty") | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 	if group.Notification != nil { | 	if group.Notification != nil { | ||||||
| 		m["notification"] = group.Notification.Value | 		m["notification"] = group.Notification.Value | ||||||
| @ -84,7 +89,7 @@ func UpdateGroupInfoEXMap(ctx context.Context, group *sdkws.GroupInfoForSetEX) m | |||||||
| 		m["ex"] = group.Ex.Value | 		m["ex"] = group.Ex.Value | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	return m | 	return m, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func UpdateGroupStatusMap(status int) map[string]any { | func UpdateGroupStatusMap(status int) map[string]any { | ||||||
|  | |||||||
| @ -167,11 +167,11 @@ func (g *groupServer) CheckGroupAdmin(ctx context.Context, groupID string) error | |||||||
| 	return nil | 	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 { | 	if len(userIDs) == 0 { | ||||||
| 		return map[string]*sdkws.PublicUserInfo{}, nil | 		return map[string]*sdkws.PublicUserInfo{}, nil | ||||||
| 	} | 	} | ||||||
| 	users, err := g.user.GetPublicUserInfos(ctx, userIDs, complete) | 	users, err := g.user.GetPublicUserInfos(ctx, userIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -304,6 +304,13 @@ func (g *groupServer) CreateGroup(ctx context.Context, req *pbgroup.CreateGroupR | |||||||
| 	} | 	} | ||||||
| 	g.notification.GroupCreatedNotification(ctx, tips) | 	g.notification.GroupCreatedNotification(ctx, tips) | ||||||
| 
 | 
 | ||||||
|  | 	if req.GroupInfo.Notification != "" { | ||||||
|  | 		g.notification.GroupInfoSetAnnouncementNotification(ctx, &sdkws.GroupInfoSetAnnouncementTips{ | ||||||
|  | 			Group:  tips.Group, | ||||||
|  | 			OpUser: tips.OpUser, | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	reqCallBackAfter := &pbgroup.CreateGroupReq{ | 	reqCallBackAfter := &pbgroup.CreateGroupReq{ | ||||||
| 		MemberUserIDs: userIDs, | 		MemberUserIDs: userIDs, | ||||||
| 		GroupInfo:     resp.GroupInfo, | 		GroupInfo:     resp.GroupInfo, | ||||||
| @ -458,7 +465,7 @@ func (g *groupServer) InviteUserToGroup(ctx context.Context, req *pbgroup.Invite | |||||||
| 		return nil, err | 		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 nil, err | ||||||
| 	} | 	} | ||||||
| 	return &pbgroup.InviteUserToGroupResp{}, nil | 	return &pbgroup.InviteUserToGroupResp{}, nil | ||||||
| @ -689,7 +696,7 @@ func (g *groupServer) GetGroupApplicationList(ctx context.Context, req *pbgroup. | |||||||
| 		userIDs = append(userIDs, gr.UserID) | 		userIDs = append(userIDs, gr.UserID) | ||||||
| 	} | 	} | ||||||
| 	userIDs = datautil.Distinct(userIDs) | 	userIDs = datautil.Distinct(userIDs) | ||||||
| 	userMap, err := g.user.GetPublicUserInfoMap(ctx, userIDs, true) | 	userMap, err := g.user.GetPublicUserInfoMap(ctx, userIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -833,7 +840,7 @@ func (g *groupServer) GroupApplicationResponse(ctx context.Context, req *pbgroup | |||||||
| 		if member == nil { | 		if member == nil { | ||||||
| 			log.ZDebug(ctx, "GroupApplicationResponse", "member is nil") | 			log.ZDebug(ctx, "GroupApplicationResponse", "member is nil") | ||||||
| 		} else { | 		} 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 | 				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}) | 			resp, err := g.GetGroupMemberUserIDs(ctx, &pbgroup.GetGroupMemberUserIDsReq{GroupID: req.GroupInfoForSet.GroupID}) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				log.ZWarn(ctx, "GetGroupMemberIDs", err) | 				log.ZWarn(ctx, "GetGroupMemberIDs is failed.", err) | ||||||
| 				return | 				return | ||||||
| 			} | 			} | ||||||
| 			conversation.GroupAtType = &wrapperspb.Int32Value{Value: constant.GroupNotification} | 			conversation.GroupAtType = &wrapperspb.Int32Value{Value: constant.GroupNotification} | ||||||
| 			if err := g.conversationRpcClient.SetConversations(ctx, resp.UserIDs, conversation); err != nil { | 			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}) | 		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 | 	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 | 	var opMember *model.GroupMember | ||||||
| 
 | 
 | ||||||
| 	if !authverify.IsAppManagerUid(ctx, g.config.Share.IMAdminUserID) { | 	if !authverify.IsAppManagerUid(ctx, g.config.Share.IMAdminUserID) { | ||||||
| 		var err error | 		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 { | 		if err != nil { | ||||||
| 			return nil, err | 			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 | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	group, err := g.db.TakeGroup(ctx, req.GroupInfoForSetEX.GroupID) | 	group, err := g.db.TakeGroup(ctx, req.GroupID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -1097,16 +1104,20 @@ func (g *groupServer) SetGroupInfoEX(ctx context.Context, req *pbgroup.SetGroupI | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	updatedData := UpdateGroupInfoEXMap(ctx, req.GroupInfoForSetEX) | 	updatedData, err := UpdateGroupInfoExMap(ctx, req) | ||||||
| 	if len(updatedData) == 0 { | 	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 { | 	if err := g.db.UpdateGroup(ctx, group.GroupID, updatedData); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	group, err = g.db.TakeGroup(ctx, req.GroupInfoForSetEX.GroupID) | 	group, err = g.db.TakeGroup(ctx, req.GroupID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -1122,43 +1133,46 @@ func (g *groupServer) SetGroupInfoEX(ctx context.Context, req *pbgroup.SetGroupI | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	num := len(updatedData) | 	num := len(updatedData) | ||||||
| 	if req.GroupInfoForSetEX.Notification != nil { | 	if req.Notification != nil { | ||||||
| 		num-- | 		num-- | ||||||
| 
 | 
 | ||||||
| 		func() { | 		if req.Notification.Value != "" { | ||||||
| 			conversation := &pbconversation.ConversationReq{ | 			func() { | ||||||
| 				ConversationID:   msgprocessor.GetConversationIDBySessionType(constant.ReadGroupChatType, req.GroupInfoForSetEX.GroupID), | 				conversation := &pbconversation.ConversationReq{ | ||||||
| 				ConversationType: constant.ReadGroupChatType, | 					ConversationID:   msgprocessor.GetConversationIDBySessionType(constant.ReadGroupChatType, req.GroupID), | ||||||
| 				GroupID:          req.GroupInfoForSetEX.GroupID, | 					ConversationType: constant.ReadGroupChatType, | ||||||
| 			} | 					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 { | 				if err != nil { | ||||||
| 				log.ZWarn(ctx, "GetGroupMemberIDs", err) | 					log.ZWarn(ctx, "GetGroupMemberIDs is failed.", err) | ||||||
| 				return | 					return | ||||||
| 			} | 				} | ||||||
| 
 | 
 | ||||||
| 			conversation.GroupAtType = &wrapperspb.Int32Value{Value: constant.GroupNotification} | 				conversation.GroupAtType = &wrapperspb.Int32Value{Value: constant.GroupNotification} | ||||||
| 
 | 
 | ||||||
| 			if err := g.conversationRpcClient.SetConversations(ctx, resp.UserIDs, conversation); err != nil { | 				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}) | 			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}) | 		g.notification.GroupInfoSetNameNotification(ctx, &sdkws.GroupInfoSetNameTips{Group: tips.Group, OpUser: tips.OpUser}) | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	if num > 0 { | 	if num > 0 { | ||||||
| 		g.notification.GroupInfoSetNotification(ctx, tips) | 		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) { | 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") | 		return nil, errs.ErrNoPermission.WrapMsg("no op user id") | ||||||
| 	} | 	} | ||||||
| 	isAppManagerUid := authverify.IsAppManagerUid(ctx, g.config.Share.IMAdminUserID) | 	isAppManagerUid := authverify.IsAppManagerUid(ctx, g.config.Share.IMAdminUserID) | ||||||
| 	for i := range req.Members { |  | ||||||
| 		req.Members[i].FaceURL = nil |  | ||||||
| 	} |  | ||||||
| 	groupMembers := make(map[string][]*pbgroup.SetGroupMemberInfo) | 	groupMembers := make(map[string][]*pbgroup.SetGroupMemberInfo) | ||||||
| 	for i, member := range req.Members { | 	for i, member := range req.Members { | ||||||
| 		if member.RoleLevel != nil { | 		if member.RoleLevel != nil { | ||||||
| @ -1515,29 +1526,61 @@ func (g *groupServer) SetGroupMemberInfo(ctx context.Context, req *pbgroup.SetGr | |||||||
| 		case 0: | 		case 0: | ||||||
| 			if !isAppManagerUid { | 			if !isAppManagerUid { | ||||||
| 				roleLevel := dbMembers[opUserIndex].RoleLevel | 				roleLevel := dbMembers[opUserIndex].RoleLevel | ||||||
| 				if roleLevel != constant.GroupOwner { | 				var ( | ||||||
| 					switch roleLevel { | 					dbSelf  = &model.GroupMember{} | ||||||
| 					case constant.GroupAdmin: | 					reqSelf *pbgroup.SetGroupMemberInfo | ||||||
| 						for _, member := range dbMembers { | 				) | ||||||
| 							if member.RoleLevel == constant.GroupOwner { | 				switch roleLevel { | ||||||
| 								return nil, errs.ErrNoPermission.WrapMsg("admin can not change group owner") | 				case constant.GroupOwner: | ||||||
| 							} | 					for _, member := range dbMembers { | ||||||
| 							if member.RoleLevel == constant.GroupAdmin && member.UserID != opUserID { | 						if member.UserID == opUserID { | ||||||
| 								return nil, errs.ErrNoPermission.WrapMsg("admin can not change other group admin") | 							dbSelf = member | ||||||
| 							} | 							break | ||||||
| 						} | 						} | ||||||
| 					case constant.GroupOrdinaryUsers: | 					} | ||||||
| 						for _, member := range dbMembers { | 				case constant.GroupAdmin: | ||||||
| 							if !(member.RoleLevel == constant.GroupOrdinaryUsers && member.UserID == opUserID) { | 					for _, member := range dbMembers { | ||||||
| 								return nil, errs.ErrNoPermission.WrapMsg("ordinary users can not change other role level") | 						if member.UserID == opUserID { | ||||||
| 							} | 							dbSelf = member | ||||||
| 						} | 						} | ||||||
| 					default: | 						if member.RoleLevel == constant.GroupOwner { | ||||||
| 						for _, member := range dbMembers { | 							return nil, errs.ErrNoPermission.WrapMsg("admin can not change group owner") | ||||||
| 							if member.RoleLevel >= roleLevel { |  | ||||||
| 								return nil, errs.ErrNoPermission.WrapMsg("can not change higher role level") |  | ||||||
| 							} |  | ||||||
| 						} | 						} | ||||||
|  | 						if member.RoleLevel == constant.GroupAdmin && member.UserID != opUserID { | ||||||
|  | 							return nil, errs.ErrNoPermission.WrapMsg("admin can not change other group admin") | ||||||
|  | 						} | ||||||
|  | 					} | ||||||
|  | 				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 | ||||||
| 					} | 					} | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| @ -1578,7 +1621,7 @@ func (g *groupServer) SetGroupMemberInfo(ctx context.Context, req *pbgroup.SetGr | |||||||
| 				g.notification.GroupMemberSetToOrdinaryUserNotification(ctx, member.GroupID, member.UserID) | 				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) | 			g.notification.GroupMemberInfoSetNotification(ctx, member.GroupID, member.UserID) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @ -1671,36 +1714,51 @@ func (g *groupServer) GetGroupUsersReqApplicationList(ctx context.Context, req * | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	if len(requests) == 0 { | 	if len(requests) == 0 { | ||||||
| 		return &pbgroup.GetGroupUsersReqApplicationListResp{}, nil | 		return &pbgroup.GetGroupUsersReqApplicationListResp{}, nil | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	groupIDs := datautil.Distinct(datautil.Slice(requests, func(e *model.GroupRequest) string { | 	groupIDs := datautil.Distinct(datautil.Slice(requests, func(e *model.GroupRequest) string { | ||||||
| 		return e.GroupID | 		return e.GroupID | ||||||
| 	})) | 	})) | ||||||
|  | 
 | ||||||
| 	groups, err := g.db.FindGroup(ctx, groupIDs) | 	groups, err := g.db.FindGroup(ctx, groupIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	groupMap := datautil.SliceToMap(groups, func(e *model.Group) string { | 	groupMap := datautil.SliceToMap(groups, func(e *model.Group) string { | ||||||
| 		return e.GroupID | 		return e.GroupID | ||||||
| 	}) | 	}) | ||||||
|  | 
 | ||||||
| 	if ids := datautil.Single(groupIDs, datautil.Keys(groupMap)); len(ids) > 0 { | 	if ids := datautil.Single(groupIDs, datautil.Keys(groupMap)); len(ids) > 0 { | ||||||
| 		return nil, servererrs.ErrGroupIDNotFound.WrapMsg(strings.Join(ids, ",")) | 		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) | 	owners, err := g.db.FindGroupsOwner(ctx, groupIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	if err := g.PopulateGroupMember(ctx, owners...); err != nil { | 	if err := g.PopulateGroupMember(ctx, owners...); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	ownerMap := datautil.SliceToMap(owners, func(e *model.GroupMember) string { | 	ownerMap := datautil.SliceToMap(owners, func(e *model.GroupMember) string { | ||||||
| 		return e.GroupID | 		return e.GroupID | ||||||
| 	}) | 	}) | ||||||
|  | 
 | ||||||
| 	groupMemberNum, err := g.db.MapGroupMemberNum(ctx, groupIDs) | 	groupMemberNum, err := g.db.MapGroupMemberNum(ctx, groupIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	return &pbgroup.GetGroupUsersReqApplicationListResp{ | 	return &pbgroup.GetGroupUsersReqApplicationListResp{ | ||||||
| 		Total: int64(len(requests)), | 		Total: int64(len(requests)), | ||||||
| 		GroupRequests: datautil.Slice(requests, func(e *model.GroupRequest) *sdkws.GroupRequest { | 		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 { | 			if owner, ok := ownerMap[e.GroupID]; ok { | ||||||
| 				ownerUserID = owner.UserID | 				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 | 	}, 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 ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 	"errors" | ||||||
| 	"fmt" | 	"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/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/database" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | 	"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/common/storage/versionctx" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/msgprocessor" | 	"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" | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/rpcclient/notification" | ||||||
| 	"github.com/openimsdk/protocol/constant" | 	"github.com/openimsdk/protocol/constant" | ||||||
| 	pbgroup "github.com/openimsdk/protocol/group" | 	pbgroup "github.com/openimsdk/protocol/group" | ||||||
|  | 	"github.com/openimsdk/protocol/msg" | ||||||
| 	"github.com/openimsdk/protocol/sdkws" | 	"github.com/openimsdk/protocol/sdkws" | ||||||
| 	"github.com/openimsdk/tools/errs" | 	"github.com/openimsdk/tools/errs" | ||||||
| 	"github.com/openimsdk/tools/log" | 	"github.com/openimsdk/tools/log" | ||||||
| 	"github.com/openimsdk/tools/mcontext" | 	"github.com/openimsdk/tools/mcontext" | ||||||
| 	"github.com/openimsdk/tools/utils/datautil" | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
| 	"github.com/openimsdk/tools/utils/stringutil" | 	"github.com/openimsdk/tools/utils/stringutil" | ||||||
|  | 	"go.mongodb.org/mongo-driver/mongo" | ||||||
|  | 	"time" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| // GroupApplicationReceiver | // 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) { | 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 { | 	if opUser == nil { | ||||||
| 		return errs.ErrInternalServer.WrapMsg("**sdkws.GroupMemberFullInfo is nil") | 		return errs.ErrInternalServer.WrapMsg("**sdkws.GroupMemberFullInfo is nil") | ||||||
| 	} | 	} | ||||||
| 	userID := mcontext.GetOpUserID(ctx) |  | ||||||
| 	if groupID != "" { | 	if groupID != "" { | ||||||
| 		if authverify.IsManagerUserID(userID, g.config.Share.IMAdminUserID) { | 		if authverify.IsManagerUserID(userID, g.config.Share.IMAdminUserID) { | ||||||
| 			*opUser = &sdkws.GroupMemberFullInfo{ | 			*opUser = &sdkws.GroupMemberFullInfo{ | ||||||
| @ -243,7 +248,7 @@ func (g *GroupNotificationSender) fillOpUser(ctx context.Context, opUser **sdkws | |||||||
| 			member, err := g.db.TakeGroupMember(ctx, groupID, userID) | 			member, err := g.db.TakeGroupMember(ctx, groupID, userID) | ||||||
| 			if err == nil { | 			if err == nil { | ||||||
| 				*opUser = g.groupMemberDB2PB(member, 0) | 				*opUser = g.groupMemberDB2PB(member, 0) | ||||||
| 			} else if !errs.ErrRecordNotFound.Is(err) { | 			} else if !(errors.Is(err, mongo.ErrNoDocuments) || errs.ErrRecordNotFound.Is(err)) { | ||||||
| 				return 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) | 	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 | 	var err error | ||||||
| 	defer func() { | 	defer func() { | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| @ -546,15 +551,75 @@ func (g *GroupNotificationSender) MemberEnterNotification(ctx context.Context, g | |||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	tips := &sdkws.MemberInvitedTips{Group: group, InvitedUserList: users} | 	tips := &sdkws.MemberInvitedTips{ | ||||||
| 	if err = g.fillOpUser(ctx, &tips.OpUser, tips.Group.GroupID); err != nil { | 		Group:           group, | ||||||
|  | 		InvitedUserList: users, | ||||||
|  | 	} | ||||||
|  | 	opUserID := mcontext.GetOpUserID(ctx) | ||||||
|  | 	if err = g.fillOpUserByUserID(ctx, opUserID, &tips.OpUser, tips.Group.GroupID); err != nil { | ||||||
| 		return 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.setVersion(ctx, &tips.GroupMemberVersion, &tips.GroupMemberVersionID, database.GroupMemberVersionName, tips.Group.GroupID) | ||||||
| 	g.Notification(ctx, mcontext.GetOpUserID(ctx), group.GroupID, constant.MemberInvitedNotification, tips) | 	g.Notification(ctx, mcontext.GetOpUserID(ctx), group.GroupID, constant.MemberInvitedNotification, tips) | ||||||
| 	return nil | 	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) { | func (g *GroupNotificationSender) GroupDismissedNotification(ctx context.Context, tips *sdkws.GroupDismissedTips) { | ||||||
| 	var err error | 	var err error | ||||||
| 	defer func() { | 	defer func() { | ||||||
|  | |||||||
| @ -55,7 +55,7 @@ func (m *msgServer) GetConversationsHasReadAndMaxSeq(ctx context.Context, req *m | |||||||
| 			conversationMaxSeqMap[conversation.ConversationID] = conversation.MaxSeq | 			conversationMaxSeqMap[conversation.ConversationID] = conversation.MaxSeq | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	maxSeqs, err := m.MsgDatabase.GetMaxSeqs(ctx, conversationIDs) | 	maxSeqs, err := m.MsgDatabase.GetMaxSeqsWithTime(ctx, conversationIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -63,7 +63,8 @@ func (m *msgServer) GetConversationsHasReadAndMaxSeq(ctx context.Context, req *m | |||||||
| 	for conversationID, maxSeq := range maxSeqs { | 	for conversationID, maxSeq := range maxSeqs { | ||||||
| 		resp.Seqs[conversationID] = &msg.Seqs{ | 		resp.Seqs[conversationID] = &msg.Seqs{ | ||||||
| 			HasReadSeq: hasReadSeqs[conversationID], | 			HasReadSeq: hasReadSeqs[conversationID], | ||||||
| 			MaxSeq:     maxSeq, | 			MaxSeq:     maxSeq.Seq, | ||||||
|  | 			MaxSeqTime: maxSeq.Time, | ||||||
| 		} | 		} | ||||||
| 		if v, ok := conversationMaxSeqMap[conversationID]; ok { | 		if v, ok := conversationMaxSeqMap[conversationID]; ok { | ||||||
| 			resp.Seqs[conversationID].MaxSeq = v | 			resp.Seqs[conversationID].MaxSeq = v | ||||||
|  | |||||||
| @ -67,6 +67,9 @@ func (m *msgServer) webhookBeforeSendSingleMsg(ctx context.Context, before *conf | |||||||
| 		if msg.MsgData.ContentType == constant.Typing { | 		if msg.MsgData.ContentType == constant.Typing { | ||||||
| 			return nil | 			return nil | ||||||
| 		} | 		} | ||||||
|  | 		if !filterBeforeMsg(msg, before) { | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
| 		cbReq := &cbapi.CallbackBeforeSendSingleMsgReq{ | 		cbReq := &cbapi.CallbackBeforeSendSingleMsgReq{ | ||||||
| 			CommonCallbackReq: toCommonCallback(ctx, msg, cbapi.CallbackBeforeSendSingleMsgCommand), | 			CommonCallbackReq: toCommonCallback(ctx, msg, cbapi.CallbackBeforeSendSingleMsgCommand), | ||||||
| 			RecvID:            msg.MsgData.RecvID, | 			RecvID:            msg.MsgData.RecvID, | ||||||
| @ -84,9 +87,7 @@ func (m *msgServer) webhookAfterSendSingleMsg(ctx context.Context, after *config | |||||||
| 	if msg.MsgData.ContentType == constant.Typing { | 	if msg.MsgData.ContentType == constant.Typing { | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	// According to the attentionIds configuration, only some users are sent | 	if !filterAfterMsg(msg, after) { | ||||||
| 	attentionIds := after.AttentionIds |  | ||||||
| 	if attentionIds != nil && !datautil.Contain(msg.MsgData.RecvID, attentionIds...) && !datautil.Contain(msg.MsgData.SendID, attentionIds...) { |  | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	cbReq := &cbapi.CallbackAfterSendSingleMsgReq{ | 	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 { | func (m *msgServer) webhookBeforeSendGroupMsg(ctx context.Context, before *config.BeforeConfig, msg *pbchat.SendMsgReq) error { | ||||||
| 	return webhook.WithCondition(ctx, before, func(ctx context.Context) error { | 	return webhook.WithCondition(ctx, before, func(ctx context.Context) error { | ||||||
|  | 		if !filterBeforeMsg(msg, before) { | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
| 		if msg.MsgData.ContentType == constant.Typing { | 		if msg.MsgData.ContentType == constant.Typing { | ||||||
| 			return nil | 			return nil | ||||||
| 		} | 		} | ||||||
| @ -117,6 +121,9 @@ func (m *msgServer) webhookAfterSendGroupMsg(ctx context.Context, after *config. | |||||||
| 	if msg.MsgData.ContentType == constant.Typing { | 	if msg.MsgData.ContentType == constant.Typing { | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  | 	if !filterAfterMsg(msg, after) { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
| 	cbReq := &cbapi.CallbackAfterSendGroupMsgReq{ | 	cbReq := &cbapi.CallbackAfterSendGroupMsgReq{ | ||||||
| 		CommonCallbackReq: toCommonCallback(ctx, msg, cbapi.CallbackAfterSendGroupMsgCommand), | 		CommonCallbackReq: toCommonCallback(ctx, msg, cbapi.CallbackAfterSendGroupMsgCommand), | ||||||
| 		GroupID:           msg.MsgData.GroupID, | 		GroupID:           msg.MsgData.GroupID, | ||||||
| @ -129,6 +136,9 @@ func (m *msgServer) webhookBeforeMsgModify(ctx context.Context, before *config.B | |||||||
| 		if msg.MsgData.ContentType != constant.Text { | 		if msg.MsgData.ContentType != constant.Text { | ||||||
| 			return nil | 			return nil | ||||||
| 		} | 		} | ||||||
|  | 		if !filterBeforeMsg(msg, before) { | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
| 		cbReq := &cbapi.CallbackMsgModifyCommandReq{ | 		cbReq := &cbapi.CallbackMsgModifyCommandReq{ | ||||||
| 			CommonCallbackReq: toCommonCallback(ctx, msg, cbapi.CallbackBeforeMsgModifyCommand), | 			CommonCallbackReq: toCommonCallback(ctx, msg, cbapi.CallbackBeforeMsgModifyCommand), | ||||||
| 		} | 		} | ||||||
|  | |||||||
| @ -67,7 +67,7 @@ func (m *msgServer) ClearMsg(ctx context.Context, req *msg.ClearMsgReq) (_ *msg. | |||||||
| 		return nil, err | 		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 | 	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 ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 	pbmsg "github.com/openimsdk/protocol/msg" | ||||||
| 	"github.com/openimsdk/tools/errs" | 	"github.com/openimsdk/tools/errs" | ||||||
| 	"github.com/redis/go-redis/v9" | 	"github.com/redis/go-redis/v9" | ||||||
| 
 | 	"sort" | ||||||
| 	pbmsg "github.com/openimsdk/protocol/msg" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) GetConversationMaxSeq(ctx context.Context, req *pbmsg.GetConversationMaxSeqReq) (*pbmsg.GetConversationMaxSeqResp, error) { | 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 | 	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 ( | import ( | ||||||
| 	"context" | 	"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/authverify" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/msgprocessor" | 	"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/constant" | ||||||
| 	"github.com/openimsdk/protocol/msg" | 	"github.com/openimsdk/protocol/msg" | ||||||
| 	"github.com/openimsdk/protocol/sdkws" | 	"github.com/openimsdk/protocol/sdkws" | ||||||
| 	"github.com/openimsdk/tools/log" | 	"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) { | 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 | 	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) { | 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 { | 	if err := authverify.CheckAccessV3(ctx, req.UserID, m.config.Share.IMAdminUserID); err != nil { | ||||||
| 		return nil, err | 		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) | 		log.ZWarn(ctx, "GetMaxSeqs error", err, "conversationIDs", conversationIDs, "maxSeqs", maxSeqs) | ||||||
| 		return nil, err | 		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 := new(sdkws.GetMaxSeqResp) | ||||||
| 	resp.MaxSeqs = maxSeqs | 	resp.MaxSeqs = maxSeqs | ||||||
| 	return resp, nil | 	return resp, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) SearchMessage(ctx context.Context, req *msg.SearchMessageReq) (resp *msg.SearchMessageResp, err error) { | 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 | 	var total int64 | ||||||
| 	resp = &msg.SearchMessageResp{} | 	resp = &msg.SearchMessageResp{} | ||||||
| 	if total, chatLogs, err = m.MsgDatabase.SearchMessage(ctx, req); err != nil { | 	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) | 		recvMap  = make(map[string]string) | ||||||
| 		groupMap = make(map[string]*sdkws.GroupInfo) | 		groupMap = make(map[string]*sdkws.GroupInfo) | ||||||
| 	) | 	) | ||||||
|  | 
 | ||||||
| 	for _, chatLog := range chatLogs { | 	for _, chatLog := range chatLogs { | ||||||
| 		if chatLog.SenderNickname == "" { | 		if chatLog.MsgData.SenderNickname == "" { | ||||||
| 			sendIDs = append(sendIDs, chatLog.SendID) | 			sendIDs = append(sendIDs, chatLog.MsgData.SendID) | ||||||
| 		} | 		} | ||||||
| 		switch chatLog.SessionType { | 		switch chatLog.MsgData.SessionType { | ||||||
| 		case constant.SingleChatType, constant.NotificationChatType: | 		case constant.SingleChatType, constant.NotificationChatType: | ||||||
| 			recvIDs = append(recvIDs, chatLog.RecvID) | 			recvIDs = append(recvIDs, chatLog.MsgData.RecvID) | ||||||
| 		case constant.WriteGroupChatType, constant.ReadGroupChatType: | 		case constant.WriteGroupChatType, constant.ReadGroupChatType: | ||||||
| 			groupIDs = append(groupIDs, chatLog.GroupID) | 			groupIDs = append(groupIDs, chatLog.MsgData.GroupID) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	// Retrieve sender and receiver information | 	// Retrieve sender and receiver information | ||||||
| 	if len(sendIDs) != 0 { | 	if len(sendIDs) != 0 { | ||||||
| 		sendInfos, err := m.UserLocalCache.GetUsersInfo(ctx, sendIDs) | 		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 | 			sendMap[sendInfo.UserID] = sendInfo.Nickname | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	if len(recvIDs) != 0 { | 	if len(recvIDs) != 0 { | ||||||
| 		recvInfos, err := m.UserLocalCache.GetUsersInfo(ctx, recvIDs) | 		recvInfos, err := m.UserLocalCache.GetUsersInfo(ctx, recvIDs) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| @ -171,20 +210,21 @@ func (m *msgServer) SearchMessage(ctx context.Context, req *msg.SearchMessageReq | |||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	// Construct response with updated information | 	// Construct response with updated information | ||||||
| 	for _, chatLog := range chatLogs { | 	for _, chatLog := range chatLogs { | ||||||
| 		pbchatLog := &msg.ChatLog{} | 		pbchatLog := &msg.ChatLog{} | ||||||
| 		datautil.CopyStructFields(pbchatLog, chatLog) | 		datautil.CopyStructFields(pbchatLog, chatLog.MsgData) | ||||||
| 		pbchatLog.SendTime = chatLog.SendTime | 		pbchatLog.SendTime = chatLog.MsgData.SendTime | ||||||
| 		pbchatLog.CreateTime = chatLog.CreateTime | 		pbchatLog.CreateTime = chatLog.MsgData.CreateTime | ||||||
| 		if chatLog.SenderNickname == "" { | 		if chatLog.MsgData.SenderNickname == "" { | ||||||
| 			pbchatLog.SenderNickname = sendMap[chatLog.SendID] | 			pbchatLog.SenderNickname = sendMap[chatLog.MsgData.SendID] | ||||||
| 		} | 		} | ||||||
| 		switch chatLog.SessionType { | 		switch chatLog.MsgData.SessionType { | ||||||
| 		case constant.SingleChatType, constant.NotificationChatType: | 		case constant.SingleChatType, constant.NotificationChatType: | ||||||
| 			pbchatLog.RecvNickname = recvMap[chatLog.RecvID] | 			pbchatLog.RecvNickname = recvMap[chatLog.MsgData.RecvID] | ||||||
| 		case constant.WriteGroupChatType, constant.ReadGroupChatType: | 		case constant.ReadGroupChatType: | ||||||
| 			groupInfo := groupMap[chatLog.GroupID] | 			groupInfo := groupMap[chatLog.MsgData.GroupID] | ||||||
| 			pbchatLog.SenderFaceURL = groupInfo.FaceURL | 			pbchatLog.SenderFaceURL = groupInfo.FaceURL | ||||||
| 			pbchatLog.GroupMemberCount = groupInfo.MemberCount // Reflects actual member count | 			pbchatLog.GroupMemberCount = groupInfo.MemberCount // Reflects actual member count | ||||||
| 			pbchatLog.RecvID = groupInfo.GroupID | 			pbchatLog.RecvID = groupInfo.GroupID | ||||||
| @ -192,7 +232,9 @@ func (m *msgServer) SearchMessage(ctx context.Context, req *msg.SearchMessageReq | |||||||
| 			pbchatLog.GroupOwner = groupInfo.OwnerUserID | 			pbchatLog.GroupOwner = groupInfo.OwnerUserID | ||||||
| 			pbchatLog.GroupType = groupInfo.GroupType | 			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) | 	resp.ChatLogsNum = int32(total) | ||||||
| 	return resp, nil | 	return resp, nil | ||||||
|  | |||||||
| @ -15,6 +15,7 @@ | |||||||
| package msg | package msg | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
|  | 	"github.com/openimsdk/protocol/msg" | ||||||
| 	"github.com/openimsdk/tools/errs" | 	"github.com/openimsdk/tools/errs" | ||||||
| 	"github.com/redis/go-redis/v9" | 	"github.com/redis/go-redis/v9" | ||||||
| 	"go.mongodb.org/mongo-driver/mongo" | 	"go.mongodb.org/mongo-driver/mongo" | ||||||
| @ -28,3 +29,63 @@ func IsNotFound(err error) bool { | |||||||
| 		return false | 		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/authverify" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/convert" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/convert" | ||||||
| 	"github.com/openimsdk/protocol/relation" | 	"github.com/openimsdk/protocol/relation" | ||||||
|  | 	"github.com/openimsdk/protocol/sdkws" | ||||||
|  | 	"github.com/openimsdk/tools/errs" | ||||||
| 	"github.com/openimsdk/tools/mcontext" | 	"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) { | 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 | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	total, blacks, err := s.blackDatabase.FindOwnerBlacks(ctx, req.UserID, req.Pagination) | 	total, blacks, err := s.blackDatabase.FindOwnerBlacks(ctx, req.UserID, req.Pagination) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		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) { | 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 | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -64,6 +68,7 @@ func (s *friendServer) RemoveBlack(ctx context.Context, req *relation.RemoveBlac | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	s.notificationSender.BlackDeletedNotification(ctx, req) | 	s.notificationSender.BlackDeletedNotification(ctx, req) | ||||||
|  | 	s.webhookAfterRemoveBlack(ctx, &s.config.WebhooksConfig.AfterRemoveBlack, req) | ||||||
| 
 | 
 | ||||||
| 	return &relation.RemoveBlackResp{}, nil | 	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 { | 	if err := authverify.CheckAccessV3(ctx, req.OwnerUserID, s.config.Share.IMAdminUserID); err != nil { | ||||||
| 		return nil, err | 		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}) | 	_, err := s.userRpcClient.GetUsersInfo(ctx, []string{req.OwnerUserID, req.BlackUserID}) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -90,3 +100,53 @@ func (s *friendServer) AddBlack(ctx context.Context, req *relation.AddBlackReq) | |||||||
| 	s.notificationSender.BlackAddedNotification(ctx, req) | 	s.notificationSender.BlackAddedNotification(ctx, req) | ||||||
| 	return &relation.AddBlackResp{}, nil | 	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 { | func (s *friendServer) webhookBeforeImportFriends(ctx context.Context, before *config.BeforeConfig, req *relation.ImportFriendReq) error { | ||||||
| 	return webhook.WithCondition(ctx, before, func(ctx context.Context) error { | 	return webhook.WithCondition(ctx, before, func(ctx context.Context) error { | ||||||
| 		cbReq := &cbapi.CallbackBeforeImportFriendsReq{ | 		cbReq := &cbapi.CallbackBeforeImportFriendsReq{ | ||||||
|  | |||||||
| @ -212,6 +212,7 @@ func (s *friendServer) RespondFriendApply(ctx context.Context, req *relation.Res | |||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return nil, err | 			return nil, err | ||||||
| 		} | 		} | ||||||
|  | 		s.webhookAfterAddFriendAgree(ctx, &s.config.WebhooksConfig.AfterAddFriendAgree, req) | ||||||
| 		s.notificationSender.FriendApplicationAgreedNotification(ctx, req) | 		s.notificationSender.FriendApplicationAgreedNotification(ctx, req) | ||||||
| 		return resp, nil | 		return resp, nil | ||||||
| 	} | 	} | ||||||
| @ -228,20 +229,23 @@ func (s *friendServer) RespondFriendApply(ctx context.Context, req *relation.Res | |||||||
| 
 | 
 | ||||||
| // ok. | // ok. | ||||||
| func (s *friendServer) DeleteFriend(ctx context.Context, req *relation.DeleteFriendReq) (resp *relation.DeleteFriendResp, err error) { | func (s *friendServer) DeleteFriend(ctx context.Context, req *relation.DeleteFriendReq) (resp *relation.DeleteFriendResp, err error) { | ||||||
| 	resp = &relation.DeleteFriendResp{} | 	if err := authverify.CheckAccessV3(ctx, req.OwnerUserID, s.config.Share.IMAdminUserID); err != nil { | ||||||
| 	if err := s.userRpcClient.Access(ctx, req.OwnerUserID); err != nil { |  | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	_, err = s.db.FindFriendsWithError(ctx, req.OwnerUserID, []string{req.FriendUserID}) | 	_, err = s.db.FindFriendsWithError(ctx, req.OwnerUserID, []string{req.FriendUserID}) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	if err := s.db.Delete(ctx, req.OwnerUserID, []string{req.FriendUserID}); err != nil { | 	if err := s.db.Delete(ctx, req.OwnerUserID, []string{req.FriendUserID}); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	s.notificationSender.FriendDeletedNotification(ctx, req) | 	s.notificationSender.FriendDeletedNotification(ctx, req) | ||||||
| 	s.webhookAfterDeleteFriend(ctx, &s.config.WebhooksConfig.AfterDeleteFriend, req) | 	s.webhookAfterDeleteFriend(ctx, &s.config.WebhooksConfig.AfterDeleteFriend, req) | ||||||
| 	return resp, nil | 
 | ||||||
|  | 	return &relation.DeleteFriendResp{}, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // ok. | // 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 { | 	if err = s.webhookBeforeSetFriendRemark(ctx, &s.config.WebhooksConfig.BeforeSetFriendRemark, req); err != nil && err != servererrs.ErrCallbackContinue { | ||||||
| 		return nil, err | 		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 | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	_, err = s.db.FindFriendsWithError(ctx, req.OwnerUserID, []string{req.FriendUserID}) | 	_, err = s.db.FindFriendsWithError(ctx, req.OwnerUserID, []string{req.FriendUserID}) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	if err := s.db.UpdateRemark(ctx, req.OwnerUserID, req.FriendUserID, req.Remark); err != nil { | 	if err := s.db.UpdateRemark(ctx, req.OwnerUserID, req.FriendUserID, req.Remark); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	s.webhookAfterSetFriendRemark(ctx, &s.config.WebhooksConfig.AfterSetFriendRemark, req) | 	s.webhookAfterSetFriendRemark(ctx, &s.config.WebhooksConfig.AfterSetFriendRemark, req) | ||||||
| 	s.notificationSender.FriendRemarkSetNotification(ctx, req.OwnerUserID, req.FriendUserID) | 	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) { | func (s *friendServer) GetDesignatedFriends(ctx context.Context, req *relation.GetDesignatedFriendsReq) (resp *relation.GetDesignatedFriendsResp, err error) { | ||||||
| 	resp = &relation.GetDesignatedFriendsResp{} | 	resp = &relation.GetDesignatedFriendsResp{} | ||||||
| 	if datautil.Duplicate(req.FriendUserIDs) { | 	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). | // 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) { | 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 | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	total, friendRequests, err := s.db.PageFriendRequestToMe(ctx, req.UserID, req.Pagination) | 	total, friendRequests, err := s.db.PageFriendRequestToMe(ctx, req.UserID, req.Pagination) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	resp = &relation.GetPaginationFriendsApplyToResp{} | 	resp = &relation.GetPaginationFriendsApplyToResp{} | ||||||
| 	resp.FriendRequests, err = convert.FriendRequestDB2Pb(ctx, friendRequests, s.userRpcClient.GetUsersInfoMap) | 	resp.FriendRequests, err = convert.FriendRequestDB2Pb(ctx, friendRequests, s.userRpcClient.GetUsersInfoMap) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	resp.Total = int32(total) | 	resp.Total = int32(total) | ||||||
|  | 
 | ||||||
| 	return resp, nil | 	return resp, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *friendServer) GetPaginationFriendsApplyFrom(ctx context.Context, req *relation.GetPaginationFriendsApplyFromReq) (resp *relation.GetPaginationFriendsApplyFromResp, err error) { | func (s *friendServer) GetPaginationFriendsApplyFrom(ctx context.Context, req *relation.GetPaginationFriendsApplyFromReq) (resp *relation.GetPaginationFriendsApplyFromResp, err error) { | ||||||
| 	resp = &relation.GetPaginationFriendsApplyFromResp{} | 	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 | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	total, friendRequests, err := s.db.PageFriendRequestFromMe(ctx, req.UserID, req.Pagination) | 	total, friendRequests, err := s.db.PageFriendRequestFromMe(ctx, req.UserID, req.Pagination) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	resp.FriendRequests, err = convert.FriendRequestDB2Pb(ctx, friendRequests, s.userRpcClient.GetUsersInfoMap) | 	resp.FriendRequests, err = convert.FriendRequestDB2Pb(ctx, friendRequests, s.userRpcClient.GetUsersInfoMap) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	resp.Total = int32(total) | 	resp.Total = int32(total) | ||||||
|  | 
 | ||||||
| 	return resp, nil | 	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) { | 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 | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	total, friends, err := s.db.PageOwnerFriends(ctx, req.UserID, req.Pagination) | 	total, friends, err := s.db.PageOwnerFriends(ctx, req.UserID, req.Pagination) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	resp = &relation.GetPaginationFriendsResp{} | 	resp = &relation.GetPaginationFriendsResp{} | ||||||
| 	resp.FriendsInfo, err = convert.FriendsDB2Pb(ctx, friends, s.userRpcClient.GetUsersInfoMap) | 	resp.FriendsInfo, err = convert.FriendsDB2Pb(ctx, friends, s.userRpcClient.GetUsersInfoMap) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	resp.Total = int32(total) | 	resp.Total = int32(total) | ||||||
|  | 
 | ||||||
| 	return resp, nil | 	return resp, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *friendServer) GetFriendIDs(ctx context.Context, req *relation.GetFriendIDsReq) (resp *relation.GetFriendIDsResp, err error) { | 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 | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	resp = &relation.GetFriendIDsResp{} | 	resp = &relation.GetFriendIDsResp{} | ||||||
| 	resp.FriendIDs, err = s.db.FindFriendUserIDs(ctx, req.UserID) | 	resp.FriendIDs, err = s.db.FindFriendUserIDs(ctx, req.UserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	return resp, nil | 	return resp, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -385,35 +415,45 @@ func (s *friendServer) GetSpecifiedFriendsInfo(ctx context.Context, req *relatio | |||||||
| 	if len(req.UserIDList) == 0 { | 	if len(req.UserIDList) == 0 { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("userIDList is empty") | 		return nil, errs.ErrArgs.WrapMsg("userIDList is empty") | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	if datautil.Duplicate(req.UserIDList) { | 	if datautil.Duplicate(req.UserIDList) { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("userIDList repeated") | 		return nil, errs.ErrArgs.WrapMsg("userIDList repeated") | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	userMap, err := s.userRpcClient.GetUsersInfoMap(ctx, req.UserIDList) | 	userMap, err := s.userRpcClient.GetUsersInfoMap(ctx, req.UserIDList) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	friends, err := s.db.FindFriendsWithError(ctx, req.OwnerUserID, req.UserIDList) | 	friends, err := s.db.FindFriendsWithError(ctx, req.OwnerUserID, req.UserIDList) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	blacks, err := s.blackDatabase.FindBlackInfos(ctx, req.OwnerUserID, req.UserIDList) | 	blacks, err := s.blackDatabase.FindBlackInfos(ctx, req.OwnerUserID, req.UserIDList) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	friendMap := datautil.SliceToMap(friends, func(e *model.Friend) string { | 	friendMap := datautil.SliceToMap(friends, func(e *model.Friend) string { | ||||||
| 		return e.FriendUserID | 		return e.FriendUserID | ||||||
| 	}) | 	}) | ||||||
|  | 
 | ||||||
| 	blackMap := datautil.SliceToMap(blacks, func(e *model.Black) string { | 	blackMap := datautil.SliceToMap(blacks, func(e *model.Black) string { | ||||||
| 		return e.BlockUserID | 		return e.BlockUserID | ||||||
| 	}) | 	}) | ||||||
|  | 
 | ||||||
| 	resp := &relation.GetSpecifiedFriendsInfoResp{ | 	resp := &relation.GetSpecifiedFriendsInfoResp{ | ||||||
| 		Infos: make([]*relation.GetSpecifiedFriendsInfoInfo, 0, len(req.UserIDList)), | 		Infos: make([]*relation.GetSpecifiedFriendsInfoInfo, 0, len(req.UserIDList)), | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	for _, userID := range req.UserIDList { | 	for _, userID := range req.UserIDList { | ||||||
| 		user := userMap[userID] | 		user := userMap[userID] | ||||||
|  | 
 | ||||||
| 		if user == nil { | 		if user == nil { | ||||||
| 			continue | 			continue | ||||||
| 		} | 		} | ||||||
|  | 
 | ||||||
| 		var friendInfo *sdkws.FriendInfo | 		var friendInfo *sdkws.FriendInfo | ||||||
| 		if friend := friendMap[userID]; friend != nil { | 		if friend := friendMap[userID]; friend != nil { | ||||||
| 			friendInfo = &sdkws.FriendInfo{ | 			friendInfo = &sdkws.FriendInfo{ | ||||||
| @ -426,6 +466,7 @@ func (s *friendServer) GetSpecifiedFriendsInfo(ctx context.Context, req *relatio | |||||||
| 				IsPinned:       friend.IsPinned, | 				IsPinned:       friend.IsPinned, | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  | 
 | ||||||
| 		var blackInfo *sdkws.BlackInfo | 		var blackInfo *sdkws.BlackInfo | ||||||
| 		if black := blackMap[userID]; black != nil { | 		if black := blackMap[userID]; black != nil { | ||||||
| 			blackInfo = &sdkws.BlackInfo{ | 			blackInfo = &sdkws.BlackInfo{ | ||||||
| @ -436,12 +477,14 @@ func (s *friendServer) GetSpecifiedFriendsInfo(ctx context.Context, req *relatio | |||||||
| 				Ex:             black.Ex, | 				Ex:             black.Ex, | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  | 
 | ||||||
| 		resp.Infos = append(resp.Infos, &relation.GetSpecifiedFriendsInfoInfo{ | 		resp.Infos = append(resp.Infos, &relation.GetSpecifiedFriendsInfoInfo{ | ||||||
| 			UserInfo:   user, | 			UserInfo:   user, | ||||||
| 			FriendInfo: friendInfo, | 			FriendInfo: friendInfo, | ||||||
| 			BlackInfo:  blackInfo, | 			BlackInfo:  blackInfo, | ||||||
| 		}) | 		}) | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	return resp, nil | 	return resp, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -17,9 +17,10 @@ package third | |||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
| 	"crypto/rand" | 	"crypto/rand" | ||||||
| 	relationtb "github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" |  | ||||||
| 	"time" | 	"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/authverify" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/servererrs" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/servererrs" | ||||||
| 	"github.com/openimsdk/protocol/constant" | 	"github.com/openimsdk/protocol/constant" | ||||||
|  | |||||||
| @ -16,6 +16,7 @@ package user | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/webhook" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/webhook" | ||||||
| 	"github.com/openimsdk/tools/utils/datautil" | 	"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 { | 	return webhook.WithCondition(ctx, before, func(ctx context.Context) error { | ||||||
| 		cbReq := &cbapi.CallbackBeforeUserRegisterReq{ | 		cbReq := &cbapi.CallbackBeforeUserRegisterReq{ | ||||||
| 			CallbackCommand: cbapi.CallbackBeforeUserRegisterCommand, | 			CallbackCommand: cbapi.CallbackBeforeUserRegisterCommand, | ||||||
| 			Secret:          req.Secret, |  | ||||||
| 			Users:           req.Users, | 			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) { | func (s *userServer) webhookAfterUserRegister(ctx context.Context, after *config.AfterConfig, req *pbuser.UserRegisterReq) { | ||||||
| 	cbReq := &cbapi.CallbackAfterUserRegisterReq{ | 	cbReq := &cbapi.CallbackAfterUserRegisterReq{ | ||||||
| 		CallbackCommand: cbapi.CallbackAfterUserRegisterCommand, | 		CallbackCommand: cbapi.CallbackAfterUserRegisterCommand, | ||||||
| 		Secret:          req.Secret, |  | ||||||
| 		Users:           req.Users, | 		Users:           req.Users, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -2,6 +2,9 @@ package user | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 
 | ||||||
|  | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
|  | 
 | ||||||
| 	"github.com/openimsdk/protocol/constant" | 	"github.com/openimsdk/protocol/constant" | ||||||
| 	pbuser "github.com/openimsdk/protocol/user" | 	pbuser "github.com/openimsdk/protocol/user" | ||||||
| ) | ) | ||||||
| @ -59,7 +62,7 @@ func (s *userServer) SetUserStatus(ctx context.Context, req *pbuser.SetUserStatu | |||||||
| 	case constant.Online: | 	case constant.Online: | ||||||
| 		online = []int32{req.PlatformID} | 		online = []int32{req.PlatformID} | ||||||
| 	case constant.Offline: | 	case constant.Offline: | ||||||
| 		online = []int32{req.PlatformID} | 		offline = []int32{req.PlatformID} | ||||||
| 	} | 	} | ||||||
| 	if err := s.online.SetUserOnline(ctx, req.UserID, online, offline); err != nil { | 	if err := s.online.SetUserOnline(ctx, req.UserID, online, offline); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -80,3 +83,22 @@ func (s *userServer) SetUserOnlineStatus(ctx context.Context, req *pbuser.SetUse | |||||||
| 	} | 	} | ||||||
| 	return &pbuser.SetUserOnlineStatusResp{}, nil | 	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" | 	"github.com/openimsdk/tools/db/pagination" | ||||||
| 	registry "github.com/openimsdk/tools/discovery" | 	registry "github.com/openimsdk/tools/discovery" | ||||||
| 	"github.com/openimsdk/tools/errs" | 	"github.com/openimsdk/tools/errs" | ||||||
| 	"github.com/openimsdk/tools/log" |  | ||||||
| 	"github.com/openimsdk/tools/utils/datautil" | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
| 	"google.golang.org/grpc" | 	"google.golang.org/grpc" | ||||||
| ) | ) | ||||||
| @ -263,10 +262,11 @@ func (s *userServer) UserRegister(ctx context.Context, req *pbuser.UserRegisterR | |||||||
| 	if len(req.Users) == 0 { | 	if len(req.Users) == 0 { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("users is empty") | 		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) | 	if err = authverify.CheckAdmin(ctx, s.config.Share.IMAdminUserID); err != nil { | ||||||
| 		return nil, errs.ErrNoPermission.WrapMsg("secret invalid") | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	if datautil.DuplicateAny(req.Users, func(e *sdkws.UserInfo) string { return e.UserID }) { | 	if datautil.DuplicateAny(req.Users, func(e *sdkws.UserInfo) string { return e.UserID }) { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("userID repeated") | 		return nil, errs.ErrArgs.WrapMsg("userID repeated") | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -79,13 +79,13 @@ func Start(ctx context.Context, config *CronTaskConfig) error { | |||||||
| 		now := time.Now() | 		now := time.Now() | ||||||
| 		deltime := now.Add(-time.Hour * 24 * time.Duration(config.CronTask.RetainChatRecords)) | 		deltime := now.Add(-time.Hour * 24 * time.Duration(config.CronTask.RetainChatRecords)) | ||||||
| 		ctx := mcontext.SetOperationID(ctx, fmt.Sprintf("cron_%d_%d", os.Getpid(), deltime.UnixMilli())) | 		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 { | 		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)) | 			log.ZError(ctx, "cron clear chat records failed", err, "deltime", deltime, "cont", time.Since(now)) | ||||||
| 			return | 			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 { | 	if _, err := crontab.AddFunc(config.CronTask.CronExecuteTime, clearMsgFunc); err != nil { | ||||||
| 		return errs.Wrap(err) | 		return errs.Wrap(err) | ||||||
| @ -95,7 +95,7 @@ func Start(ctx context.Context, config *CronTaskConfig) error { | |||||||
| 	msgDestructFunc := func() { | 	msgDestructFunc := func() { | ||||||
| 		now := time.Now() | 		now := time.Now() | ||||||
| 		ctx := mcontext.SetOperationID(ctx, fmt.Sprintf("cron_%d_%d", os.Getpid(), now.UnixMilli())) | 		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{}) | 		conversations, err := conversationClient.GetConversationsNeedDestructMsgs(ctx, &pbconversation.GetConversationsNeedDestructMsgsReq{}) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| @ -108,7 +108,7 @@ func Start(ctx context.Context, config *CronTaskConfig) error { | |||||||
| 				return | 				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 { | 	if _, err := crontab.AddFunc(config.CronTask.CronExecuteTime, msgDestructFunc); err != nil { | ||||||
| 		return errs.Wrap(err) | 		return errs.Wrap(err) | ||||||
| @ -119,18 +119,18 @@ func Start(ctx context.Context, config *CronTaskConfig) error { | |||||||
| 	// 	now := time.Now() | 	// 	now := time.Now() | ||||||
| 	// 	deleteTime := now.Add(-time.Hour * 24 * time.Duration(config.CronTask.FileExpireTime)) | 	// 	deleteTime := now.Add(-time.Hour * 24 * time.Duration(config.CronTask.FileExpireTime)) | ||||||
| 	// 	ctx := mcontext.SetOperationID(ctx, fmt.Sprintf("cron_%d_%d", os.Getpid(), deleteTime.UnixMilli())) | 	// 	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 { | 	// 	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)) | 	// 		log.ZError(ctx, "cron deleteoutDatedData failed", err, "deleteTime", deleteTime, "cont", time.Since(now)) | ||||||
| 	// 		return | 	// 		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 { | 	// if _, err := crontab.AddFunc(config.CronTask.CronExecuteTime, deleteObjectFunc); err != nil { | ||||||
| 	// 	return errs.Wrap(err) | 	// 	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() | 	crontab.Start() | ||||||
| 	<-ctx.Done() | 	<-ctx.Done() | ||||||
| 	return nil | 	return nil | ||||||
|  | |||||||
| @ -18,13 +18,14 @@ const ( | |||||||
| 	CallbackBeforeInviteJoinGroupCommand    = "callbackBeforeInviteJoinGroupCommand" | 	CallbackBeforeInviteJoinGroupCommand    = "callbackBeforeInviteJoinGroupCommand" | ||||||
| 	CallbackAfterJoinGroupCommand           = "callbackAfterJoinGroupCommand" | 	CallbackAfterJoinGroupCommand           = "callbackAfterJoinGroupCommand" | ||||||
| 	CallbackAfterSetGroupInfoCommand        = "callbackAfterSetGroupInfoCommand" | 	CallbackAfterSetGroupInfoCommand        = "callbackAfterSetGroupInfoCommand" | ||||||
| 	CallbackAfterSetGroupInfoEXCommand      = "callbackAfterSetGroupInfoCommandEX" | 	CallbackAfterSetGroupInfoExCommand      = "callbackAfterSetGroupInfoExCommand" | ||||||
| 	CallbackBeforeSetGroupInfoCommand       = "callbackBeforeSetGroupInfoCommand" | 	CallbackBeforeSetGroupInfoCommand       = "callbackBeforeSetGroupInfoCommand" | ||||||
| 	CallbackBeforeSetGroupInfoEXCommand     = "callbackBeforeSetGroupInfoEXCommand" | 	CallbackBeforeSetGroupInfoExCommand     = "callbackBeforeSetGroupInfoExCommand" | ||||||
| 	CallbackAfterRevokeMsgCommand           = "callbackBeforeAfterMsgCommand" | 	CallbackAfterRevokeMsgCommand           = "callbackBeforeAfterMsgCommand" | ||||||
| 	CallbackBeforeAddBlackCommand           = "callbackBeforeAddBlackCommand" | 	CallbackBeforeAddBlackCommand           = "callbackBeforeAddBlackCommand" | ||||||
| 	CallbackAfterAddFriendCommand           = "callbackAfterAddFriendCommand" | 	CallbackAfterAddFriendCommand           = "callbackAfterAddFriendCommand" | ||||||
| 	CallbackBeforeAddFriendAgreeCommand     = "callbackBeforeAddFriendAgreeCommand" | 	CallbackBeforeAddFriendAgreeCommand     = "callbackBeforeAddFriendAgreeCommand" | ||||||
|  | 	CallbackAfterAddFriendAgreeCommand      = "callbackAfterAddFriendAgreeCommand" | ||||||
| 	CallbackAfterDeleteFriendCommand        = "callbackAfterDeleteFriendCommand" | 	CallbackAfterDeleteFriendCommand        = "callbackAfterDeleteFriendCommand" | ||||||
| 	CallbackBeforeImportFriendsCommand      = "callbackBeforeImportFriendsCommand" | 	CallbackBeforeImportFriendsCommand      = "callbackBeforeImportFriendsCommand" | ||||||
| 	CallbackAfterImportFriendsCommand       = "callbackAfterImportFriendsCommand" | 	CallbackAfterImportFriendsCommand       = "callbackAfterImportFriendsCommand" | ||||||
|  | |||||||
| @ -90,6 +90,18 @@ type CallbackBeforeAddFriendAgreeResp struct { | |||||||
| 	CommonCallbackResp | 	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 { | type CallbackAfterDeleteFriendReq struct { | ||||||
| 	CallbackCommand `json:"callbackCommand"` | 	CallbackCommand `json:"callbackCommand"` | ||||||
| 	OwnerUserID     string `json:"ownerUserID" ` | 	OwnerUserID     string `json:"ownerUserID" ` | ||||||
|  | |||||||
| @ -244,11 +244,11 @@ type CallbackAfterSetGroupInfoResp struct { | |||||||
| 	CommonCallbackResp | 	CommonCallbackResp | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| type CallbackBeforeSetGroupInfoEXReq struct { | type CallbackBeforeSetGroupInfoExReq struct { | ||||||
| 	CallbackCommand   `json:"callbackCommand"` | 	CallbackCommand   `json:"callbackCommand"` | ||||||
| 	OperationID       string                  `json:"operationID"` | 	OperationID       string                  `json:"operationID"` | ||||||
| 	GroupID           string                  `json:"groupID"` | 	GroupID           string                  `json:"groupID"` | ||||||
| 	GroupName         string                  `json:"groupName"` | 	GroupName         *wrapperspb.StringValue `json:"groupName"` | ||||||
| 	Notification      *wrapperspb.StringValue `json:"notification"` | 	Notification      *wrapperspb.StringValue `json:"notification"` | ||||||
| 	Introduction      *wrapperspb.StringValue `json:"introduction"` | 	Introduction      *wrapperspb.StringValue `json:"introduction"` | ||||||
| 	FaceURL           *wrapperspb.StringValue `json:"faceURL"` | 	FaceURL           *wrapperspb.StringValue `json:"faceURL"` | ||||||
| @ -258,10 +258,10 @@ type CallbackBeforeSetGroupInfoEXReq struct { | |||||||
| 	ApplyMemberFriend *wrapperspb.Int32Value  `json:"applyMemberFriend"` | 	ApplyMemberFriend *wrapperspb.Int32Value  `json:"applyMemberFriend"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| type CallbackBeforeSetGroupInfoEXResp struct { | type CallbackBeforeSetGroupInfoExResp struct { | ||||||
| 	CommonCallbackResp | 	CommonCallbackResp | ||||||
| 	GroupID           string                  `json:"groupID"` | 	GroupID           string                  `json:"groupID"` | ||||||
| 	GroupName         string                  `json:"groupName"` | 	GroupName         *wrapperspb.StringValue `json:"groupName"` | ||||||
| 	Notification      *wrapperspb.StringValue `json:"notification"` | 	Notification      *wrapperspb.StringValue `json:"notification"` | ||||||
| 	Introduction      *wrapperspb.StringValue `json:"introduction"` | 	Introduction      *wrapperspb.StringValue `json:"introduction"` | ||||||
| 	FaceURL           *wrapperspb.StringValue `json:"faceURL"` | 	FaceURL           *wrapperspb.StringValue `json:"faceURL"` | ||||||
| @ -271,11 +271,11 @@ type CallbackBeforeSetGroupInfoEXResp struct { | |||||||
| 	ApplyMemberFriend *wrapperspb.Int32Value  `json:"applyMemberFriend"` | 	ApplyMemberFriend *wrapperspb.Int32Value  `json:"applyMemberFriend"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| type CallbackAfterSetGroupInfoEXReq struct { | type CallbackAfterSetGroupInfoExReq struct { | ||||||
| 	CallbackCommand   `json:"callbackCommand"` | 	CallbackCommand   `json:"callbackCommand"` | ||||||
| 	OperationID       string                  `json:"operationID"` | 	OperationID       string                  `json:"operationID"` | ||||||
| 	GroupID           string                  `json:"groupID"` | 	GroupID           string                  `json:"groupID"` | ||||||
| 	GroupName         string                  `json:"groupName"` | 	GroupName         *wrapperspb.StringValue `json:"groupName"` | ||||||
| 	Notification      *wrapperspb.StringValue `json:"notification"` | 	Notification      *wrapperspb.StringValue `json:"notification"` | ||||||
| 	Introduction      *wrapperspb.StringValue `json:"introduction"` | 	Introduction      *wrapperspb.StringValue `json:"introduction"` | ||||||
| 	FaceURL           *wrapperspb.StringValue `json:"faceURL"` | 	FaceURL           *wrapperspb.StringValue `json:"faceURL"` | ||||||
| @ -285,6 +285,6 @@ type CallbackAfterSetGroupInfoEXReq struct { | |||||||
| 	ApplyMemberFriend *wrapperspb.Int32Value  `json:"applyMemberFriend"` | 	ApplyMemberFriend *wrapperspb.Int32Value  `json:"applyMemberFriend"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| type CallbackAfterSetGroupInfoEXResp struct { | type CallbackAfterSetGroupInfoExResp struct { | ||||||
| 	CommonCallbackResp | 	CommonCallbackResp | ||||||
| } | } | ||||||
|  | |||||||
| @ -72,7 +72,6 @@ type CallbackAfterUpdateUserInfoExResp struct { | |||||||
| 
 | 
 | ||||||
| type CallbackBeforeUserRegisterReq struct { | type CallbackBeforeUserRegisterReq struct { | ||||||
| 	CallbackCommand `json:"callbackCommand"` | 	CallbackCommand `json:"callbackCommand"` | ||||||
| 	Secret          string            `json:"secret"` |  | ||||||
| 	Users           []*sdkws.UserInfo `json:"users"` | 	Users           []*sdkws.UserInfo `json:"users"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -83,7 +82,6 @@ type CallbackBeforeUserRegisterResp struct { | |||||||
| 
 | 
 | ||||||
| type CallbackAfterUserRegisterReq struct { | type CallbackAfterUserRegisterReq struct { | ||||||
| 	CallbackCommand `json:"callbackCommand"` | 	CallbackCommand `json:"callbackCommand"` | ||||||
| 	Secret          string            `json:"secret"` |  | ||||||
| 	Users           []*sdkws.UserInfo `json:"users"` | 	Users           []*sdkws.UserInfo `json:"users"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -15,13 +15,14 @@ | |||||||
| package cmd | package cmd | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
|  | 	"math" | ||||||
|  | 	"testing" | ||||||
|  | 
 | ||||||
| 	"github.com/openimsdk/protocol/auth" | 	"github.com/openimsdk/protocol/auth" | ||||||
| 	"github.com/openimsdk/tools/apiresp" | 	"github.com/openimsdk/tools/apiresp" | ||||||
| 	"github.com/openimsdk/tools/utils/jsonutil" | 	"github.com/openimsdk/tools/utils/jsonutil" | ||||||
| 	"github.com/stretchr/testify/mock" | 	"github.com/stretchr/testify/mock" | ||||||
| 	"go.mongodb.org/mongo-driver/bson/primitive" | 	"go.mongodb.org/mongo-driver/bson/primitive" | ||||||
| 	"math" |  | ||||||
| 	"testing" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| // MockRootCmd is a mock type for the RootCmd type | // MockRootCmd is a mock type for the RootCmd type | ||||||
| @ -39,7 +40,7 @@ func TestName(t *testing.T) { | |||||||
| 		ErrCode: 1234, | 		ErrCode: 1234, | ||||||
| 		ErrMsg:  "test", | 		ErrMsg:  "test", | ||||||
| 		ErrDlt:  "4567", | 		ErrDlt:  "4567", | ||||||
| 		Data: &auth.UserTokenResp{ | 		Data: &auth.GetUserTokenResp{ | ||||||
| 			Token:             "1234567", | 			Token:             "1234567", | ||||||
| 			ExpireTimeSeconds: math.MaxInt64, | 			ExpireTimeSeconds: math.MaxInt64, | ||||||
| 		}, | 		}, | ||||||
| @ -51,7 +52,7 @@ func TestName(t *testing.T) { | |||||||
| 	t.Log(string(data)) | 	t.Log(string(data)) | ||||||
| 
 | 
 | ||||||
| 	var rReso apiresp.ApiResponse | 	var rReso apiresp.ApiResponse | ||||||
| 	rReso.Data = &auth.UserTokenResp{} | 	rReso.Data = &auth.GetUserTokenResp{} | ||||||
| 
 | 
 | ||||||
| 	if err := jsonutil.JsonUnmarshal(data, &rReso); err != nil { | 	if err := jsonutil.JsonUnmarshal(data, &rReso); err != nil { | ||||||
| 		panic(err) | 		panic(err) | ||||||
|  | |||||||
| @ -37,7 +37,6 @@ func NewPushRpcCmd() *PushRpcCmd { | |||||||
| 	ret.configMap = map[string]any{ | 	ret.configMap = map[string]any{ | ||||||
| 		OpenIMPushCfgFileName:    &pushConfig.RpcConfig, | 		OpenIMPushCfgFileName:    &pushConfig.RpcConfig, | ||||||
| 		RedisConfigFileName:      &pushConfig.RedisConfig, | 		RedisConfigFileName:      &pushConfig.RedisConfig, | ||||||
| 		MongodbConfigFileName:    &pushConfig.MongodbConfig, |  | ||||||
| 		KafkaConfigFileName:      &pushConfig.KafkaConfig, | 		KafkaConfigFileName:      &pushConfig.KafkaConfig, | ||||||
| 		ShareFileName:            &pushConfig.Share, | 		ShareFileName:            &pushConfig.Share, | ||||||
| 		NotificationFileName:     &pushConfig.NotificationConfig, | 		NotificationFileName:     &pushConfig.NotificationConfig, | ||||||
|  | |||||||
| @ -129,10 +129,11 @@ func (r *RootCmd) applyOptions(opts ...func(*CmdOpts)) *CmdOpts { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (r *RootCmd) initializeLogger(cmdOpts *CmdOpts) error { | func (r *RootCmd) initializeLogger(cmdOpts *CmdOpts) error { | ||||||
| 	err := log.InitFromConfig( | 	err := log.InitLoggerFromConfig( | ||||||
| 
 | 
 | ||||||
| 		cmdOpts.loggerPrefixName, | 		cmdOpts.loggerPrefixName, | ||||||
| 		r.processName, | 		r.processName, | ||||||
|  | 		"", "", | ||||||
| 		r.log.RemainLogLevel, | 		r.log.RemainLogLevel, | ||||||
| 		r.log.IsStdout, | 		r.log.IsStdout, | ||||||
| 		r.log.IsJson, | 		r.log.IsJson, | ||||||
|  | |||||||
| @ -73,18 +73,21 @@ type Mongo struct { | |||||||
| 	MaxRetry    int      `mapstructure:"maxRetry"` | 	MaxRetry    int      `mapstructure:"maxRetry"` | ||||||
| } | } | ||||||
| type Kafka struct { | type Kafka struct { | ||||||
| 	Username       string    `mapstructure:"username"` | 	Username           string   `mapstructure:"username"` | ||||||
| 	Password       string    `mapstructure:"password"` | 	Password           string   `mapstructure:"password"` | ||||||
| 	ProducerAck    string    `mapstructure:"producerAck"` | 	ProducerAck        string   `mapstructure:"producerAck"` | ||||||
| 	CompressType   string    `mapstructure:"compressType"` | 	CompressType       string   `mapstructure:"compressType"` | ||||||
| 	Address        []string  `mapstructure:"address"` | 	Address            []string `mapstructure:"address"` | ||||||
| 	ToRedisTopic   string    `mapstructure:"toRedisTopic"` | 	ToRedisTopic       string   `mapstructure:"toRedisTopic"` | ||||||
| 	ToMongoTopic   string    `mapstructure:"toMongoTopic"` | 	ToMongoTopic       string   `mapstructure:"toMongoTopic"` | ||||||
| 	ToPushTopic    string    `mapstructure:"toPushTopic"` | 	ToPushTopic        string   `mapstructure:"toPushTopic"` | ||||||
| 	ToRedisGroupID string    `mapstructure:"toRedisGroupID"` | 	ToOfflinePushTopic string   `mapstructure:"toOfflinePushTopic"` | ||||||
| 	ToMongoGroupID string    `mapstructure:"toMongoGroupID"` | 	ToRedisGroupID     string   `mapstructure:"toRedisGroupID"` | ||||||
| 	ToPushGroupID  string    `mapstructure:"toPushGroupID"` | 	ToMongoGroupID     string   `mapstructure:"toMongoGroupID"` | ||||||
| 	Tls            TLSConfig `mapstructure:"tls"` | 	ToPushGroupID      string   `mapstructure:"toPushGroupID"` | ||||||
|  | 	ToOfflineGroupID   string   `mapstructure:"toOfflinePushGroupID"` | ||||||
|  | 
 | ||||||
|  | 	Tls TLSConfig `mapstructure:"tls"` | ||||||
| } | } | ||||||
| type TLSConfig struct { | type TLSConfig struct { | ||||||
| 	EnableTLS          bool   `mapstructure:"enableTLS"` | 	EnableTLS          bool   `mapstructure:"enableTLS"` | ||||||
| @ -97,8 +100,9 @@ type TLSConfig struct { | |||||||
| 
 | 
 | ||||||
| type API struct { | type API struct { | ||||||
| 	Api struct { | 	Api struct { | ||||||
| 		ListenIP string `mapstructure:"listenIP"` | 		ListenIP         string `mapstructure:"listenIP"` | ||||||
| 		Ports    []int  `mapstructure:"ports"` | 		Ports            []int  `mapstructure:"ports"` | ||||||
|  | 		CompressionLevel int    `mapstructure:"compressionLevel"` | ||||||
| 	} `mapstructure:"api"` | 	} `mapstructure:"api"` | ||||||
| 	Prometheus struct { | 	Prometheus struct { | ||||||
| 		Enable     bool   `mapstructure:"enable"` | 		Enable     bool   `mapstructure:"enable"` | ||||||
| @ -181,7 +185,6 @@ type MsgGateway struct { | |||||||
| 		WebsocketMaxMsgLen  int   `mapstructure:"websocketMaxMsgLen"` | 		WebsocketMaxMsgLen  int   `mapstructure:"websocketMaxMsgLen"` | ||||||
| 		WebsocketTimeout    int   `mapstructure:"websocketTimeout"` | 		WebsocketTimeout    int   `mapstructure:"websocketTimeout"` | ||||||
| 	} `mapstructure:"longConnSvr"` | 	} `mapstructure:"longConnSvr"` | ||||||
| 	MultiLoginPolicy int `mapstructure:"multiLoginPolicy"` |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| type MsgTransfer struct { | type MsgTransfer struct { | ||||||
| @ -220,6 +223,7 @@ type Push struct { | |||||||
| 		BadgeCount bool   `mapstructure:"badgeCount"` | 		BadgeCount bool   `mapstructure:"badgeCount"` | ||||||
| 		Production bool   `mapstructure:"production"` | 		Production bool   `mapstructure:"production"` | ||||||
| 	} `mapstructure:"iosPush"` | 	} `mapstructure:"iosPush"` | ||||||
|  | 	FullUserCache bool `mapstructure:"fullUserCache"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| type Auth struct { | type Auth struct { | ||||||
| @ -336,26 +340,50 @@ type Redis struct { | |||||||
| 	Password    string   `mapstructure:"password"` | 	Password    string   `mapstructure:"password"` | ||||||
| 	ClusterMode bool     `mapstructure:"clusterMode"` | 	ClusterMode bool     `mapstructure:"clusterMode"` | ||||||
| 	DB          int      `mapstructure:"storage"` | 	DB          int      `mapstructure:"storage"` | ||||||
| 	MaxRetry    int      `mapstructure:"MaxRetry"` | 	MaxRetry    int      `mapstructure:"maxRetry"` | ||||||
|  | 	PoolSize    int      `mapstructure:"poolSize"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| type BeforeConfig struct { | type BeforeConfig struct { | ||||||
| 	Enable         bool `mapstructure:"enable"` | 	Enable         bool     `mapstructure:"enable"` | ||||||
| 	Timeout        int  `mapstructure:"timeout"` | 	Timeout        int      `mapstructure:"timeout"` | ||||||
| 	FailedContinue bool `mapstructure:"failedContinue"` | 	FailedContinue bool     `mapstructure:"failedContinue"` | ||||||
|  | 	AllowedTypes   []string `mapstructure:"allowedTypes"` | ||||||
|  | 	DeniedTypes    []string `mapstructure:"deniedTypes"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| type AfterConfig struct { | type AfterConfig struct { | ||||||
| 	Enable       bool     `mapstructure:"enable"` | 	Enable       bool     `mapstructure:"enable"` | ||||||
| 	Timeout      int      `mapstructure:"timeout"` | 	Timeout      int      `mapstructure:"timeout"` | ||||||
| 	AttentionIds []string `mapstructure:"attentionIds"` | 	AttentionIds []string `mapstructure:"attentionIds"` | ||||||
|  | 	AllowedTypes []string `mapstructure:"allowedTypes"` | ||||||
|  | 	DeniedTypes  []string `mapstructure:"deniedTypes"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| type Share struct { | type Share struct { | ||||||
| 	Secret          string          `mapstructure:"secret"` | 	Secret          string          `mapstructure:"secret"` | ||||||
| 	RpcRegisterName RpcRegisterName `mapstructure:"rpcRegisterName"` | 	RpcRegisterName RpcRegisterName `mapstructure:"rpcRegisterName"` | ||||||
| 	IMAdminUserID   []string        `mapstructure:"imAdminUserID"` | 	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 { | type RpcRegisterName struct { | ||||||
| 	User           string `mapstructure:"user"` | 	User           string `mapstructure:"user"` | ||||||
| 	Friend         string `mapstructure:"friend"` | 	Friend         string `mapstructure:"friend"` | ||||||
| @ -422,12 +450,13 @@ type Webhooks struct { | |||||||
| 	BeforeInviteUserToGroup  BeforeConfig `mapstructure:"beforeInviteUserToGroup"` | 	BeforeInviteUserToGroup  BeforeConfig `mapstructure:"beforeInviteUserToGroup"` | ||||||
| 	AfterSetGroupInfo        AfterConfig  `mapstructure:"afterSetGroupInfo"` | 	AfterSetGroupInfo        AfterConfig  `mapstructure:"afterSetGroupInfo"` | ||||||
| 	BeforeSetGroupInfo       BeforeConfig `mapstructure:"beforeSetGroupInfo"` | 	BeforeSetGroupInfo       BeforeConfig `mapstructure:"beforeSetGroupInfo"` | ||||||
| 	AfterSetGroupInfoEX      AfterConfig  `mapstructure:"afterSetGroupInfoEX"` | 	AfterSetGroupInfoEx      AfterConfig  `mapstructure:"afterSetGroupInfoEx"` | ||||||
| 	BeforeSetGroupInfoEX     BeforeConfig `mapstructure:"beforeSetGroupInfoEX"` | 	BeforeSetGroupInfoEx     BeforeConfig `mapstructure:"beforeSetGroupInfoEx"` | ||||||
| 	AfterRevokeMsg           AfterConfig  `mapstructure:"afterRevokeMsg"` | 	AfterRevokeMsg           AfterConfig  `mapstructure:"afterRevokeMsg"` | ||||||
| 	BeforeAddBlack           BeforeConfig `mapstructure:"beforeAddBlack"` | 	BeforeAddBlack           BeforeConfig `mapstructure:"beforeAddBlack"` | ||||||
| 	AfterAddFriend           AfterConfig  `mapstructure:"afterAddFriend"` | 	AfterAddFriend           AfterConfig  `mapstructure:"afterAddFriend"` | ||||||
| 	BeforeAddFriendAgree     BeforeConfig `mapstructure:"beforeAddFriendAgree"` | 	BeforeAddFriendAgree     BeforeConfig `mapstructure:"beforeAddFriendAgree"` | ||||||
|  | 	AfterAddFriendAgree      AfterConfig  `mapstructure:"afterAddFriendAgree"` | ||||||
| 	AfterDeleteFriend        AfterConfig  `mapstructure:"afterDeleteFriend"` | 	AfterDeleteFriend        AfterConfig  `mapstructure:"afterDeleteFriend"` | ||||||
| 	BeforeImportFriends      BeforeConfig `mapstructure:"beforeImportFriends"` | 	BeforeImportFriends      BeforeConfig `mapstructure:"beforeImportFriends"` | ||||||
| 	AfterImportFriends       AfterConfig  `mapstructure:"afterImportFriends"` | 	AfterImportFriends       AfterConfig  `mapstructure:"afterImportFriends"` | ||||||
| @ -474,6 +503,7 @@ func (r *Redis) Build() *redisutil.Config { | |||||||
| 		Password:    r.Password, | 		Password:    r.Password, | ||||||
| 		DB:          r.DB, | 		DB:          r.DB, | ||||||
| 		MaxRetry:    r.MaxRetry, | 		MaxRetry:    r.MaxRetry, | ||||||
|  | 		PoolSize:    r.PoolSize, | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -18,6 +18,7 @@ import ( | |||||||
| 	"context" | 	"context" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||||
|  | 	"github.com/openimsdk/protocol/relation" | ||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/protocol/sdkws" | 	"github.com/openimsdk/protocol/sdkws" | ||||||
| 	"github.com/openimsdk/tools/utils/datautil" | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
| @ -35,9 +36,7 @@ func FriendPb2DB(friend *sdkws.FriendInfo) *model.Friend { | |||||||
| 	return dbFriend | 	return dbFriend | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func FriendDB2Pb(ctx context.Context, friendDB *model.Friend, | func FriendDB2Pb(ctx context.Context, friendDB *model.Friend, getUsers func(ctx context.Context, userIDs []string) (map[string]*sdkws.UserInfo, error)) (*sdkws.FriendInfo, error) { | ||||||
| 	getUsers func(ctx context.Context, userIDs []string) (map[string]*sdkws.UserInfo, error), |  | ||||||
| ) (*sdkws.FriendInfo, error) { |  | ||||||
| 	users, err := getUsers(ctx, []string{friendDB.FriendUserID}) | 	users, err := getUsers(ctx, []string{friendDB.FriendUserID}) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -53,11 +52,7 @@ func FriendDB2Pb(ctx context.Context, friendDB *model.Friend, | |||||||
| 	}, nil | 	}, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func FriendsDB2Pb( | 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) { | ||||||
| 	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 { | 	if len(friendsDB) == 0 { | ||||||
| 		return nil, nil | 		return nil, nil | ||||||
| 	} | 	} | ||||||
| @ -86,7 +81,21 @@ func FriendsDB2Pb( | |||||||
| 		friendsPb = append(friendsPb, friendPb) | 		friendsPb = append(friendsPb, friendPb) | ||||||
| 	} | 	} | ||||||
| 	return friendsPb, nil | 	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) { | 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", | 		Name: "msg_offline_push_failed_total", | ||||||
| 		Help: "The number of msg failed offline pushed", | 		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: | 	case share.RpcRegisterName.MessageGateway: | ||||||
| 		return []prometheus.Collector{OnlineUserGauge} | 		return []prometheus.Collector{OnlineUserGauge} | ||||||
| 	case share.RpcRegisterName.Msg: | 	case share.RpcRegisterName.Msg: | ||||||
| 		return []prometheus.Collector{SingleChatMsgProcessSuccessCounter, SingleChatMsgProcessFailedCounter, GroupChatMsgProcessSuccessCounter, GroupChatMsgProcessFailedCounter} | 		return []prometheus.Collector{ | ||||||
|  | 			SingleChatMsgProcessSuccessCounter, | ||||||
|  | 			SingleChatMsgProcessFailedCounter, | ||||||
|  | 			GroupChatMsgProcessSuccessCounter, | ||||||
|  | 			GroupChatMsgProcessFailedCounter, | ||||||
|  | 		} | ||||||
| 	case share.RpcRegisterName.Push: | 	case share.RpcRegisterName.Push: | ||||||
| 		return []prometheus.Collector{MsgOfflinePushFailedCounter} | 		return []prometheus.Collector{ | ||||||
|  | 			MsgOfflinePushFailedCounter, | ||||||
|  | 			MsgLoneTimePushCounter, | ||||||
|  | 		} | ||||||
| 	case share.RpcRegisterName.Auth: | 	case share.RpcRegisterName.Auth: | ||||||
| 		return []prometheus.Collector{UserLoginCounter} | 		return []prometheus.Collector{UserLoginCounter} | ||||||
| 	case share.RpcRegisterName.User: | 	case share.RpcRegisterName.User: | ||||||
|  | |||||||
| @ -25,7 +25,6 @@ import ( | |||||||
| 	"os" | 	"os" | ||||||
| 	"os/signal" | 	"os/signal" | ||||||
| 	"strconv" | 	"strconv" | ||||||
| 	"sync" |  | ||||||
| 	"syscall" | 	"syscall" | ||||||
| 	"time" | 	"time" | ||||||
| 
 | 
 | ||||||
| @ -35,7 +34,6 @@ import ( | |||||||
| 	"github.com/openimsdk/tools/errs" | 	"github.com/openimsdk/tools/errs" | ||||||
| 	"github.com/openimsdk/tools/log" | 	"github.com/openimsdk/tools/log" | ||||||
| 	"github.com/openimsdk/tools/mw" | 	"github.com/openimsdk/tools/mw" | ||||||
| 	"github.com/openimsdk/tools/system/program" |  | ||||||
| 	"github.com/openimsdk/tools/utils/network" | 	"github.com/openimsdk/tools/utils/network" | ||||||
| 	"google.golang.org/grpc" | 	"google.golang.org/grpc" | ||||||
| 	"google.golang.org/grpc/credentials/insecure" | 	"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, | 	log.CInfo(ctx, "RPC server is initializing", "rpcRegisterName", rpcRegisterName, "rpcPort", rpcPort, | ||||||
| 		"prometheusPorts", prometheusConfig.Ports) | 		"prometheusPorts", prometheusConfig.Ports) | ||||||
| 	rpcTcpAddr := net.JoinHostPort(network.GetListenIP(listenIP), strconv.Itoa(rpcPort)) | 	rpcTcpAddr := net.JoinHostPort(network.GetListenIP(listenIP), strconv.Itoa(rpcPort)) | ||||||
|  | 
 | ||||||
| 	listener, err := net.Listen( | 	listener, err := net.Listen( | ||||||
| 		"tcp", | 		"tcp", | ||||||
| 		rpcTcpAddr, | 		rpcTcpAddr, | ||||||
| @ -61,7 +60,6 @@ func Start[T any](ctx context.Context, discovery *config.Discovery, prometheusCo | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return errs.WrapMsg(err, "listen err", "rpcTcpAddr", rpcTcpAddr) | 		return errs.WrapMsg(err, "listen err", "rpcTcpAddr", rpcTcpAddr) | ||||||
| 	} | 	} | ||||||
| 
 |  | ||||||
| 	defer listener.Close() | 	defer listener.Close() | ||||||
| 	client, err := kdisc.NewDiscoveryRegister(discovery, share) | 	client, err := kdisc.NewDiscoveryRegister(discovery, share) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -92,10 +90,6 @@ func Start[T any](ctx context.Context, discovery *config.Discovery, prometheusCo | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	srv := grpc.NewServer(options...) | 	srv := grpc.NewServer(options...) | ||||||
| 	once := sync.Once{} |  | ||||||
| 	defer func() { |  | ||||||
| 		once.Do(srv.GracefulStop) |  | ||||||
| 	}() |  | ||||||
| 
 | 
 | ||||||
| 	err = rpcFn(ctx, config, client, srv) | 	err = rpcFn(ctx, config, client, srv) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -113,9 +107,8 @@ func Start[T any](ctx context.Context, discovery *config.Discovery, prometheusCo | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	var ( | 	var ( | ||||||
| 		netDone    = make(chan struct{}, 2) | 		netDone = make(chan struct{}, 2) | ||||||
| 		netErr     error | 		netErr  error | ||||||
| 		httpServer *http.Server |  | ||||||
| 	) | 	) | ||||||
| 	if prometheusConfig.Enable { | 	if prometheusConfig.Enable { | ||||||
| 		go func() { | 		go func() { | ||||||
| @ -152,18 +145,11 @@ func Start[T any](ctx context.Context, discovery *config.Discovery, prometheusCo | |||||||
| 	signal.Notify(sigs, syscall.SIGTERM) | 	signal.Notify(sigs, syscall.SIGTERM) | ||||||
| 	select { | 	select { | ||||||
| 	case <-sigs: | 	case <-sigs: | ||||||
| 		program.SIGTERMExit() | 		ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) | ||||||
| 		ctx, cancel := context.WithTimeout(context.Background(), 15*time.Second) |  | ||||||
| 		defer cancel() | 		defer cancel() | ||||||
| 		if err := gracefulStopWithCtx(ctx, srv.GracefulStop); err != nil { | 		if err := gracefulStopWithCtx(ctx, srv.GracefulStop); err != nil { | ||||||
| 			return err | 			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 | 		return nil | ||||||
| 	case <-netDone: | 	case <-netDone: | ||||||
| 		return netErr | 		return netErr | ||||||
|  | |||||||
| @ -17,6 +17,8 @@ package cachekey | |||||||
| const ( | const ( | ||||||
| 	ConversationKey                          = "CONVERSATION:" | 	ConversationKey                          = "CONVERSATION:" | ||||||
| 	ConversationIDsKey                       = "CONVERSATION_IDS:" | 	ConversationIDsKey                       = "CONVERSATION_IDS:" | ||||||
|  | 	NotNotifyConversationIDsKey              = "NOT_NOTIFY_CONVERSATION_IDS:" | ||||||
|  | 	PinnedConversationIDsKey                 = "PINNED_CONVERSATION_IDS:" | ||||||
| 	ConversationIDsHashKey                   = "CONVERSATION_IDS_HASH:" | 	ConversationIDsHashKey                   = "CONVERSATION_IDS_HASH:" | ||||||
| 	ConversationHasReadSeqKey                = "CONVERSATION_HAS_READ_SEQ:" | 	ConversationHasReadSeqKey                = "CONVERSATION_HAS_READ_SEQ:" | ||||||
| 	RecvMsgOptKey                            = "RECV_MSG_OPT:" | 	RecvMsgOptKey                            = "RECV_MSG_OPT:" | ||||||
| @ -34,6 +36,14 @@ func GetConversationIDsKey(ownerUserID string) string { | |||||||
| 	return ConversationIDsKey + ownerUserID | 	return ConversationIDsKey + ownerUserID | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func GetNotNotifyConversationIDsKey(ownerUserID string) string { | ||||||
|  | 	return NotNotifyConversationIDsKey + ownerUserID | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func GetPinnedConversationIDs(ownerUserID string) string { | ||||||
|  | 	return PinnedConversationIDsKey + ownerUserID | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func GetSuperGroupRecvNotNotifyUserIDsKey(groupID string) string { | func GetSuperGroupRecvNotNotifyUserIDsKey(groupID string) string { | ||||||
| 	return SuperGroupRecvMsgNotNotifyUserIDsKey + groupID | 	return SuperGroupRecvMsgNotNotifyUserIDsKey + groupID | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										21
									
								
								pkg/common/storage/cache/cachekey/group.go
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										21
									
								
								pkg/common/storage/cache/cachekey/group.go
									
									
									
									
										vendored
									
									
								
							| @ -20,16 +20,17 @@ import ( | |||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| const ( | const ( | ||||||
| 	groupExpireTime            = time.Second * 60 * 60 * 12 | 	groupExpireTime             = time.Second * 60 * 60 * 12 | ||||||
| 	GroupInfoKey               = "GROUP_INFO:" | 	GroupInfoKey                = "GROUP_INFO:" | ||||||
| 	GroupMemberIDsKey          = "GROUP_MEMBER_IDS:" | 	GroupMemberIDsKey           = "GROUP_MEMBER_IDS:" | ||||||
| 	GroupMembersHashKey        = "GROUP_MEMBERS_HASH2:" | 	GroupMembersHashKey         = "GROUP_MEMBERS_HASH2:" | ||||||
| 	GroupMemberInfoKey         = "GROUP_MEMBER_INFO:" | 	GroupMemberInfoKey          = "GROUP_MEMBER_INFO:" | ||||||
| 	JoinedGroupsKey            = "JOIN_GROUPS_KEY:" | 	JoinedGroupsKey             = "JOIN_GROUPS_KEY:" | ||||||
| 	GroupMemberNumKey          = "GROUP_MEMBER_NUM_CACHE:" | 	GroupMemberNumKey           = "GROUP_MEMBER_NUM_CACHE:" | ||||||
| 	GroupRoleLevelMemberIDsKey = "GROUP_ROLE_LEVEL_MEMBER_IDS:" | 	GroupRoleLevelMemberIDsKey  = "GROUP_ROLE_LEVEL_MEMBER_IDS:" | ||||||
| 	GroupMemberMaxVersionKey   = "GROUP_MEMBER_MAX_VERSION:" | 	GroupAdminLevelMemberIDsKey = "GROUP_ADMIN_LEVEL_MEMBER_IDS:" | ||||||
| 	GroupJoinMaxVersionKey     = "GROUP_JOIN_MAX_VERSION:" | 	GroupMemberMaxVersionKey    = "GROUP_MEMBER_MAX_VERSION:" | ||||||
|  | 	GroupJoinMaxVersionKey      = "GROUP_JOIN_MAX_VERSION:" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func GetGroupInfoKey(groupID string) string { | func GetGroupInfoKey(groupID string) string { | ||||||
|  | |||||||
							
								
								
									
										9
									
								
								pkg/common/storage/cache/cachekey/online.go
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								pkg/common/storage/cache/cachekey/online.go
									
									
									
									
										vendored
									
									
								
							| @ -1,6 +1,9 @@ | |||||||
| package cachekey | package cachekey | ||||||
| 
 | 
 | ||||||
| import "time" | import ( | ||||||
|  | 	"strings" | ||||||
|  | 	"time" | ||||||
|  | ) | ||||||
| 
 | 
 | ||||||
| const ( | const ( | ||||||
| 	OnlineKey     = "ONLINE:" | 	OnlineKey     = "ONLINE:" | ||||||
| @ -11,3 +14,7 @@ const ( | |||||||
| func GetOnlineKey(userID string) string { | func GetOnlineKey(userID string) string { | ||||||
| 	return OnlineKey + userID | 	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 | package cachekey | ||||||
| 
 | 
 | ||||||
| import "github.com/openimsdk/protocol/constant" | import ( | ||||||
|  | 	"github.com/openimsdk/protocol/constant" | ||||||
|  | 	"strings" | ||||||
|  | ) | ||||||
| 
 | 
 | ||||||
| const ( | const ( | ||||||
| 	UidPidToken = "UID_PID_TOKEN_STATUS:" | 	UidPidToken = "UID_PID_TOKEN_STATUS:" | ||||||
| @ -9,3 +12,17 @@ const ( | |||||||
| func GetTokenKey(userID string, platformID int) string { | func GetTokenKey(userID string, platformID int) string { | ||||||
| 	return UidPidToken + userID + ":" + constant.PlatformIDToName(platformID) | 	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 | 	CloneConversationCache() ConversationCache | ||||||
| 	// get user's conversationIDs from msgCache | 	// get user's conversationIDs from msgCache | ||||||
| 	GetUserConversationIDs(ctx context.Context, ownerUserID string) ([]string, error) | 	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 | 	DelConversationIDs(userIDs ...string) ConversationCache | ||||||
| 
 | 
 | ||||||
| 	GetUserConversationIDsHash(ctx context.Context, ownerUserID string) (hash uint64, err error) | 	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) | 	GetConversationNotReceiveMessageUserIDs(ctx context.Context, conversationID string) ([]string, error) | ||||||
| 	DelConversationNotReceiveMessageUserIDs(conversationIDs ...string) ConversationCache | 	DelConversationNotReceiveMessageUserIDs(conversationIDs ...string) ConversationCache | ||||||
| 
 | 	DelConversationNotNotifyMessageUserIDs(userIDs ...string) ConversationCache | ||||||
|  | 	DelConversationPinnedMessageUserIDs(userIDs ...string) ConversationCache | ||||||
| 	DelConversationVersionUserIDs(userIDs ...string) ConversationCache | 	DelConversationVersionUserIDs(userIDs ...string) ConversationCache | ||||||
| 
 | 
 | ||||||
| 	FindMaxConversationUserVersion(ctx context.Context, userID string) (*relationtb.VersionLog, error) | 	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 ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/common" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/common" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | 	"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 { | type OnlineCache interface { | ||||||
| 	GetOnline(ctx context.Context, userID string) ([]int32, error) | 	GetOnline(ctx context.Context, userID string) ([]int32, error) | ||||||
| 	SetUserOnline(ctx context.Context, userID string, online, offline []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" | 	"context" | ||||||
| 	"encoding/json" | 	"encoding/json" | ||||||
| 	"github.com/dtm-labs/rockscache" | 	"github.com/dtm-labs/rockscache" | ||||||
|  | 	"github.com/openimsdk/tools/errs" | ||||||
| 	"github.com/openimsdk/tools/log" | 	"github.com/openimsdk/tools/log" | ||||||
| 	"github.com/redis/go-redis/v9" | 	"github.com/redis/go-redis/v9" | ||||||
| 	"golang.org/x/sync/singleflight" | 	"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) | 				bs, err := json.Marshal(value) | ||||||
| 				if err != nil { | 				if err != nil { | ||||||
|  | 					log.ZError(ctx, "marshal failed", err) | ||||||
| 					return nil, err | 					return nil, err | ||||||
| 				} | 				} | ||||||
| 				cacheIndex[index] = string(bs) | 				cacheIndex[index] = string(bs) | ||||||
| @ -72,7 +74,7 @@ func batchGetCache2[K comparable, V any](ctx context.Context, rcClient *rockscac | |||||||
| 			return cacheIndex, nil | 			return cacheIndex, nil | ||||||
| 		}) | 		}) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return nil, err | 			return nil, errs.WrapMsg(err, "FetchBatch2 failed") | ||||||
| 		} | 		} | ||||||
| 		for index, data := range indexCache { | 		for index, data := range indexCache { | ||||||
| 			if data == "" { | 			if data == "" { | ||||||
| @ -80,7 +82,7 @@ func batchGetCache2[K comparable, V any](ctx context.Context, rcClient *rockscac | |||||||
| 			} | 			} | ||||||
| 			var value V | 			var value V | ||||||
| 			if err := json.Unmarshal([]byte(data), &value); err != nil { | 			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 { | 			if cb, ok := any(&value).(BatchCacheCallback[K]); ok { | ||||||
| 				cb.BatchCache(keyId[keys[index]]) | 				cb.BatchCache(keyId[keys[index]]) | ||||||
|  | |||||||
| @ -28,6 +28,10 @@ import ( | |||||||
| 	"time" | 	"time" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | const ( | ||||||
|  | 	rocksCacheTimeout = 11 * time.Second | ||||||
|  | ) | ||||||
|  | 
 | ||||||
| // BatchDeleterRedis is a concrete implementation of the BatchDeleter interface based on Redis and RocksCache. | // BatchDeleterRedis is a concrete implementation of the BatchDeleter interface based on Redis and RocksCache. | ||||||
| type BatchDeleterRedis struct { | type BatchDeleterRedis struct { | ||||||
| 	redisClient    redis.UniversalClient | 	redisClient    redis.UniversalClient | ||||||
| @ -106,6 +110,8 @@ func (c *BatchDeleterRedis) AddKeys(keys ...string) { | |||||||
| // GetRocksCacheOptions returns the default configuration options for RocksCache. | // GetRocksCacheOptions returns the default configuration options for RocksCache. | ||||||
| func GetRocksCacheOptions() *rockscache.Options { | func GetRocksCacheOptions() *rockscache.Options { | ||||||
| 	opts := rockscache.NewDefaultOptions() | 	opts := rockscache.NewDefaultOptions() | ||||||
|  | 	opts.LockExpire = rocksCacheTimeout | ||||||
|  | 	opts.WaitReplicasTimeout = rocksCacheTimeout | ||||||
| 	opts.StrongConsistency = true | 	opts.StrongConsistency = true | ||||||
| 	opts.RandomExpireAdjustment = 0.2 | 	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) { | 	v, err := rcClient.Fetch2(ctx, key, expire, func() (s string, err error) { | ||||||
| 		t, err = fn(ctx) | 		t, err = fn(ctx) | ||||||
| 		if err != nil { | 		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 | 			return "", err | ||||||
| 		} | 		} | ||||||
| 		bs, err := json.Marshal(t) | 		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) | 	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 { | func (c *ConversationRedisCache) getSuperGroupRecvNotNotifyUserIDsKey(groupID string) string { | ||||||
| 	return cachekey.GetSuperGroupRecvNotNotifyUserIDsKey(groupID) | 	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 { | func (c *ConversationRedisCache) DelConversationIDs(userIDs ...string) cache.ConversationCache { | ||||||
| 	keys := make([]string, 0, len(userIDs)) | 	keys := make([]string, 0, len(userIDs)) | ||||||
| 	for _, userID := range userIDs { | 	for _, userID := range userIDs { | ||||||
| @ -242,6 +262,22 @@ func (c *ConversationRedisCache) DelConversationNotReceiveMessageUserIDs(convers | |||||||
| 	return cache | 	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 { | func (c *ConversationRedisCache) DelConversationVersionUserIDs(userIDs ...string) cache.ConversationCache { | ||||||
| 	cache := c.CloneConversationCache() | 	cache := c.CloneConversationCache() | ||||||
| 	for _, userID := range userIDs { | 	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 ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 	"fmt" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache" | 	"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/open-im-server/v3/pkg/common/storage/cache/cachekey" | ||||||
|  | 	"github.com/openimsdk/protocol/constant" | ||||||
| 	"github.com/openimsdk/tools/errs" | 	"github.com/openimsdk/tools/errs" | ||||||
| 	"github.com/openimsdk/tools/log" | 	"github.com/openimsdk/tools/log" | ||||||
| 	"github.com/redis/go-redis/v9" | 	"github.com/redis/go-redis/v9" | ||||||
| @ -49,6 +51,36 @@ func (s *userOnline) GetOnline(ctx context.Context, userID string) ([]int32, err | |||||||
| 	return platformIDs, nil | 	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 { | func (s *userOnline) SetUserOnline(ctx context.Context, userID string, online, offline []int32) error { | ||||||
| 	script := ` | 	script := ` | ||||||
| 	local key = KEYS[1] | 	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/errs" | ||||||
| 	"github.com/openimsdk/tools/log" | 	"github.com/openimsdk/tools/log" | ||||||
| 	"github.com/redis/go-redis/v9" | 	"github.com/redis/go-redis/v9" | ||||||
|  | 	"strconv" | ||||||
| 	"time" | 	"time" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| @ -57,6 +58,14 @@ func (s *seqConversationCacheRedis) getSingleMaxSeq(ctx context.Context, convers | |||||||
| 	return map[string]int64{conversationID: seq}, nil | 	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 { | func (s *seqConversationCacheRedis) batchGetMaxSeq(ctx context.Context, keys []string, keyConversationID map[string]string, seqs map[string]int64) error { | ||||||
| 	result := make([]*redis.StringCmd, len(keys)) | 	result := make([]*redis.StringCmd, len(keys)) | ||||||
| 	pipe := s.rdb.Pipeline() | 	pipe := s.rdb.Pipeline() | ||||||
| @ -88,6 +97,46 @@ func (s *seqConversationCacheRedis) batchGetMaxSeq(ctx context.Context, keys []s | |||||||
| 	return nil | 	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) { | func (s *seqConversationCacheRedis) GetMaxSeqs(ctx context.Context, conversationIDs []string) (map[string]int64, error) { | ||||||
| 	switch len(conversationIDs) { | 	switch len(conversationIDs) { | ||||||
| 	case 0: | 	case 0: | ||||||
| @ -121,11 +170,44 @@ func (s *seqConversationCacheRedis) GetMaxSeqs(ctx context.Context, conversation | |||||||
| 	return seqs, nil | 	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 { | func (s *seqConversationCacheRedis) getSeqMallocKey(conversationID string) string { | ||||||
| 	return cachekey.GetMallocSeqKey(conversationID) | 	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 { | 	if lastSeq < currSeq { | ||||||
| 		return 0, errs.New("lastSeq must be greater than currSeq") | 		return 0, errs.New("lastSeq must be greater than currSeq") | ||||||
| 	} | 	} | ||||||
| @ -138,8 +220,9 @@ local lockValue = ARGV[1] | |||||||
| local dataSecond = ARGV[2] | local dataSecond = ARGV[2] | ||||||
| local curr_seq = tonumber(ARGV[3]) | local curr_seq = tonumber(ARGV[3]) | ||||||
| local last_seq = tonumber(ARGV[4]) | local last_seq = tonumber(ARGV[4]) | ||||||
|  | local mallocTime = ARGV[5] | ||||||
| if redis.call("EXISTS", key) == 0 then | 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) | 	redis.call("EXPIRE", key, dataSecond) | ||||||
| 	return 1 | 	return 1 | ||||||
| end | end | ||||||
| @ -147,11 +230,11 @@ if redis.call("HGET", key, "LOCK") ~= lockValue then | |||||||
| 	return 2 | 	return 2 | ||||||
| end | end | ||||||
| redis.call("HDEL", key, "LOCK") | 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) | redis.call("EXPIRE", key, dataSecond) | ||||||
| return 0 | 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 { | 	if err != nil { | ||||||
| 		return 0, errs.Wrap(err) | 		return 0, errs.Wrap(err) | ||||||
| 	} | 	} | ||||||
| @ -169,6 +252,7 @@ local key = KEYS[1] | |||||||
| local size = tonumber(ARGV[1]) | local size = tonumber(ARGV[1]) | ||||||
| local lockSecond = ARGV[2] | local lockSecond = ARGV[2] | ||||||
| local dataSecond = ARGV[3] | local dataSecond = ARGV[3] | ||||||
|  | local mallocTime = ARGV[4] | ||||||
| local result = {} | local result = {} | ||||||
| if redis.call("EXISTS", key) == 0 then | if redis.call("EXISTS", key) == 0 then | ||||||
| 	local lockValue = math.random(0, 999999999) | 	local lockValue = math.random(0, 999999999) | ||||||
| @ -176,6 +260,7 @@ if redis.call("EXISTS", key) == 0 then | |||||||
| 	redis.call("EXPIRE", key, lockSecond) | 	redis.call("EXPIRE", key, lockSecond) | ||||||
| 	table.insert(result, 1) | 	table.insert(result, 1) | ||||||
| 	table.insert(result, lockValue) | 	table.insert(result, lockValue) | ||||||
|  | 	table.insert(result, mallocTime) | ||||||
| 	return result | 	return result | ||||||
| end | end | ||||||
| if redis.call("HEXISTS", key, "LOCK") == 1 then | if redis.call("HEXISTS", key, "LOCK") == 1 then | ||||||
| @ -189,6 +274,12 @@ if size == 0 then | |||||||
| 	table.insert(result, 0) | 	table.insert(result, 0) | ||||||
| 	table.insert(result, curr_seq) | 	table.insert(result, curr_seq) | ||||||
| 	table.insert(result, last_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 | 	return result | ||||||
| end | end | ||||||
| local max_seq = curr_seq + size | local max_seq = curr_seq + size | ||||||
| @ -196,21 +287,25 @@ if max_seq > last_seq then | |||||||
| 	local lockValue = math.random(0, 999999999) | 	local lockValue = math.random(0, 999999999) | ||||||
| 	redis.call("HSET", key, "LOCK", lockValue) | 	redis.call("HSET", key, "LOCK", lockValue) | ||||||
| 	redis.call("HSET", key, "CURR", last_seq) | 	redis.call("HSET", key, "CURR", last_seq) | ||||||
|  | 	redis.call("HSET", key, "TIME", mallocTime) | ||||||
| 	redis.call("EXPIRE", key, lockSecond) | 	redis.call("EXPIRE", key, lockSecond) | ||||||
| 	table.insert(result, 3) | 	table.insert(result, 3) | ||||||
| 	table.insert(result, curr_seq) | 	table.insert(result, curr_seq) | ||||||
| 	table.insert(result, last_seq) | 	table.insert(result, last_seq) | ||||||
| 	table.insert(result, lockValue) | 	table.insert(result, lockValue) | ||||||
|  | 	table.insert(result, mallocTime) | ||||||
| 	return result | 	return result | ||||||
| end | end | ||||||
| redis.call("HSET", key, "CURR", max_seq) | redis.call("HSET", key, "CURR", max_seq) | ||||||
|  | redis.call("HSET", key, "TIME", ARGV[4]) | ||||||
| redis.call("EXPIRE", key, dataSecond) | redis.call("EXPIRE", key, dataSecond) | ||||||
| table.insert(result, 0) | table.insert(result, 0) | ||||||
| table.insert(result, curr_seq) | table.insert(result, curr_seq) | ||||||
| table.insert(result, last_seq) | table.insert(result, last_seq) | ||||||
|  | table.insert(result, mallocTime) | ||||||
| return result | 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 { | 	if err != nil { | ||||||
| 		return nil, errs.Wrap(err) | 		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++ { | 	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 { | 		if err != nil { | ||||||
| 			log.ZError(ctx, "set seq cache failed", err, "key", key, "owner", owner, "currSeq", currSeq, "lastSeq", lastSeq, "count", i+1) | 			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 { | 			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) { | 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 { | 	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) | 	key := s.getSeqMallocKey(conversationID) | ||||||
| 	for i := 0; i < 10; i++ { | 	for i := 0; i < 10; i++ { | ||||||
| 		states, err := s.malloc(ctx, key, size) | 		states, err := s.malloc(ctx, key, size) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return 0, err | 			return 0, 0, err | ||||||
| 		} | 		} | ||||||
| 		switch states[0] { | 		switch states[0] { | ||||||
| 		case 0: // success | 		case 0: // success | ||||||
| 			return states[1], nil | 			return states[1], states[3], nil | ||||||
| 		case 1: // not found | 		case 1: // not found | ||||||
| 			mallocSize := s.getMallocSize(conversationID, size) | 			mallocSize := s.getMallocSize(conversationID, size) | ||||||
| 			seq, err := s.mgo.Malloc(ctx, conversationID, mallocSize) | 			seq, err := s.mgo.Malloc(ctx, conversationID, mallocSize) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				return 0, err | 				return 0, 0, err | ||||||
| 			} | 			} | ||||||
| 			s.setSeqRetry(ctx, key, states[1], seq+size, seq+mallocSize) | 			s.setSeqRetry(ctx, key, states[1], seq+size, seq+mallocSize, states[2]) | ||||||
| 			return seq, nil | 			return seq, 0, nil | ||||||
| 		case 2: // locked | 		case 2: // locked | ||||||
| 			if err := s.wait(ctx); err != nil { | 			if err := s.wait(ctx); err != nil { | ||||||
| 				return 0, err | 				return 0, 0, err | ||||||
| 			} | 			} | ||||||
| 			continue | 			continue | ||||||
| 		case 3: // exceeded cache max value | 		case 3: // exceeded cache max value | ||||||
| 			currSeq := states[1] | 			currSeq := states[1] | ||||||
| 			lastSeq := states[2] | 			lastSeq := states[2] | ||||||
|  | 			mill := states[4] | ||||||
| 			mallocSize := s.getMallocSize(conversationID, size) | 			mallocSize := s.getMallocSize(conversationID, size) | ||||||
| 			seq, err := s.mgo.Malloc(ctx, conversationID, mallocSize) | 			seq, err := s.mgo.Malloc(ctx, conversationID, mallocSize) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				return 0, err | 				return 0, 0, err | ||||||
| 			} | 			} | ||||||
| 			if lastSeq == seq { | 			if lastSeq == seq { | ||||||
| 				s.setSeqRetry(ctx, key, states[3], currSeq+size, seq+mallocSize) | 				s.setSeqRetry(ctx, key, states[3], currSeq+size, seq+mallocSize, mill) | ||||||
| 				return currSeq, nil | 				return currSeq, states[4], nil | ||||||
| 			} else { | 			} else { | ||||||
| 				log.ZWarn(ctx, "malloc seq not equal cache last seq", nil, "conversationID", conversationID, "currSeq", currSeq, "lastSeq", lastSeq, "mallocSeq", seq) | 				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) | 				s.setSeqRetry(ctx, key, states[3], seq+size, seq+mallocSize, mill) | ||||||
| 				return seq, nil | 				return seq, mill, nil | ||||||
| 			} | 			} | ||||||
| 		default: | 		default: | ||||||
| 			log.ZError(ctx, "malloc seq unknown state", nil, "state", states[0], "conversationID", conversationID, "size", size) | 			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) | 	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) { | func (s *seqConversationCacheRedis) GetMaxSeq(ctx context.Context, conversationID string) (int64, error) { | ||||||
| 	return s.Malloc(ctx, conversationID, 0) | 	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 { | func (s *seqConversationCacheRedis) SetMinSeqs(ctx context.Context, seqs map[string]int64) error { | ||||||
| 	keys := make([]string, 0, len(seqs)) | 	keys := make([]string, 0, len(seqs)) | ||||||
| 	for conversationID, seq := range 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) | 	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