mirror of
				https://github.com/openimsdk/open-im-server.git
				synced 2025-10-26 21:22:16 +08:00 
			
		
		
		
	Merge branch 'main' into fix-fcm-push
This commit is contained in:
		
						commit
						2052eacd5e
					
				
							
								
								
									
										59
									
								
								.github/workflows/go-build-test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										59
									
								
								.github/workflows/go-build-test.yml
									
									
									
									
										vendored
									
									
								
							| @ -89,6 +89,65 @@ jobs: | ||||
|           mage start | ||||
|           mage check | ||||
| 
 | ||||
|   go-test: | ||||
|     name: Benchmark Test with go ${{ matrix.go_version }} on ${{ matrix.os }} | ||||
|     runs-on: ${{ matrix.os }} | ||||
|     permissions: | ||||
|       contents: write | ||||
|     env: | ||||
|       SDK_DIR: openim-sdk-core | ||||
|       CONFIG_PATH: config/notification.yml | ||||
|     #   pull-requests: write | ||||
|     strategy: | ||||
|       matrix: | ||||
|         os: [ ubuntu-latest ] | ||||
|         go_version: [ "1.22.x" ] | ||||
| 
 | ||||
|     steps: | ||||
|       - name: Checkout Server repository | ||||
|         uses: actions/checkout@v4 | ||||
| 
 | ||||
|       - name: Checkout SDK repository | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           repository: 'openimsdk/openim-sdk-core' | ||||
|           path: ${{ env.SDK_DIR }} | ||||
| 
 | ||||
|       - name: Set up Go ${{ matrix.go_version }} | ||||
|         uses: actions/setup-go@v5 | ||||
|         with: | ||||
|           go-version: ${{ matrix.go_version }} | ||||
| 
 | ||||
|       - name: Get Server dependencies | ||||
|         run: | | ||||
|           go install github.com/magefile/mage@latest | ||||
|           go mod download | ||||
| 
 | ||||
|       - name: Install yq | ||||
|         run: | | ||||
|           sudo wget https://github.com/mikefarah/yq/releases/download/v4.34.1/yq_linux_amd64 -O /usr/bin/yq | ||||
|           sudo chmod +x /usr/bin/yq | ||||
| 
 | ||||
|       - name: Modify Server Configuration | ||||
|         run: | | ||||
|           yq e '.groupCreated.unreadCount = true' -i ${{ env.CONFIG_PATH }} | ||||
|           yq e '.friendApplicationApproved.unreadCount = true' -i ${{ env.CONFIG_PATH }} | ||||
| 
 | ||||
|       - name: Start Server Services | ||||
|         run: | | ||||
|           docker compose up -d | ||||
|           mage build | ||||
|           mage start | ||||
|           mage check | ||||
| 
 | ||||
|       - name: Build test SDK core | ||||
|         run: | | ||||
|           cd ${{ env.SDK_DIR }} | ||||
|           go mod tidy | ||||
|           cd integration_test | ||||
|           mkdir data | ||||
|           go run main.go -lgr 0.8 -imf -crg -ckgn -ckcon -sem -ckmsn -u 20 -su 5 -lg 2 -cg 2 -cgm 3 -sm 10 -gm 10 -reg | ||||
| 
 | ||||
|   dockerfile-test: | ||||
|     name: Build and Test Dockerfile | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
| @ -23,5 +23,4 @@ longConnSvr: | ||||
|   # WebSocket connection handshake timeout in seconds | ||||
|   websocketTimeout: 10 | ||||
| 
 | ||||
| # 1: For Android, iOS, Windows, Mac, and web platforms, only one instance can be online at a time | ||||
| multiLoginPolicy: 1 | ||||
| 
 | ||||
|  | ||||
| @ -3,4 +3,4 @@ prometheus: | ||||
|   enable: true | ||||
|   # List of ports that Prometheus listens on; each port corresponds to an instance of monitoring. Ensure these are managed accordingly | ||||
|   # Because four instances have been launched, four ports need to be specified | ||||
|   ports: [ 12020, 12021, 12022, 12023, 12024, 12025, 12026, 12027 ] | ||||
|   ports: [ 12020, 12021, 12022, 12023, 12024, 12025, 12026, 12027, 12028, 12029, 12030, 12031, 12032, 12033, 12034, 12035 ] | ||||
|  | ||||
| @ -10,7 +10,7 @@ prometheus: | ||||
|   # Enable or disable Prometheus monitoring | ||||
|   enable: true | ||||
|   # List of ports that Prometheus listens on; these must match the number of rpc.ports to ensure correct monitoring setup | ||||
|   ports: [ 12170, 12171, 12172, 12173, 12174, 12175, 12176, 12177, 12178, 12179, 12180, 12181, 12182, 12183, 12184, 12185 ] | ||||
|   ports: [ 12170, 12171, 12172, 12173, 12174, 12175, 12176, 12177, 12178, 12179, 12180, 12182, 12183, 12184, 12185, 12186 ] | ||||
| 
 | ||||
| maxConcurrentWorkers: 3 | ||||
| #Use geTui for offline push notifications, or choose fcm or jpns; corresponding configuration settings must be specified. | ||||
|  | ||||
| @ -46,7 +46,7 @@ scrape_configs: | ||||
|   - job_name: openimserver-openim-push | ||||
|     static_configs: | ||||
|       - 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:12181, internal_ip:12182, internal_ip:12183, internal_ip:12184, internal_ip:12185 ] | ||||
| #      - targets: [ internal_ip:12170, internal_ip:12171, internal_ip:12172, internal_ip:12173, internal_ip:12174, internal_ip:12175, internal_ip:12176, internal_ip:12177, internal_ip:12178, internal_ip:12179, internal_ip:12180,  internal_ip:12182, internal_ip:12183, internal_ip:12184, internal_ip:12185, internal_ip:12186 ] | ||||
|         labels: | ||||
|           namespace: default | ||||
|   - job_name: openimserver-openim-rpc-auth | ||||
|  | ||||
| @ -12,3 +12,5 @@ rpcRegisterName: | ||||
| 
 | ||||
| imAdminUserID: [ imAdmin ] | ||||
| 
 | ||||
| # 1: For Android, iOS, Windows, Mac, and web platforms, only one instance can be online at a time | ||||
| multiLoginPolicy: 1 | ||||
|  | ||||
| @ -130,10 +130,10 @@ beforeSetGroupInfo: | ||||
|   enable: false | ||||
|   timeout: 5 | ||||
|   failedContinue: true | ||||
| afterSetGroupInfoEX: | ||||
| afterSetGroupInfoEx: | ||||
|   enable: false | ||||
|   timeout: 5 | ||||
| beforeSetGroupInfoEX: | ||||
| beforeSetGroupInfoEx: | ||||
|   enable: false | ||||
|   timeout: 5 | ||||
|   failedContinue: true | ||||
|  | ||||
| @ -1,5 +1,3 @@ | ||||
| version: '3' | ||||
| 
 | ||||
| networks: | ||||
|   openim: | ||||
|     driver: bridge | ||||
|  | ||||
							
								
								
									
										27
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										27
									
								
								go.mod
									
									
									
									
									
								
							| @ -12,13 +12,13 @@ require ( | ||||
| 	github.com/gorilla/websocket v1.5.1 | ||||
| 	github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0 | ||||
| 	github.com/mitchellh/mapstructure v1.5.0 | ||||
| 	github.com/openimsdk/protocol v0.0.72-alpha.21 | ||||
| 	github.com/openimsdk/protocol v0.0.72-alpha.27 | ||||
| 	github.com/openimsdk/tools v0.0.50-alpha.12 | ||||
| 	github.com/pkg/errors v0.9.1 // indirect | ||||
| 	github.com/prometheus/client_golang v1.18.0 | ||||
| 	github.com/stretchr/testify v1.9.0 | ||||
| 	go.mongodb.org/mongo-driver v1.14.0 | ||||
| 	google.golang.org/api v0.165.0 | ||||
| 	google.golang.org/api v0.170.0 | ||||
| 	google.golang.org/grpc v1.66.2 | ||||
| 	google.golang.org/protobuf v1.34.2 | ||||
| 	gopkg.in/yaml.v3 v3.0.1 | ||||
| @ -27,6 +27,7 @@ require ( | ||||
| require github.com/google/uuid v1.6.0 | ||||
| 
 | ||||
| require ( | ||||
| 	firebase.google.com/go/v4 v4.14.1 | ||||
| 	github.com/IBM/sarama v1.43.0 | ||||
| 	github.com/fatih/color v1.14.1 | ||||
| 	github.com/gin-contrib/gzip v1.0.1 | ||||
| @ -47,12 +48,12 @@ require ( | ||||
| ) | ||||
| 
 | ||||
| require ( | ||||
| 	cloud.google.com/go v0.112.0 // indirect | ||||
| 	cloud.google.com/go v0.112.1 // indirect | ||||
| 	cloud.google.com/go/compute/metadata v0.3.0 // indirect | ||||
| 	cloud.google.com/go/firestore v1.14.0 // indirect | ||||
| 	cloud.google.com/go/iam v1.1.5 // indirect | ||||
| 	cloud.google.com/go/longrunning v0.5.4 // indirect | ||||
| 	cloud.google.com/go/storage v1.36.0 // indirect | ||||
| 	cloud.google.com/go/firestore v1.15.0 // indirect | ||||
| 	cloud.google.com/go/iam v1.1.7 // indirect | ||||
| 	cloud.google.com/go/longrunning v0.5.5 // indirect | ||||
| 	cloud.google.com/go/storage v1.40.0 // indirect | ||||
| 	github.com/MicahParks/keyfunc v1.9.0 // indirect | ||||
| 	github.com/aliyun/aliyun-oss-go-sdk v3.0.2+incompatible // indirect | ||||
| 	github.com/aws/aws-sdk-go-v2 v1.23.1 // indirect | ||||
| @ -103,7 +104,7 @@ require ( | ||||
| 	github.com/google/go-querystring v1.1.0 // indirect | ||||
| 	github.com/google/s2a-go v0.1.7 // indirect | ||||
| 	github.com/googleapis/enterprise-certificate-proxy v0.3.2 // indirect | ||||
| 	github.com/googleapis/gax-go/v2 v2.12.0 // indirect | ||||
| 	github.com/googleapis/gax-go/v2 v2.12.3 // indirect | ||||
| 	github.com/hashicorp/errwrap v1.1.0 // indirect | ||||
| 	github.com/hashicorp/go-multierror v1.1.1 // indirect | ||||
| 	github.com/hashicorp/go-uuid v1.0.3 // indirect | ||||
| @ -165,11 +166,11 @@ require ( | ||||
| 	go.etcd.io/etcd/client/pkg/v3 v3.5.13 // indirect | ||||
| 	go.etcd.io/etcd/client/v3 v3.5.13 // indirect | ||||
| 	go.opencensus.io v0.24.0 // indirect | ||||
| 	go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.47.0 // indirect | ||||
| 	go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.47.0 // indirect | ||||
| 	go.opentelemetry.io/otel v1.23.0 // indirect | ||||
| 	go.opentelemetry.io/otel/metric v1.23.0 // indirect | ||||
| 	go.opentelemetry.io/otel/trace v1.23.0 // indirect | ||||
| 	go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.49.0 // indirect | ||||
| 	go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.49.0 // indirect | ||||
| 	go.opentelemetry.io/otel v1.24.0 // indirect | ||||
| 	go.opentelemetry.io/otel/metric v1.24.0 // indirect | ||||
| 	go.opentelemetry.io/otel/trace v1.24.0 // indirect | ||||
| 	go.uber.org/atomic v1.9.0 // indirect | ||||
| 	go.uber.org/multierr v1.11.0 // indirect | ||||
| 	golang.org/x/arch v0.7.0 // indirect | ||||
|  | ||||
							
								
								
									
										77
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										77
									
								
								go.sum
									
									
									
									
									
								
							| @ -1,18 +1,18 @@ | ||||
| cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= | ||||
| cloud.google.com/go v0.112.0 h1:tpFCD7hpHFlQ8yPwT3x+QeXqc2T6+n6T+hmABHfDUSM= | ||||
| cloud.google.com/go v0.112.0/go.mod h1:3jEEVwZ/MHU4djK5t5RHuKOA/GbLddgTdVubX1qnPD4= | ||||
| cloud.google.com/go v0.112.1 h1:uJSeirPke5UNZHIb4SxfZklVSiWWVqW4oXlETwZziwM= | ||||
| cloud.google.com/go v0.112.1/go.mod h1:+Vbu+Y1UU+I1rjmzeMOb/8RfkKJK2Gyxi1X6jJCZLo4= | ||||
| cloud.google.com/go/compute/metadata v0.3.0 h1:Tz+eQXMEqDIKRsmY3cHTL6FVaynIjX2QxYC4trgAKZc= | ||||
| cloud.google.com/go/compute/metadata v0.3.0/go.mod h1:zFmK7XCadkQkj6TtorcaGlCW1hT1fIilQDwofLpJ20k= | ||||
| cloud.google.com/go/firestore v1.14.0 h1:8aLcKnMPoldYU3YHgu4t2exrKhLQkqaXAGqT0ljrFVw= | ||||
| cloud.google.com/go/firestore v1.14.0/go.mod h1:96MVaHLsEhbvkBEdZgfN+AS/GIkco1LRpH9Xp9YZfzQ= | ||||
| cloud.google.com/go/iam v1.1.5 h1:1jTsCu4bcsNsE4iiqNT5SHwrDRCfRmIaaaVFhRveTJI= | ||||
| cloud.google.com/go/iam v1.1.5/go.mod h1:rB6P/Ic3mykPbFio+vo7403drjlgvoWfYpJhMXEbzv8= | ||||
| cloud.google.com/go/longrunning v0.5.4 h1:w8xEcbZodnA2BbW6sVirkkoC+1gP8wS57EUUgGS0GVg= | ||||
| cloud.google.com/go/longrunning v0.5.4/go.mod h1:zqNVncI0BOP8ST6XQD1+VcvuShMmq7+xFSzOL++V0dI= | ||||
| cloud.google.com/go/storage v1.36.0 h1:P0mOkAcaJxhCTvAkMhxMfrTKiNcub4YmmPBtlhAyTr8= | ||||
| cloud.google.com/go/storage v1.36.0/go.mod h1:M6M/3V/D3KpzMTJyPOR/HU6n2Si5QdaXYEsng2xgOs8= | ||||
| firebase.google.com/go/v4 v4.13.0 h1:meFz9nvDNh/FDyrEykoAzSfComcQbmnQSjoHrePRqeI= | ||||
| firebase.google.com/go/v4 v4.13.0/go.mod h1:e1/gaR6EnbQfsmTnAMx1hnz+ninJIrrr/RAh59Tpfn8= | ||||
| cloud.google.com/go/firestore v1.15.0 h1:/k8ppuWOtNuDHt2tsRV42yI21uaGnKDEQnRFeBpbFF8= | ||||
| cloud.google.com/go/firestore v1.15.0/go.mod h1:GWOxFXcv8GZUtYpWHw/w6IuYNux/BtmeVTMmjrm4yhk= | ||||
| cloud.google.com/go/iam v1.1.7 h1:z4VHOhwKLF/+UYXAJDFwGtNF0b6gjsW1Pk9Ml0U/IoM= | ||||
| cloud.google.com/go/iam v1.1.7/go.mod h1:J4PMPg8TtyurAUvSmPj8FF3EDgY1SPRZxcUGrn7WXGA= | ||||
| cloud.google.com/go/longrunning v0.5.5 h1:GOE6pZFdSrTb4KAiKnXsJBtlE6mEyaW44oKyMILWnOg= | ||||
| cloud.google.com/go/longrunning v0.5.5/go.mod h1:WV2LAxD8/rg5Z1cNW6FJ/ZpX4E4VnDnoTk0yawPBB7s= | ||||
| cloud.google.com/go/storage v1.40.0 h1:VEpDQV5CJxFmJ6ueWNsKxcr1QAYOXEgxDa+sBbJahPw= | ||||
| cloud.google.com/go/storage v1.40.0/go.mod h1:Rrj7/hKlG87BLqDJYtwR0fbPld8uJPbQ2ucUMY7Ir0g= | ||||
| firebase.google.com/go/v4 v4.14.1 h1:4qiUETaFRWoFGE1XP5VbcEdtPX93Qs+8B/7KvP2825g= | ||||
| firebase.google.com/go/v4 v4.14.1/go.mod h1:fgk2XshgNDEKaioKco+AouiegSI9oTWVqRaBdTTGBoM= | ||||
| github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= | ||||
| github.com/IBM/sarama v1.43.0 h1:YFFDn8mMI2QL0wOrG0J2sFoVIAFl7hS9JQi2YZsXtJc= | ||||
| github.com/IBM/sarama v1.43.0/go.mod h1:zlE6HEbC/SMQ9mhEYaF7nNLYOUyrs0obySKCckWP9BM= | ||||
| @ -80,8 +80,6 @@ github.com/cloudwego/base64x v0.1.4/go.mod h1:0zlkT4Wn5C6NdauXdJRhSKRlJvmclQ1hhJ | ||||
| github.com/cloudwego/iasm v0.2.0 h1:1KNIy1I1H9hNNFEEH3DVnI4UujN+1zjpuk6gwHLTssg= | ||||
| github.com/cloudwego/iasm v0.2.0/go.mod h1:8rXZaNYT2n95jn+zTI1sDr+IgcD2GVs0nlbbQPiEFhY= | ||||
| github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc= | ||||
| github.com/cncf/xds/go v0.0.0-20240423153145-555b57ec207b h1:ga8SEFjZ60pxLcmhnThWgvH2wg8376yUJmPhEH4H3kw= | ||||
| github.com/cncf/xds/go v0.0.0-20240423153145-555b57ec207b/go.mod h1:W+zGtBO5Y1IgJhy4+A9GOqVhqLpfZi+vwmdNXUehLA8= | ||||
| github.com/coreos/go-semver v0.3.0 h1:wkHLiw0WNATZnSG7epLsujiMCgPAc9xhjJ4tgnAxmfM= | ||||
| github.com/coreos/go-semver v0.3.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk= | ||||
| github.com/coreos/go-systemd/v22 v22.3.2 h1:D9/bQk5vlXQFZ6Kwuu6zaiXJ9oTPe68++AzAJc1DzSI= | ||||
| @ -109,8 +107,6 @@ github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymF | ||||
| github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= | ||||
| github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98= | ||||
| github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= | ||||
| github.com/envoyproxy/protoc-gen-validate v1.0.4 h1:gVPz/FMfvh57HdSJQyvBtF00j8JU4zdyUgIUNhlgg0A= | ||||
| github.com/envoyproxy/protoc-gen-validate v1.0.4/go.mod h1:qys6tmnRsYrQqIhm2bvKZH4Blx/1gTIZ2UKVY1M+Yew= | ||||
| github.com/fatih/color v1.14.1 h1:qfhVLaG5s+nCROl1zJsZRxFeYrHLqWroPOQ8BWiNb4w= | ||||
| github.com/fatih/color v1.14.1/go.mod h1:2oHN61fhTpgcxD3TSWCgKDiH1+x4OiDVVGH8WlgGZGg= | ||||
| github.com/felixge/httpsnoop v1.0.4 h1:NFTV2Zj1bL4mc9sqWACXbQFVBBg2W3GPvqp8/ESS2Wg= | ||||
| @ -179,8 +175,6 @@ github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:W | ||||
| github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0= | ||||
| github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8= | ||||
| github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= | ||||
| github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= | ||||
| github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= | ||||
| github.com/golang/protobuf v1.5.4 h1:i7eJL8qZTpSEXOPTxNKhASYpMn+8e5Q6AdndVa1dWek= | ||||
| github.com/golang/protobuf v1.5.4/go.mod h1:lnTiLA8Wa4RWRcIUkrtSVa5nRhsEGBg48fD6rSs7xps= | ||||
| github.com/golang/snappy v0.0.4 h1:yAGX7huGHXlcLOEtBnF4w7FQwA26wojNCwOYAEhLjQM= | ||||
| @ -192,7 +186,6 @@ github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/ | ||||
| github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.2/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.3/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= | ||||
| github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= | ||||
| github.com/google/go-querystring v1.0.0/go.mod h1:odCYkC5MyYFN7vkCjXpyrEuKhc/BUO6wN/zVPAxq5ck= | ||||
| @ -209,8 +202,8 @@ github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= | ||||
| github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/googleapis/enterprise-certificate-proxy v0.3.2 h1:Vie5ybvEvT75RniqhfFxPRy3Bf7vr3h0cechB90XaQs= | ||||
| github.com/googleapis/enterprise-certificate-proxy v0.3.2/go.mod h1:VLSiSSBs/ksPL8kq3OBOQ6WRI2QnaFynd1DCjZ62+V0= | ||||
| github.com/googleapis/gax-go/v2 v2.12.0 h1:A+gCJKdRfqXkr+BIRGtZLibNXf0m1f9E4HG56etFpas= | ||||
| github.com/googleapis/gax-go/v2 v2.12.0/go.mod h1:y+aIqrI5eb1YGMVJfuV3185Ts/D7qKpsEkdD5+I6QGU= | ||||
| github.com/googleapis/gax-go/v2 v2.12.3 h1:5/zPPDvw8Q1SuXjrqrZslrqT7dL/uJT2CQii/cLCKqA= | ||||
| github.com/googleapis/gax-go/v2 v2.12.3/go.mod h1:AKloxT6GtNbaLm8QTNSidHUVsHYcBHwWRvkNFJUQcS4= | ||||
| github.com/gorilla/securecookie v1.1.1/go.mod h1:ra0sb63/xPlUeL+yeDciTfxMRAA+MP+HVt/4epWDjd4= | ||||
| github.com/gorilla/sessions v1.2.1/go.mod h1:dk2InVEVJ0sfLlnXv9EAgkf6ecYs/i80K/zI+bUmuGM= | ||||
| github.com/gorilla/websocket v1.5.1 h1:gmztn0JnHVt9JZquRuzLw3g4wouNVzKL15iLr/zn/QY= | ||||
| @ -326,8 +319,8 @@ github.com/onsi/gomega v1.25.0 h1:Vw7br2PCDYijJHSfBOWhov+8cAnUf8MfMaIOV323l6Y= | ||||
| github.com/onsi/gomega v1.25.0/go.mod h1:r+zV744Re+DiYCIPRlYOTxn0YkOLcAnW8k1xXdMPGhM= | ||||
| github.com/openimsdk/gomake v0.0.14-alpha.5 h1:VY9c5x515lTfmdhhPjMvR3BBRrRquAUCFsz7t7vbv7Y= | ||||
| github.com/openimsdk/gomake v0.0.14-alpha.5/go.mod h1:PndCozNc2IsQIciyn9mvEblYWZwJmAI+06z94EY+csI= | ||||
| github.com/openimsdk/protocol v0.0.72-alpha.21 h1:MRSFDHVRsFymglbv2FSGPtiKo4RXZDTBwQTWNWiUf/U= | ||||
| github.com/openimsdk/protocol v0.0.72-alpha.21/go.mod h1:OZQA9FR55lseYoN2Ql1XAHYKHJGu7OMNkUbuekrKCM8= | ||||
| github.com/openimsdk/protocol v0.0.72-alpha.27 h1:S6n3uj7YhKjo2NCHHSnUijaJ9YYiy8TTMquc4EJOm50= | ||||
| github.com/openimsdk/protocol v0.0.72-alpha.27/go.mod h1:OZQA9FR55lseYoN2Ql1XAHYKHJGu7OMNkUbuekrKCM8= | ||||
| github.com/openimsdk/tools v0.0.50-alpha.12 h1:rV3BxgqN+F79vZvdoQ+97Eob8ScsRVEM8D+Wrcl23uo= | ||||
| github.com/openimsdk/tools v0.0.50-alpha.12/go.mod h1:h1cYmfyaVtgFbKmb1Cfsl8XwUOMTt8ubVUQrdGtsUh4= | ||||
| github.com/pelletier/go-toml/v2 v2.2.2 h1:aYUidT7k73Pcl9nb2gScu7NSrKCSHIDE89b3+6Wq+LM= | ||||
| @ -440,18 +433,18 @@ go.mongodb.org/mongo-driver v1.14.0 h1:P98w8egYRjYe3XDjxhYJagTokP/H6HzlsnojRgZRd | ||||
| go.mongodb.org/mongo-driver v1.14.0/go.mod h1:Vzb0Mk/pa7e6cWw85R4F/endUC3u0U9jGcNU603k65c= | ||||
| go.opencensus.io v0.24.0 h1:y73uSU6J157QMP2kn2r30vwW1A2W2WFwSCGnAVxeaD0= | ||||
| go.opencensus.io v0.24.0/go.mod h1:vNK8G9p7aAivkbmorf4v+7Hgx+Zs0yY+0fOtgBfjQKo= | ||||
| go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.47.0 h1:UNQQKPfTDe1J81ViolILjTKPr9WetKW6uei2hFgJmFs= | ||||
| go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.47.0/go.mod h1:r9vWsPS/3AQItv3OSlEJ/E4mbrhUbbw18meOjArPtKQ= | ||||
| go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.47.0 h1:sv9kVfal0MK0wBMCOGr+HeJm9v803BkJxGrk2au7j08= | ||||
| go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.47.0/go.mod h1:SK2UL73Zy1quvRPonmOmRDiWk1KBV3LyIeeIxcEApWw= | ||||
| go.opentelemetry.io/otel v1.23.0 h1:Df0pqjqExIywbMCMTxkAwzjLZtRf+bBKLbUcpxO2C9E= | ||||
| go.opentelemetry.io/otel v1.23.0/go.mod h1:YCycw9ZeKhcJFrb34iVSkyT0iczq/zYDtZYFufObyB0= | ||||
| go.opentelemetry.io/otel/metric v1.23.0 h1:pazkx7ss4LFVVYSxYew7L5I6qvLXHA0Ap2pwV+9Cnpo= | ||||
| go.opentelemetry.io/otel/metric v1.23.0/go.mod h1:MqUW2X2a6Q8RN96E2/nqNoT+z9BSms20Jb7Bbp+HiTo= | ||||
| go.opentelemetry.io/otel/sdk v1.21.0 h1:FTt8qirL1EysG6sTQRZ5TokkU8d0ugCj8htOgThZXQ8= | ||||
| go.opentelemetry.io/otel/sdk v1.21.0/go.mod h1:Nna6Yv7PWTdgJHVRD9hIYywQBRx7pbox6nwBnZIxl/E= | ||||
| go.opentelemetry.io/otel/trace v1.23.0 h1:37Ik5Ib7xfYVb4V1UtnT97T1jI+AoIYkJyPkuL4iJgI= | ||||
| go.opentelemetry.io/otel/trace v1.23.0/go.mod h1:GSGTbIClEsuZrGIzoEHqsVfxgn5UkggkflQwDScNUsk= | ||||
| go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.49.0 h1:4Pp6oUg3+e/6M4C0A/3kJ2VYa++dsWVTtGgLVj5xtHg= | ||||
| go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.49.0/go.mod h1:Mjt1i1INqiaoZOMGR1RIUJN+i3ChKoFRqzrRQhlkbs0= | ||||
| go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.49.0 h1:jq9TW8u3so/bN+JPT166wjOI6/vQPF6Xe7nMNIltagk= | ||||
| go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.49.0/go.mod h1:p8pYQP+m5XfbZm9fxtSKAbM6oIllS7s2AfxrChvc7iw= | ||||
| go.opentelemetry.io/otel v1.24.0 h1:0LAOdjNmQeSTzGBzduGe/rU4tZhMwL5rWgtp9Ku5Jfo= | ||||
| go.opentelemetry.io/otel v1.24.0/go.mod h1:W7b9Ozg4nkF5tWI5zsXkaKKDjdVjpD4oAt9Qi/MArHo= | ||||
| go.opentelemetry.io/otel/metric v1.24.0 h1:6EhoGWWK28x1fbpA4tYTOWBkPefTDQnb8WSGXlc88kI= | ||||
| go.opentelemetry.io/otel/metric v1.24.0/go.mod h1:VYhLe1rFfxuTXLgj4CBiyz+9WYBA8pNGJgDcSFRKBco= | ||||
| go.opentelemetry.io/otel/sdk v1.22.0 h1:6coWHw9xw7EfClIC/+O31R8IY3/+EiRFHevmHafB2Gw= | ||||
| go.opentelemetry.io/otel/sdk v1.22.0/go.mod h1:iu7luyVGYovrRpe2fmj3CVKouQNdTOkxtLzPvPz1DOc= | ||||
| go.opentelemetry.io/otel/trace v1.24.0 h1:CsKnnL4dUAr/0llH9FKuc698G04IrpWV0MQA/Y1YELI= | ||||
| go.opentelemetry.io/otel/trace v1.24.0/go.mod h1:HPc3Xr/cOApsBI154IU0OI0HJexz+aw5uPdbs3UCjNU= | ||||
| go.uber.org/atomic v1.9.0 h1:ECmE8Bn/WFTYwEW/bpKD3M8VtR/zQVbavAoalC1PYyE= | ||||
| go.uber.org/atomic v1.9.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc= | ||||
| go.uber.org/automaxprocs v1.5.3 h1:kWazyxZUrS3Gs4qUpbwo5kEIMGe/DAvi5Z4tl2NW4j8= | ||||
| @ -559,10 +552,10 @@ golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8T | ||||
| golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20220907171357-04be3eba64a2 h1:H2TDz8ibqkAF6YGhCdN3jS9O0/s90v0rJh3X/OLHEUk= | ||||
| golang.org/x/xerrors v0.0.0-20220907171357-04be3eba64a2/go.mod h1:K8+ghG5WaK9qNqU5K3HdILfMLy1f3aNYFI/wnl100a8= | ||||
| google.golang.org/api v0.165.0 h1:zd5d4JIIIaYYsfVy1HzoXYZ9rWCSBxxAglbczzo7Bgc= | ||||
| google.golang.org/api v0.165.0/go.mod h1:2OatzO7ZDQsoS7IFf3rvsE17/TldiU3F/zxFHeqUB5o= | ||||
| golang.org/x/xerrors v0.0.0-20231012003039-104605ab7028 h1:+cNy6SZtPcJQH3LJVLOSmiC7MMxXNOb3PU/VUEz+EhU= | ||||
| golang.org/x/xerrors v0.0.0-20231012003039-104605ab7028/go.mod h1:NDW/Ps6MPRej6fsCIbMTohpP40sJ/P/vI1MoTEGwX90= | ||||
| google.golang.org/api v0.170.0 h1:zMaruDePM88zxZBG+NG8+reALO2rfLhe/JShitLyT48= | ||||
| google.golang.org/api v0.170.0/go.mod h1:/xql9M2btF85xac/VAm4PsLMTLVGUOpq4BE9R8jyNy8= | ||||
| google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= | ||||
| google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= | ||||
| google.golang.org/appengine v1.6.8 h1:IhEN5q69dyKagZPYMSdIjS2HqprW324FRQZJcGqPAsM= | ||||
| @ -572,8 +565,8 @@ google.golang.org/appengine/v2 v2.0.2/go.mod h1:PkgRUWz4o1XOvbqtWTkBtCitEJ5Tp4Ho | ||||
| google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= | ||||
| google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= | ||||
| google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo= | ||||
| google.golang.org/genproto v0.0.0-20240125205218-1f4bbc51befe h1:USL2DhxfgRchafRvt/wYyyQNzwgL7ZiURcozOE/Pkvo= | ||||
| google.golang.org/genproto v0.0.0-20240125205218-1f4bbc51befe/go.mod h1:cc8bqMqtv9gMOr0zHg2Vzff5ULhhL2IXP4sbcn32Dro= | ||||
| google.golang.org/genproto v0.0.0-20240213162025-012b6fc9bca9 h1:9+tzLLstTlPTRyJTh+ah5wIMsBW5c4tQwGTN3thOW9Y= | ||||
| google.golang.org/genproto v0.0.0-20240213162025-012b6fc9bca9/go.mod h1:mqHbVIp48Muh7Ywss/AD6I5kNVKZMmAa/QEW58Gxp2s= | ||||
| google.golang.org/genproto/googleapis/api v0.0.0-20240604185151-ef581f913117 h1:+rdxYoE3E5htTEWIe15GlN6IfvbURM//Jt0mmkmm6ZU= | ||||
| google.golang.org/genproto/googleapis/api v0.0.0-20240604185151-ef581f913117/go.mod h1:OimBR/bc1wPO9iV4NC2bpyjy3VnAwZh5EBPQdtaE5oo= | ||||
| google.golang.org/genproto/googleapis/rpc v0.0.0-20240903143218-8af14fe29dc1 h1:pPJltXNxVzT4pK9yD8vR9X75DaWYYmLGMsEvBfFQZzQ= | ||||
| @ -594,8 +587,6 @@ google.golang.org/protobuf v1.22.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2 | ||||
| google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= | ||||
| google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= | ||||
| google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c= | ||||
| google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= | ||||
| google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= | ||||
| google.golang.org/protobuf v1.34.2 h1:6xV6lTsCfpGD21XK49h7MhtcApnLqkfYgPcdHftf6hg= | ||||
| google.golang.org/protobuf v1.34.2/go.mod h1:qYOHts0dSfpeUzUFpOMr/WGzszTmLH+DiWniOlNbLDw= | ||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
|  | ||||
| @ -62,3 +62,7 @@ func (o *ConversationApi) GetIncrementalConversation(c *gin.Context) { | ||||
| func (o *ConversationApi) GetOwnerConversation(c *gin.Context) { | ||||
| 	a2r.Call(conversation.ConversationClient.GetOwnerConversation, o.Client, c) | ||||
| } | ||||
| 
 | ||||
| func (o *ConversationApi) GetNotNotifyConversationIDs(c *gin.Context) { | ||||
| 	a2r.Call(conversation.ConversationClient.GetNotNotifyConversationIDs, o.Client, c) | ||||
| } | ||||
|  | ||||
| @ -72,6 +72,10 @@ func (o *FriendApi) GetPaginationBlacks(c *gin.Context) { | ||||
| 	a2r.Call(relation.FriendClient.GetPaginationBlacks, o.Client, c) | ||||
| } | ||||
| 
 | ||||
| func (o *FriendApi) GetSpecifiedBlacks(c *gin.Context) { | ||||
| 	a2r.Call(relation.FriendClient.GetSpecifiedBlacks, o.Client, c) | ||||
| } | ||||
| 
 | ||||
| func (o *FriendApi) RemoveBlack(c *gin.Context) { | ||||
| 	a2r.Call(relation.FriendClient.RemoveBlack, o.Client, c) | ||||
| } | ||||
|  | ||||
| @ -35,8 +35,8 @@ func (o *GroupApi) SetGroupInfo(c *gin.Context) { | ||||
| 	a2r.Call(group.GroupClient.SetGroupInfo, o.Client, c) | ||||
| } | ||||
| 
 | ||||
| func (o *GroupApi) SetGroupInfoEX(c *gin.Context) { | ||||
| 	a2r.Call(group.GroupClient.SetGroupInfoEX, o.Client, c) | ||||
| func (o *GroupApi) SetGroupInfoEx(c *gin.Context) { | ||||
| 	a2r.Call(group.GroupClient.SetGroupInfoEx, o.Client, c) | ||||
| } | ||||
| 
 | ||||
| func (o *GroupApi) JoinGroup(c *gin.Context) { | ||||
|  | ||||
| @ -2,6 +2,7 @@ package api | ||||
| 
 | ||||
| import ( | ||||
| 	"fmt" | ||||
| 
 | ||||
| 	"github.com/gin-contrib/gzip" | ||||
| 
 | ||||
| 	"github.com/gin-gonic/gin" | ||||
| @ -115,6 +116,7 @@ func newGinRouter(disCov discovery.SvcDiscoveryRegistry, config *Config) *gin.En | ||||
| 		friendRouterGroup.POST("/set_friend_remark", f.SetFriendRemark) | ||||
| 		friendRouterGroup.POST("/add_black", f.AddBlack) | ||||
| 		friendRouterGroup.POST("/get_black_list", f.GetPaginationBlacks) | ||||
| 		friendRouterGroup.POST("/get_specified_blacks", f.GetSpecifiedBlacks) | ||||
| 		friendRouterGroup.POST("/remove_black", f.RemoveBlack) | ||||
| 		friendRouterGroup.POST("/get_incremental_blacks", f.GetIncrementalBlacks) | ||||
| 		friendRouterGroup.POST("/import_friend", f.ImportFriends) | ||||
| @ -130,7 +132,7 @@ func newGinRouter(disCov discovery.SvcDiscoveryRegistry, config *Config) *gin.En | ||||
| 	{ | ||||
| 		groupRouterGroup.POST("/create_group", g.CreateGroup) | ||||
| 		groupRouterGroup.POST("/set_group_info", g.SetGroupInfo) | ||||
| 		groupRouterGroup.POST("/set_group_info_ex", g.SetGroupInfoEX) | ||||
| 		groupRouterGroup.POST("/set_group_info_ex", g.SetGroupInfoEx) | ||||
| 		groupRouterGroup.POST("/join_group", g.JoinGroup) | ||||
| 		groupRouterGroup.POST("/quit_group", g.QuitGroup) | ||||
| 		groupRouterGroup.POST("/group_application_response", g.ApplicationGroupResponse) | ||||
| @ -230,6 +232,7 @@ func newGinRouter(disCov discovery.SvcDiscoveryRegistry, config *Config) *gin.En | ||||
| 		conversationGroup.POST("/get_full_conversation_ids", c.GetFullOwnerConversationIDs) | ||||
| 		conversationGroup.POST("/get_incremental_conversations", c.GetIncrementalConversation) | ||||
| 		conversationGroup.POST("/get_owner_conversation", c.GetOwnerConversation) | ||||
| 		conversationGroup.POST("/get_not_notify_conversation_ids", c.GetNotNotifyConversationIDs) | ||||
| 	} | ||||
| 
 | ||||
| 	statisticsGroup := r.Group("/statistics") | ||||
| @ -275,7 +278,6 @@ func GinParseToken(authRPC *rpcclient.Auth) gin.HandlerFunc { | ||||
| 
 | ||||
| // Whitelist api not parse token | ||||
| var Whitelist = []string{ | ||||
| 	"/user/user_register", | ||||
| 	"/auth/user_token", | ||||
| 	"/auth/parse_token", | ||||
| } | ||||
|  | ||||
| @ -321,7 +321,7 @@ func (ws *WsServer) KickUserConn(client *Client) error { | ||||
| } | ||||
| 
 | ||||
| func (ws *WsServer) multiTerminalLoginChecker(clientOK bool, oldClients []*Client, newClient *Client) { | ||||
| 	switch ws.msgGatewayConfig.MsgGateway.MultiLoginPolicy { | ||||
| 	switch ws.msgGatewayConfig.Share.MultiLoginPolicy { | ||||
| 	case constant.DefalutNotKick: | ||||
| 	case constant.PCAndOther: | ||||
| 		if constant.PlatformIDToClass(newClient.PlatformID) == constant.TerminalPC { | ||||
|  | ||||
| @ -154,7 +154,7 @@ func (f *Fcm) Push(ctx context.Context, userIDs []string, title, content string, | ||||
| 	} | ||||
| 	messageCount := len(messages) | ||||
| 	if messageCount > 0 { | ||||
| 		response, err := f.fcmMsgCli.SendAll(ctx, messages) | ||||
| 		response, err := f.fcmMsgCli.SendEach(ctx, messages) | ||||
| 		if err != nil { | ||||
| 			Fail = Fail + messageCount | ||||
| 		} else { | ||||
|  | ||||
| @ -64,6 +64,7 @@ func Start(ctx context.Context, config *Config, client discovery.SvcDiscoveryReg | ||||
| 			redis2.NewTokenCacheModel(rdb, config.RpcConfig.TokenPolicy.Expire), | ||||
| 			config.Share.Secret, | ||||
| 			config.RpcConfig.TokenPolicy.Expire, | ||||
| 			config.Share.MultiLoginPolicy, | ||||
| 		), | ||||
| 		config: config, | ||||
| 	}) | ||||
|  | ||||
| @ -278,8 +278,8 @@ func (c *conversationServer) SetConversations(ctx context.Context, req *pbconver | ||||
| 		if req.Conversation.MsgDestructTime != nil { | ||||
| 			m["msg_destruct_time"] = req.Conversation.MsgDestructTime.Value | ||||
| 		} | ||||
| 		if req.Conversation.MsgDestructTime != nil { | ||||
| 			m["msg_destruct_time"] = req.Conversation.MsgDestructTime.Value | ||||
| 		if req.Conversation.IsMsgDestruct != nil { | ||||
| 			m["is_msg_destruct"] = req.Conversation.IsMsgDestruct.Value | ||||
| 		} | ||||
| 		if req.Conversation.BurnDuration != nil { | ||||
| 			m["burn_duration"] = req.Conversation.BurnDuration.Value | ||||
| @ -710,3 +710,11 @@ func (c *conversationServer) GetConversationsNeedDestructMsgs(ctx context.Contex | ||||
| 
 | ||||
| 	return &pbconversation.GetConversationsNeedDestructMsgsResp{Conversations: convert.ConversationsDB2Pb(temp)}, nil | ||||
| } | ||||
| 
 | ||||
| func (c *conversationServer) GetNotNotifyConversationIDs(ctx context.Context, req *pbconversation.GetNotNotifyConversationIDsReq) (*pbconversation.GetNotNotifyConversationIDsResp, error) { | ||||
| 	conversationIDs, err := c.conversationDatabase.GetNotNotifyConversationIDs(ctx, req.UserID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &pbconversation.GetNotNotifyConversationIDsResp{ConversationIDs: conversationIDs}, nil | ||||
| } | ||||
| @ -218,6 +218,7 @@ func (s *groupServer) webhookAfterKickGroupMember(ctx context.Context, after *co | ||||
| 		CallbackCommand: callbackstruct.CallbackAfterKickGroupCommand, | ||||
| 		GroupID:         req.GroupID, | ||||
| 		KickedUserIDs:   req.KickedUserIDs, | ||||
| 		Reason:          req.Reason, | ||||
| 	} | ||||
| 	s.webhookClient.AsyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, &callbackstruct.CallbackKillGroupMemberResp{}, after) | ||||
| } | ||||
| @ -359,73 +360,73 @@ func (s *groupServer) webhookAfterSetGroupInfo(ctx context.Context, after *confi | ||||
| 	s.webhookClient.AsyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, &callbackstruct.CallbackAfterSetGroupInfoResp{}, after) | ||||
| } | ||||
| 
 | ||||
| func (s *groupServer) webhookBeforeSetGroupInfoEX(ctx context.Context, before *config.BeforeConfig, req *group.SetGroupInfoEXReq) error { | ||||
| func (s *groupServer) webhookBeforeSetGroupInfoEx(ctx context.Context, before *config.BeforeConfig, req *group.SetGroupInfoExReq) error { | ||||
| 	return webhook.WithCondition(ctx, before, func(ctx context.Context) error { | ||||
| 		cbReq := &callbackstruct.CallbackBeforeSetGroupInfoEXReq{ | ||||
| 			CallbackCommand: callbackstruct.CallbackBeforeSetGroupInfoCommand, | ||||
| 			GroupID:         req.GroupInfoForSet.GroupID, | ||||
| 			GroupName:       req.GroupInfoForSet.GroupName, | ||||
| 			Notification:    req.GroupInfoForSet.Notification, | ||||
| 			Introduction:    req.GroupInfoForSet.Introduction, | ||||
| 			FaceURL:         req.GroupInfoForSet.FaceURL, | ||||
| 		cbReq := &callbackstruct.CallbackBeforeSetGroupInfoExReq{ | ||||
| 			CallbackCommand: callbackstruct.CallbackBeforeSetGroupInfoExCommand, | ||||
| 			GroupID:         req.GroupID, | ||||
| 			GroupName:       req.GroupName, | ||||
| 			Notification:    req.Notification, | ||||
| 			Introduction:    req.Introduction, | ||||
| 			FaceURL:         req.FaceURL, | ||||
| 		} | ||||
| 
 | ||||
| 		if req.GroupInfoForSet.Ex != nil { | ||||
| 			cbReq.Ex = req.GroupInfoForSet.Ex | ||||
| 		if req.Ex != nil { | ||||
| 			cbReq.Ex = req.Ex | ||||
| 		} | ||||
| 		log.ZDebug(ctx, "debug CallbackBeforeSetGroupInfoEX", "ex", cbReq.Ex) | ||||
| 
 | ||||
| 		if req.GroupInfoForSet.NeedVerification != nil { | ||||
| 			cbReq.NeedVerification = req.GroupInfoForSet.NeedVerification | ||||
| 		if req.NeedVerification != nil { | ||||
| 			cbReq.NeedVerification = req.NeedVerification | ||||
| 		} | ||||
| 		if req.GroupInfoForSet.LookMemberInfo != nil { | ||||
| 			cbReq.LookMemberInfo = req.GroupInfoForSet.LookMemberInfo | ||||
| 		if req.LookMemberInfo != nil { | ||||
| 			cbReq.LookMemberInfo = req.LookMemberInfo | ||||
| 		} | ||||
| 		if req.GroupInfoForSet.ApplyMemberFriend != nil { | ||||
| 			cbReq.ApplyMemberFriend = req.GroupInfoForSet.ApplyMemberFriend | ||||
| 		if req.ApplyMemberFriend != nil { | ||||
| 			cbReq.ApplyMemberFriend = req.ApplyMemberFriend | ||||
| 		} | ||||
| 
 | ||||
| 		resp := &callbackstruct.CallbackBeforeSetGroupInfoEXResp{} | ||||
| 		resp := &callbackstruct.CallbackBeforeSetGroupInfoExResp{} | ||||
| 
 | ||||
| 		if err := s.webhookClient.SyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, resp, before); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 
 | ||||
| 		datautil.NotNilReplace(&req.GroupInfoForSet.GroupID, &resp.GroupID) | ||||
| 		datautil.NotNilReplace(&req.GroupInfoForSet.GroupName, &resp.GroupName) | ||||
| 		datautil.NotNilReplace(&req.GroupInfoForSet.FaceURL, &resp.FaceURL) | ||||
| 		datautil.NotNilReplace(&req.GroupInfoForSet.Introduction, &resp.Introduction) | ||||
| 		datautil.NotNilReplace(&req.GroupInfoForSet.Ex, &resp.Ex) | ||||
| 		datautil.NotNilReplace(&req.GroupInfoForSet.NeedVerification, &resp.NeedVerification) | ||||
| 		datautil.NotNilReplace(&req.GroupInfoForSet.LookMemberInfo, &resp.LookMemberInfo) | ||||
| 		datautil.NotNilReplace(&req.GroupInfoForSet.ApplyMemberFriend, &resp.ApplyMemberFriend) | ||||
| 		datautil.NotNilReplace(&req.GroupID, &resp.GroupID) | ||||
| 		datautil.NotNilReplace(&req.GroupName, &resp.GroupName) | ||||
| 		datautil.NotNilReplace(&req.FaceURL, &resp.FaceURL) | ||||
| 		datautil.NotNilReplace(&req.Introduction, &resp.Introduction) | ||||
| 		datautil.NotNilReplace(&req.Ex, &resp.Ex) | ||||
| 		datautil.NotNilReplace(&req.NeedVerification, &resp.NeedVerification) | ||||
| 		datautil.NotNilReplace(&req.LookMemberInfo, &resp.LookMemberInfo) | ||||
| 		datautil.NotNilReplace(&req.ApplyMemberFriend, &resp.ApplyMemberFriend) | ||||
| 
 | ||||
| 		return nil | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func (s *groupServer) webhookAfterSetGroupInfoEX(ctx context.Context, after *config.AfterConfig, req *group.SetGroupInfoEXReq) { | ||||
| 	cbReq := &callbackstruct.CallbackAfterSetGroupInfoEXReq{ | ||||
| 		CallbackCommand: callbackstruct.CallbackAfterSetGroupInfoCommand, | ||||
| 		GroupID:         req.GroupInfoForSet.GroupID, | ||||
| 		GroupName:       req.GroupInfoForSet.GroupName, | ||||
| 		Notification:    req.GroupInfoForSet.Notification, | ||||
| 		Introduction:    req.GroupInfoForSet.Introduction, | ||||
| 		FaceURL:         req.GroupInfoForSet.FaceURL, | ||||
| func (s *groupServer) webhookAfterSetGroupInfoEx(ctx context.Context, after *config.AfterConfig, req *group.SetGroupInfoExReq) { | ||||
| 	cbReq := &callbackstruct.CallbackAfterSetGroupInfoExReq{ | ||||
| 		CallbackCommand: callbackstruct.CallbackAfterSetGroupInfoExCommand, | ||||
| 		GroupID:         req.GroupID, | ||||
| 		GroupName:       req.GroupName, | ||||
| 		Notification:    req.Notification, | ||||
| 		Introduction:    req.Introduction, | ||||
| 		FaceURL:         req.FaceURL, | ||||
| 	} | ||||
| 
 | ||||
| 	if req.GroupInfoForSet.Ex != nil { | ||||
| 		cbReq.Ex = req.GroupInfoForSet.Ex | ||||
| 	if req.Ex != nil { | ||||
| 		cbReq.Ex = req.Ex | ||||
| 	} | ||||
| 	if req.GroupInfoForSet.NeedVerification != nil { | ||||
| 		cbReq.NeedVerification = req.GroupInfoForSet.NeedVerification | ||||
| 	if req.NeedVerification != nil { | ||||
| 		cbReq.NeedVerification = req.NeedVerification | ||||
| 	} | ||||
| 	if req.GroupInfoForSet.LookMemberInfo != nil { | ||||
| 		cbReq.LookMemberInfo = req.GroupInfoForSet.LookMemberInfo | ||||
| 	if req.LookMemberInfo != nil { | ||||
| 		cbReq.LookMemberInfo = req.LookMemberInfo | ||||
| 	} | ||||
| 	if req.GroupInfoForSet.ApplyMemberFriend != nil { | ||||
| 		cbReq.ApplyMemberFriend = req.GroupInfoForSet.ApplyMemberFriend | ||||
| 	if req.ApplyMemberFriend != nil { | ||||
| 		cbReq.ApplyMemberFriend = req.ApplyMemberFriend | ||||
| 	} | ||||
| 
 | ||||
| 	s.webhookClient.AsyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, &callbackstruct.CallbackAfterSetGroupInfoEXResp{}, after) | ||||
| 	s.webhookClient.AsyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, &callbackstruct.CallbackAfterSetGroupInfoExResp{}, after) | ||||
| } | ||||
|  | ||||
| @ -20,6 +20,7 @@ import ( | ||||
| 
 | ||||
| 	pbgroup "github.com/openimsdk/protocol/group" | ||||
| 	"github.com/openimsdk/protocol/sdkws" | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| 	"github.com/openimsdk/tools/mcontext" | ||||
| ) | ||||
| 
 | ||||
| @ -54,11 +55,11 @@ func UpdateGroupInfoMap(ctx context.Context, group *sdkws.GroupInfoForSet) map[s | ||||
| 	return m | ||||
| } | ||||
| 
 | ||||
| func UpdateGroupInfoEXMap(ctx context.Context, group *sdkws.GroupInfoForSetEX) map[string]any { | ||||
| func UpdateGroupInfoExMap(ctx context.Context, group *pbgroup.SetGroupInfoExReq) (map[string]any, error) { | ||||
| 	m := make(map[string]any) | ||||
| 
 | ||||
| 	if group.GroupName != "" { | ||||
| 		m["group_name"] = group.GroupName | ||||
| 	if group.GroupName != nil && group.GroupName.Value != "" { | ||||
| 		return nil, errs.ErrArgs.WrapMsg("group name is empty") | ||||
| 	} | ||||
| 	if group.Notification != nil { | ||||
| 		m["notification"] = group.Notification.Value | ||||
| @ -84,7 +85,7 @@ func UpdateGroupInfoEXMap(ctx context.Context, group *sdkws.GroupInfoForSetEX) m | ||||
| 		m["ex"] = group.Ex.Value | ||||
| 	} | ||||
| 
 | ||||
| 	return m | ||||
| 	return m, nil | ||||
| } | ||||
| 
 | ||||
| func UpdateGroupStatusMap(status int) map[string]any { | ||||
|  | ||||
| @ -167,11 +167,11 @@ func (g *groupServer) CheckGroupAdmin(ctx context.Context, groupID string) error | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func (g *groupServer) GetPublicUserInfoMap(ctx context.Context, userIDs []string, complete bool) (map[string]*sdkws.PublicUserInfo, error) { | ||||
| func (g *groupServer) GetPublicUserInfoMap(ctx context.Context, userIDs []string) (map[string]*sdkws.PublicUserInfo, error) { | ||||
| 	if len(userIDs) == 0 { | ||||
| 		return map[string]*sdkws.PublicUserInfo{}, nil | ||||
| 	} | ||||
| 	users, err := g.user.GetPublicUserInfos(ctx, userIDs, complete) | ||||
| 	users, err := g.user.GetPublicUserInfos(ctx, userIDs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| @ -696,7 +696,7 @@ func (g *groupServer) GetGroupApplicationList(ctx context.Context, req *pbgroup. | ||||
| 		userIDs = append(userIDs, gr.UserID) | ||||
| 	} | ||||
| 	userIDs = datautil.Distinct(userIDs) | ||||
| 	userMap, err := g.user.GetPublicUserInfoMap(ctx, userIDs, true) | ||||
| 	userMap, err := g.user.GetPublicUserInfoMap(ctx, userIDs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| @ -1058,13 +1058,13 @@ func (g *groupServer) SetGroupInfo(ctx context.Context, req *pbgroup.SetGroupInf | ||||
| 	return &pbgroup.SetGroupInfoResp{}, nil | ||||
| } | ||||
| 
 | ||||
| func (g *groupServer) SetGroupInfoEX(ctx context.Context, req *pbgroup.SetGroupInfoEXReq) (*pbgroup.SetGroupInfoEXResp, error) { | ||||
| func (g *groupServer) SetGroupInfoEx(ctx context.Context, req *pbgroup.SetGroupInfoExReq) (*pbgroup.SetGroupInfoExResp, error) { | ||||
| 	var opMember *model.GroupMember | ||||
| 
 | ||||
| 	if !authverify.IsAppManagerUid(ctx, g.config.Share.IMAdminUserID) { | ||||
| 		var err error | ||||
| 
 | ||||
| 		opMember, err = g.db.TakeGroupMember(ctx, req.GroupInfoForSet.GroupID, mcontext.GetOpUserID(ctx)) | ||||
| 		opMember, err = g.db.TakeGroupMember(ctx, req.GroupID, mcontext.GetOpUserID(ctx)) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| @ -1078,11 +1078,11 @@ func (g *groupServer) SetGroupInfoEX(ctx context.Context, req *pbgroup.SetGroupI | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	if err := g.webhookBeforeSetGroupInfoEX(ctx, &g.config.WebhooksConfig.BeforeSetGroupInfoEX, req); err != nil && err != servererrs.ErrCallbackContinue { | ||||
| 	if err := g.webhookBeforeSetGroupInfoEx(ctx, &g.config.WebhooksConfig.BeforeSetGroupInfoEx, req); err != nil && err != servererrs.ErrCallbackContinue { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	group, err := g.db.TakeGroup(ctx, req.GroupInfoForSet.GroupID) | ||||
| 	group, err := g.db.TakeGroup(ctx, req.GroupID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| @ -1104,16 +1104,20 @@ func (g *groupServer) SetGroupInfoEX(ctx context.Context, req *pbgroup.SetGroupI | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	updatedData := UpdateGroupInfoEXMap(ctx, req.GroupInfoForSet) | ||||
| 	updatedData, err := UpdateGroupInfoExMap(ctx, req) | ||||
| 	if len(updatedData) == 0 { | ||||
| 		return &pbgroup.SetGroupInfoEXResp{}, nil | ||||
| 		return &pbgroup.SetGroupInfoExResp{}, nil | ||||
| 	} | ||||
| 
 | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	if err := g.db.UpdateGroup(ctx, group.GroupID, updatedData); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	group, err = g.db.TakeGroup(ctx, req.GroupInfoForSet.GroupID) | ||||
| 	group, err = g.db.TakeGroup(ctx, req.GroupID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| @ -1129,18 +1133,18 @@ func (g *groupServer) SetGroupInfoEX(ctx context.Context, req *pbgroup.SetGroupI | ||||
| 	} | ||||
| 
 | ||||
| 	num := len(updatedData) | ||||
| 	if req.GroupInfoForSet.Notification != nil { | ||||
| 	if req.Notification != nil { | ||||
| 		num-- | ||||
| 
 | ||||
| 		if req.GroupInfoForSet.Notification.Value != "" { | ||||
| 		if req.Notification.Value != "" { | ||||
| 			func() { | ||||
| 				conversation := &pbconversation.ConversationReq{ | ||||
| 					ConversationID:   msgprocessor.GetConversationIDBySessionType(constant.ReadGroupChatType, req.GroupInfoForSet.GroupID), | ||||
| 					ConversationID:   msgprocessor.GetConversationIDBySessionType(constant.ReadGroupChatType, req.GroupID), | ||||
| 					ConversationType: constant.ReadGroupChatType, | ||||
| 					GroupID:          req.GroupInfoForSet.GroupID, | ||||
| 					GroupID:          req.GroupID, | ||||
| 				} | ||||
| 
 | ||||
| 				resp, err := g.GetGroupMemberUserIDs(ctx, &pbgroup.GetGroupMemberUserIDsReq{GroupID: req.GroupInfoForSet.GroupID}) | ||||
| 				resp, err := g.GetGroupMemberUserIDs(ctx, &pbgroup.GetGroupMemberUserIDsReq{GroupID: req.GroupID}) | ||||
| 				if err != nil { | ||||
| 					log.ZWarn(ctx, "GetGroupMemberIDs is failed.", err) | ||||
| 					return | ||||
| @ -1157,7 +1161,7 @@ func (g *groupServer) SetGroupInfoEX(ctx context.Context, req *pbgroup.SetGroupI | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	if req.GroupInfoForSet.GroupName != "" { | ||||
| 	if req.GroupName != nil { | ||||
| 		num-- | ||||
| 		g.notification.GroupInfoSetNameNotification(ctx, &sdkws.GroupInfoSetNameTips{Group: tips.Group, OpUser: tips.OpUser}) | ||||
| 	} | ||||
| @ -1166,9 +1170,9 @@ func (g *groupServer) SetGroupInfoEX(ctx context.Context, req *pbgroup.SetGroupI | ||||
| 		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) { | ||||
|  | ||||
| @ -23,13 +23,17 @@ import ( | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/convert" | ||||
| 	"github.com/openimsdk/protocol/relation" | ||||
| 	"github.com/openimsdk/protocol/sdkws" | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| 	"github.com/openimsdk/tools/mcontext" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| ) | ||||
| 
 | ||||
| func (s *friendServer) GetPaginationBlacks(ctx context.Context, req *relation.GetPaginationBlacksReq) (resp *relation.GetPaginationBlacksResp, err error) { | ||||
| 	if err := s.userRpcClient.Access(ctx, req.UserID); err != nil { | ||||
| 	if err := authverify.CheckAccessV3(ctx, req.UserID, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	total, blacks, err := s.blackDatabase.FindOwnerBlacks(ctx, req.UserID, req.Pagination) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -55,7 +59,7 @@ func (s *friendServer) IsBlack(ctx context.Context, req *relation.IsBlackReq) (* | ||||
| } | ||||
| 
 | ||||
| func (s *friendServer) RemoveBlack(ctx context.Context, req *relation.RemoveBlackReq) (*relation.RemoveBlackResp, error) { | ||||
| 	if err := s.userRpcClient.Access(ctx, req.OwnerUserID); err != nil { | ||||
| 	if err := authverify.CheckAccessV3(ctx, req.OwnerUserID, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| @ -64,6 +68,7 @@ func (s *friendServer) RemoveBlack(ctx context.Context, req *relation.RemoveBlac | ||||
| 	} | ||||
| 
 | ||||
| 	s.notificationSender.BlackDeletedNotification(ctx, req) | ||||
| 	s.webhookAfterRemoveBlack(ctx, &s.config.WebhooksConfig.AfterRemoveBlack, req) | ||||
| 
 | ||||
| 	return &relation.RemoveBlackResp{}, nil | ||||
| } | ||||
| @ -72,6 +77,11 @@ func (s *friendServer) AddBlack(ctx context.Context, req *relation.AddBlackReq) | ||||
| 	if err := authverify.CheckAccessV3(ctx, req.OwnerUserID, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	if err := s.webhookBeforeAddBlack(ctx, &s.config.WebhooksConfig.BeforeAddBlack, req); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	_, err := s.userRpcClient.GetUsersInfo(ctx, []string{req.OwnerUserID, req.BlackUserID}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @ -90,3 +100,53 @@ func (s *friendServer) AddBlack(ctx context.Context, req *relation.AddBlackReq) | ||||
| 	s.notificationSender.BlackAddedNotification(ctx, req) | ||||
| 	return &relation.AddBlackResp{}, nil | ||||
| } | ||||
| 
 | ||||
| func (s *friendServer) GetSpecifiedBlacks(ctx context.Context, req *relation.GetSpecifiedBlacksReq) (*relation.GetSpecifiedBlacksResp, error) { | ||||
| 	if err := authverify.CheckAccessV3(ctx, req.OwnerUserID, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	if len(req.UserIDList) == 0 { | ||||
| 		return nil, errs.ErrArgs.WrapMsg("userIDList is empty") | ||||
| 	} | ||||
| 
 | ||||
| 	if datautil.Duplicate(req.UserIDList) { | ||||
| 		return nil, errs.ErrArgs.WrapMsg("userIDList repeated") | ||||
| 	} | ||||
| 
 | ||||
| 	userMap, err := s.userRpcClient.GetPublicUserInfoMap(ctx, req.UserIDList) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	blacks, err := s.blackDatabase.FindBlackInfos(ctx, req.OwnerUserID, req.UserIDList) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	blackMap := datautil.SliceToMap(blacks, func(e *model.Black) string { | ||||
| 		return e.BlockUserID | ||||
| 	}) | ||||
| 
 | ||||
| 	resp := &relation.GetSpecifiedBlacksResp{ | ||||
| 		Blacks: make([]*sdkws.BlackInfo, 0, len(req.UserIDList)), | ||||
| 	} | ||||
| 
 | ||||
| 	for _, userID := range req.UserIDList { | ||||
| 		if black := blackMap[userID]; black != nil { | ||||
| 			resp.Blacks = append(resp.Blacks, | ||||
| 				&sdkws.BlackInfo{ | ||||
| 					OwnerUserID:    black.OwnerUserID, | ||||
| 					CreateTime:     black.CreateTime.UnixMilli(), | ||||
| 					BlackUserInfo:  userMap[userID], | ||||
| 					AddSource:      black.AddSource, | ||||
| 					OperatorUserID: black.OperatorUserID, | ||||
| 					Ex:             black.Ex, | ||||
| 				}) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	resp.Total = int32(len(resp.Blacks)) | ||||
| 
 | ||||
| 	return resp, nil | ||||
| } | ||||
|  | ||||
| @ -228,20 +228,23 @@ func (s *friendServer) RespondFriendApply(ctx context.Context, req *relation.Res | ||||
| 
 | ||||
| // ok. | ||||
| func (s *friendServer) DeleteFriend(ctx context.Context, req *relation.DeleteFriendReq) (resp *relation.DeleteFriendResp, err error) { | ||||
| 	resp = &relation.DeleteFriendResp{} | ||||
| 	if err := s.userRpcClient.Access(ctx, req.OwnerUserID); err != nil { | ||||
| 	if err := authverify.CheckAccessV3(ctx, req.OwnerUserID, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	_, err = s.db.FindFriendsWithError(ctx, req.OwnerUserID, []string{req.FriendUserID}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	if err := s.db.Delete(ctx, req.OwnerUserID, []string{req.FriendUserID}); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	s.notificationSender.FriendDeletedNotification(ctx, req) | ||||
| 	s.webhookAfterDeleteFriend(ctx, &s.config.WebhooksConfig.AfterDeleteFriend, req) | ||||
| 	return resp, nil | ||||
| 
 | ||||
| 	return &relation.DeleteFriendResp{}, nil | ||||
| } | ||||
| 
 | ||||
| // ok. | ||||
| @ -249,20 +252,24 @@ func (s *friendServer) SetFriendRemark(ctx context.Context, req *relation.SetFri | ||||
| 	if err = s.webhookBeforeSetFriendRemark(ctx, &s.config.WebhooksConfig.BeforeSetFriendRemark, req); err != nil && err != servererrs.ErrCallbackContinue { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	resp = &relation.SetFriendRemarkResp{} | ||||
| 	if err := s.userRpcClient.Access(ctx, req.OwnerUserID); err != nil { | ||||
| 
 | ||||
| 	if err := authverify.CheckAccessV3(ctx, req.OwnerUserID, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	_, err = s.db.FindFriendsWithError(ctx, req.OwnerUserID, []string{req.FriendUserID}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	if err := s.db.UpdateRemark(ctx, req.OwnerUserID, req.FriendUserID, req.Remark); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	s.webhookAfterSetFriendRemark(ctx, &s.config.WebhooksConfig.AfterSetFriendRemark, req) | ||||
| 	s.notificationSender.FriendRemarkSetNotification(ctx, req.OwnerUserID, req.FriendUserID) | ||||
| 	return resp, nil | ||||
| 
 | ||||
| 	return &relation.SetFriendRemarkResp{}, nil | ||||
| } | ||||
| 
 | ||||
| // ok. | ||||
| @ -309,7 +316,7 @@ func (s *friendServer) GetDesignatedFriendsApply(ctx context.Context, | ||||
| 
 | ||||
| // Get received friend requests (i.e., those initiated by others). | ||||
| func (s *friendServer) GetPaginationFriendsApplyTo(ctx context.Context, req *relation.GetPaginationFriendsApplyToReq) (resp *relation.GetPaginationFriendsApplyToResp, err error) { | ||||
| 	if err := s.userRpcClient.Access(ctx, req.UserID); err != nil { | ||||
| 	if err := authverify.CheckAccessV3(ctx, req.UserID, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| @ -331,18 +338,23 @@ func (s *friendServer) GetPaginationFriendsApplyTo(ctx context.Context, req *rel | ||||
| 
 | ||||
| func (s *friendServer) GetPaginationFriendsApplyFrom(ctx context.Context, req *relation.GetPaginationFriendsApplyFromReq) (resp *relation.GetPaginationFriendsApplyFromResp, err error) { | ||||
| 	resp = &relation.GetPaginationFriendsApplyFromResp{} | ||||
| 	if err := s.userRpcClient.Access(ctx, req.UserID); err != nil { | ||||
| 
 | ||||
| 	if err := authverify.CheckAccessV3(ctx, req.UserID, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	total, friendRequests, err := s.db.PageFriendRequestFromMe(ctx, req.UserID, req.Pagination) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	resp.FriendRequests, err = convert.FriendRequestDB2Pb(ctx, friendRequests, s.userRpcClient.GetUsersInfoMap) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	resp.Total = int32(total) | ||||
| 
 | ||||
| 	return resp, nil | ||||
| } | ||||
| 
 | ||||
| @ -357,31 +369,37 @@ func (s *friendServer) IsFriend(ctx context.Context, req *relation.IsFriendReq) | ||||
| } | ||||
| 
 | ||||
| func (s *friendServer) GetPaginationFriends(ctx context.Context, req *relation.GetPaginationFriendsReq) (resp *relation.GetPaginationFriendsResp, err error) { | ||||
| 	if err := s.userRpcClient.Access(ctx, req.UserID); err != nil { | ||||
| 	if err := authverify.CheckAccessV3(ctx, req.UserID, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	total, friends, err := s.db.PageOwnerFriends(ctx, req.UserID, req.Pagination) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	resp = &relation.GetPaginationFriendsResp{} | ||||
| 	resp.FriendsInfo, err = convert.FriendsDB2Pb(ctx, friends, s.userRpcClient.GetUsersInfoMap) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	resp.Total = int32(total) | ||||
| 
 | ||||
| 	return resp, nil | ||||
| } | ||||
| 
 | ||||
| func (s *friendServer) GetFriendIDs(ctx context.Context, req *relation.GetFriendIDsReq) (resp *relation.GetFriendIDsResp, err error) { | ||||
| 	if err := s.userRpcClient.Access(ctx, req.UserID); err != nil { | ||||
| 	if err := authverify.CheckAccessV3(ctx, req.UserID, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	resp = &relation.GetFriendIDsResp{} | ||||
| 	resp.FriendIDs, err = s.db.FindFriendUserIDs(ctx, req.UserID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	return resp, nil | ||||
| } | ||||
| 
 | ||||
| @ -389,35 +407,45 @@ func (s *friendServer) GetSpecifiedFriendsInfo(ctx context.Context, req *relatio | ||||
| 	if len(req.UserIDList) == 0 { | ||||
| 		return nil, errs.ErrArgs.WrapMsg("userIDList is empty") | ||||
| 	} | ||||
| 
 | ||||
| 	if datautil.Duplicate(req.UserIDList) { | ||||
| 		return nil, errs.ErrArgs.WrapMsg("userIDList repeated") | ||||
| 	} | ||||
| 
 | ||||
| 	userMap, err := s.userRpcClient.GetUsersInfoMap(ctx, req.UserIDList) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	friends, err := s.db.FindFriendsWithError(ctx, req.OwnerUserID, req.UserIDList) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	blacks, err := s.blackDatabase.FindBlackInfos(ctx, req.OwnerUserID, req.UserIDList) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	friendMap := datautil.SliceToMap(friends, func(e *model.Friend) string { | ||||
| 		return e.FriendUserID | ||||
| 	}) | ||||
| 
 | ||||
| 	blackMap := datautil.SliceToMap(blacks, func(e *model.Black) string { | ||||
| 		return e.BlockUserID | ||||
| 	}) | ||||
| 
 | ||||
| 	resp := &relation.GetSpecifiedFriendsInfoResp{ | ||||
| 		Infos: make([]*relation.GetSpecifiedFriendsInfoInfo, 0, len(req.UserIDList)), | ||||
| 	} | ||||
| 
 | ||||
| 	for _, userID := range req.UserIDList { | ||||
| 		user := userMap[userID] | ||||
| 
 | ||||
| 		if user == nil { | ||||
| 			continue | ||||
| 		} | ||||
| 
 | ||||
| 		var friendInfo *sdkws.FriendInfo | ||||
| 		if friend := friendMap[userID]; friend != nil { | ||||
| 			friendInfo = &sdkws.FriendInfo{ | ||||
| @ -430,6 +458,7 @@ func (s *friendServer) GetSpecifiedFriendsInfo(ctx context.Context, req *relatio | ||||
| 				IsPinned:       friend.IsPinned, | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		var blackInfo *sdkws.BlackInfo | ||||
| 		if black := blackMap[userID]; black != nil { | ||||
| 			blackInfo = &sdkws.BlackInfo{ | ||||
| @ -440,12 +469,14 @@ func (s *friendServer) GetSpecifiedFriendsInfo(ctx context.Context, req *relatio | ||||
| 				Ex:             black.Ex, | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		resp.Infos = append(resp.Infos, &relation.GetSpecifiedFriendsInfoInfo{ | ||||
| 			UserInfo:   user, | ||||
| 			FriendInfo: friendInfo, | ||||
| 			BlackInfo:  blackInfo, | ||||
| 		}) | ||||
| 	} | ||||
| 
 | ||||
| 	return resp, nil | ||||
| } | ||||
| 
 | ||||
|  | ||||
| @ -16,6 +16,7 @@ package user | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 
 | ||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/webhook" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| 
 | ||||
| @ -88,7 +89,6 @@ func (s *userServer) webhookBeforeUserRegister(ctx context.Context, before *conf | ||||
| 	return webhook.WithCondition(ctx, before, func(ctx context.Context) error { | ||||
| 		cbReq := &cbapi.CallbackBeforeUserRegisterReq{ | ||||
| 			CallbackCommand: cbapi.CallbackBeforeUserRegisterCommand, | ||||
| 			Secret:          req.Secret, | ||||
| 			Users:           req.Users, | ||||
| 		} | ||||
| 
 | ||||
| @ -108,7 +108,6 @@ func (s *userServer) webhookBeforeUserRegister(ctx context.Context, before *conf | ||||
| func (s *userServer) webhookAfterUserRegister(ctx context.Context, after *config.AfterConfig, req *pbuser.UserRegisterReq) { | ||||
| 	cbReq := &cbapi.CallbackAfterUserRegisterReq{ | ||||
| 		CallbackCommand: cbapi.CallbackAfterUserRegisterCommand, | ||||
| 		Secret:          req.Secret, | ||||
| 		Users:           req.Users, | ||||
| 	} | ||||
| 
 | ||||
|  | ||||
| @ -47,7 +47,6 @@ import ( | ||||
| 	"github.com/openimsdk/tools/db/pagination" | ||||
| 	registry "github.com/openimsdk/tools/discovery" | ||||
| 	"github.com/openimsdk/tools/errs" | ||||
| 	"github.com/openimsdk/tools/log" | ||||
| 	"github.com/openimsdk/tools/utils/datautil" | ||||
| 	"google.golang.org/grpc" | ||||
| ) | ||||
| @ -263,10 +262,11 @@ func (s *userServer) UserRegister(ctx context.Context, req *pbuser.UserRegisterR | ||||
| 	if len(req.Users) == 0 { | ||||
| 		return nil, errs.ErrArgs.WrapMsg("users is empty") | ||||
| 	} | ||||
| 	if req.Secret != s.config.Share.Secret { | ||||
| 		log.ZDebug(ctx, "UserRegister", s.config.Share.Secret, req.Secret) | ||||
| 		return nil, errs.ErrNoPermission.WrapMsg("secret invalid") | ||||
| 
 | ||||
| 	if err = authverify.CheckAdmin(ctx, s.config.Share.IMAdminUserID); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	if datautil.DuplicateAny(req.Users, func(e *sdkws.UserInfo) string { return e.UserID }) { | ||||
| 		return nil, errs.ErrArgs.WrapMsg("userID repeated") | ||||
| 	} | ||||
|  | ||||
| @ -18,9 +18,9 @@ const ( | ||||
| 	CallbackBeforeInviteJoinGroupCommand    = "callbackBeforeInviteJoinGroupCommand" | ||||
| 	CallbackAfterJoinGroupCommand           = "callbackAfterJoinGroupCommand" | ||||
| 	CallbackAfterSetGroupInfoCommand        = "callbackAfterSetGroupInfoCommand" | ||||
| 	CallbackAfterSetGroupInfoEXCommand      = "callbackAfterSetGroupInfoEXCommand" | ||||
| 	CallbackAfterSetGroupInfoExCommand      = "callbackAfterSetGroupInfoExCommand" | ||||
| 	CallbackBeforeSetGroupInfoCommand       = "callbackBeforeSetGroupInfoCommand" | ||||
| 	CallbackBeforeSetGroupInfoEXCommand     = "callbackBeforeSetGroupInfoEXCommand" | ||||
| 	CallbackBeforeSetGroupInfoExCommand     = "callbackBeforeSetGroupInfoExCommand" | ||||
| 	CallbackAfterRevokeMsgCommand           = "callbackBeforeAfterMsgCommand" | ||||
| 	CallbackBeforeAddBlackCommand           = "callbackBeforeAddBlackCommand" | ||||
| 	CallbackAfterAddFriendCommand           = "callbackAfterAddFriendCommand" | ||||
|  | ||||
| @ -244,11 +244,11 @@ type CallbackAfterSetGroupInfoResp struct { | ||||
| 	CommonCallbackResp | ||||
| } | ||||
| 
 | ||||
| type CallbackBeforeSetGroupInfoEXReq struct { | ||||
| type CallbackBeforeSetGroupInfoExReq struct { | ||||
| 	CallbackCommand   `json:"callbackCommand"` | ||||
| 	OperationID       string                  `json:"operationID"` | ||||
| 	GroupID           string                  `json:"groupID"` | ||||
| 	GroupName         string                  `json:"groupName"` | ||||
| 	GroupName         *wrapperspb.StringValue `json:"groupName"` | ||||
| 	Notification      *wrapperspb.StringValue `json:"notification"` | ||||
| 	Introduction      *wrapperspb.StringValue `json:"introduction"` | ||||
| 	FaceURL           *wrapperspb.StringValue `json:"faceURL"` | ||||
| @ -258,10 +258,10 @@ type CallbackBeforeSetGroupInfoEXReq struct { | ||||
| 	ApplyMemberFriend *wrapperspb.Int32Value  `json:"applyMemberFriend"` | ||||
| } | ||||
| 
 | ||||
| type CallbackBeforeSetGroupInfoEXResp struct { | ||||
| type CallbackBeforeSetGroupInfoExResp struct { | ||||
| 	CommonCallbackResp | ||||
| 	GroupID           string                  `json:"groupID"` | ||||
| 	GroupName         string                  `json:"groupName"` | ||||
| 	GroupName         *wrapperspb.StringValue `json:"groupName"` | ||||
| 	Notification      *wrapperspb.StringValue `json:"notification"` | ||||
| 	Introduction      *wrapperspb.StringValue `json:"introduction"` | ||||
| 	FaceURL           *wrapperspb.StringValue `json:"faceURL"` | ||||
| @ -271,11 +271,11 @@ type CallbackBeforeSetGroupInfoEXResp struct { | ||||
| 	ApplyMemberFriend *wrapperspb.Int32Value  `json:"applyMemberFriend"` | ||||
| } | ||||
| 
 | ||||
| type CallbackAfterSetGroupInfoEXReq struct { | ||||
| type CallbackAfterSetGroupInfoExReq struct { | ||||
| 	CallbackCommand   `json:"callbackCommand"` | ||||
| 	OperationID       string                  `json:"operationID"` | ||||
| 	GroupID           string                  `json:"groupID"` | ||||
| 	GroupName         string                  `json:"groupName"` | ||||
| 	GroupName         *wrapperspb.StringValue `json:"groupName"` | ||||
| 	Notification      *wrapperspb.StringValue `json:"notification"` | ||||
| 	Introduction      *wrapperspb.StringValue `json:"introduction"` | ||||
| 	FaceURL           *wrapperspb.StringValue `json:"faceURL"` | ||||
| @ -285,6 +285,6 @@ type CallbackAfterSetGroupInfoEXReq struct { | ||||
| 	ApplyMemberFriend *wrapperspb.Int32Value  `json:"applyMemberFriend"` | ||||
| } | ||||
| 
 | ||||
| type CallbackAfterSetGroupInfoEXResp struct { | ||||
| type CallbackAfterSetGroupInfoExResp struct { | ||||
| 	CommonCallbackResp | ||||
| } | ||||
|  | ||||
| @ -72,7 +72,6 @@ type CallbackAfterUpdateUserInfoExResp struct { | ||||
| 
 | ||||
| type CallbackBeforeUserRegisterReq struct { | ||||
| 	CallbackCommand `json:"callbackCommand"` | ||||
| 	Secret          string            `json:"secret"` | ||||
| 	Users           []*sdkws.UserInfo `json:"users"` | ||||
| } | ||||
| 
 | ||||
| @ -83,7 +82,6 @@ type CallbackBeforeUserRegisterResp struct { | ||||
| 
 | ||||
| type CallbackAfterUserRegisterReq struct { | ||||
| 	CallbackCommand `json:"callbackCommand"` | ||||
| 	Secret          string            `json:"secret"` | ||||
| 	Users           []*sdkws.UserInfo `json:"users"` | ||||
| } | ||||
| 
 | ||||
|  | ||||
| @ -185,7 +185,6 @@ type MsgGateway struct { | ||||
| 		WebsocketMaxMsgLen  int   `mapstructure:"websocketMaxMsgLen"` | ||||
| 		WebsocketTimeout    int   `mapstructure:"websocketTimeout"` | ||||
| 	} `mapstructure:"longConnSvr"` | ||||
| 	MultiLoginPolicy int `mapstructure:"multiLoginPolicy"` | ||||
| } | ||||
| 
 | ||||
| type MsgTransfer struct { | ||||
| @ -361,6 +360,7 @@ type Share struct { | ||||
| 	Secret           string          `mapstructure:"secret"` | ||||
| 	RpcRegisterName  RpcRegisterName `mapstructure:"rpcRegisterName"` | ||||
| 	IMAdminUserID    []string        `mapstructure:"imAdminUserID"` | ||||
| 	MultiLoginPolicy int             `mapstructure:"multiLoginPolicy"` | ||||
| } | ||||
| type RpcRegisterName struct { | ||||
| 	User           string `mapstructure:"user"` | ||||
| @ -428,8 +428,8 @@ type Webhooks struct { | ||||
| 	BeforeInviteUserToGroup  BeforeConfig `mapstructure:"beforeInviteUserToGroup"` | ||||
| 	AfterSetGroupInfo        AfterConfig  `mapstructure:"afterSetGroupInfo"` | ||||
| 	BeforeSetGroupInfo       BeforeConfig `mapstructure:"beforeSetGroupInfo"` | ||||
| 	AfterSetGroupInfoEX      AfterConfig  `mapstructure:"afterSetGroupInfoEX"` | ||||
| 	BeforeSetGroupInfoEX     BeforeConfig `mapstructure:"beforeSetGroupInfoEX"` | ||||
| 	AfterSetGroupInfoEx      AfterConfig  `mapstructure:"afterSetGroupInfoEx"` | ||||
| 	BeforeSetGroupInfoEx     BeforeConfig `mapstructure:"beforeSetGroupInfoEx"` | ||||
| 	AfterRevokeMsg           AfterConfig  `mapstructure:"afterRevokeMsg"` | ||||
| 	BeforeAddBlack           BeforeConfig `mapstructure:"beforeAddBlack"` | ||||
| 	AfterAddFriend           AfterConfig  `mapstructure:"afterAddFriend"` | ||||
|  | ||||
| @ -17,6 +17,7 @@ package cachekey | ||||
| const ( | ||||
| 	ConversationKey                          = "CONVERSATION:" | ||||
| 	ConversationIDsKey                       = "CONVERSATION_IDS:" | ||||
| 	NotNotifyConversationIDsKey              = "NOT_NOTIFY_CONVERSATION_IDS:" | ||||
| 	ConversationIDsHashKey                   = "CONVERSATION_IDS_HASH:" | ||||
| 	ConversationHasReadSeqKey                = "CONVERSATION_HAS_READ_SEQ:" | ||||
| 	RecvMsgOptKey                            = "RECV_MSG_OPT:" | ||||
| @ -34,6 +35,10 @@ func GetConversationIDsKey(ownerUserID string) string { | ||||
| 	return ConversationIDsKey + ownerUserID | ||||
| } | ||||
| 
 | ||||
| func GetNotNotifyConversationIDsKey(ownerUserID string) string { | ||||
| 	return NotNotifyConversationIDsKey + ownerUserID | ||||
| } | ||||
| 
 | ||||
| func GetSuperGroupRecvNotNotifyUserIDsKey(groupID string) string { | ||||
| 	return SuperGroupRecvMsgNotNotifyUserIDsKey + groupID | ||||
| } | ||||
|  | ||||
							
								
								
									
										3
									
								
								pkg/common/storage/cache/conversation.go
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								pkg/common/storage/cache/conversation.go
									
									
									
									
										vendored
									
									
								
							| @ -25,6 +25,7 @@ type ConversationCache interface { | ||||
| 	CloneConversationCache() ConversationCache | ||||
| 	// get user's conversationIDs from msgCache | ||||
| 	GetUserConversationIDs(ctx context.Context, ownerUserID string) ([]string, error) | ||||
| 	GetUserNotNotifyConversationIDs(ctx context.Context, userID string) ([]string, error) | ||||
| 	DelConversationIDs(userIDs ...string) ConversationCache | ||||
| 
 | ||||
| 	GetUserConversationIDsHash(ctx context.Context, ownerUserID string) (hash uint64, err error) | ||||
| @ -54,7 +55,7 @@ type ConversationCache interface { | ||||
| 
 | ||||
| 	GetConversationNotReceiveMessageUserIDs(ctx context.Context, conversationID string) ([]string, error) | ||||
| 	DelConversationNotReceiveMessageUserIDs(conversationIDs ...string) ConversationCache | ||||
| 
 | ||||
| 	DelConversationNotNotifyMessageUserIDs(userIDs ...string) ConversationCache | ||||
| 	DelConversationVersionUserIDs(userIDs ...string) ConversationCache | ||||
| 
 | ||||
| 	FindMaxConversationUserVersion(ctx context.Context, userID string) (*relationtb.VersionLog, error) | ||||
|  | ||||
							
								
								
									
										18
									
								
								pkg/common/storage/cache/redis/conversation.go
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										18
									
								
								pkg/common/storage/cache/redis/conversation.go
									
									
									
									
										vendored
									
									
								
							| @ -71,6 +71,10 @@ func (c *ConversationRedisCache) getConversationIDsKey(ownerUserID string) strin | ||||
| 	return cachekey.GetConversationIDsKey(ownerUserID) | ||||
| } | ||||
| 
 | ||||
| func (c *ConversationRedisCache) getNotNotifyConversationIDsKey(ownerUserID string) string { | ||||
| 	return cachekey.GetNotNotifyConversationIDsKey(ownerUserID) | ||||
| } | ||||
| 
 | ||||
| func (c *ConversationRedisCache) getSuperGroupRecvNotNotifyUserIDsKey(groupID string) string { | ||||
| 	return cachekey.GetSuperGroupRecvNotNotifyUserIDsKey(groupID) | ||||
| } | ||||
| @ -105,6 +109,12 @@ 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) DelConversationIDs(userIDs ...string) cache.ConversationCache { | ||||
| 	keys := make([]string, 0, len(userIDs)) | ||||
| 	for _, userID := range userIDs { | ||||
| @ -242,6 +252,14 @@ func (c *ConversationRedisCache) DelConversationNotReceiveMessageUserIDs(convers | ||||
| 	return cache | ||||
| } | ||||
| 
 | ||||
| func (c *ConversationRedisCache) DelConversationNotNotifyMessageUserIDs(userIDs ...string) cache.ConversationCache { | ||||
| 	cache := c.CloneConversationCache() | ||||
| 	for _, userID := range userIDs { | ||||
| 		cache.AddKeys(c.getNotNotifyConversationIDsKey(userID)) | ||||
| 	} | ||||
| 	return cache | ||||
| } | ||||
| 
 | ||||
| func (c *ConversationRedisCache) DelConversationVersionUserIDs(userIDs ...string) cache.ConversationCache { | ||||
| 	cache := c.CloneConversationCache() | ||||
| 	for _, userID := range userIDs { | ||||
|  | ||||
							
								
								
									
										9
									
								
								pkg/common/storage/cache/redis/token.go
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								pkg/common/storage/cache/redis/token.go
									
									
									
									
										vendored
									
									
								
							| @ -19,8 +19,8 @@ import ( | ||||
| 	"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/tools/errs" | ||||
| 	"github.com/openimsdk/tools/utils/stringutil" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"strconv" | ||||
| 	"time" | ||||
| ) | ||||
| 
 | ||||
| @ -58,9 +58,12 @@ func (c *tokenCache) GetTokensWithoutError(ctx context.Context, userID string, p | ||||
| 	} | ||||
| 	mm := make(map[string]int) | ||||
| 	for k, v := range m { | ||||
| 		mm[k] = stringutil.StringToInt(v) | ||||
| 		state, err := strconv.Atoi(v) | ||||
| 		if err != nil { | ||||
| 			return nil, errs.WrapMsg(err, "redis token value is not int", "value", v, "userID", userID, "platformID", platformID) | ||||
| 		} | ||||
| 		mm[k] = state | ||||
| 	} | ||||
| 
 | ||||
| 	return mm, nil | ||||
| } | ||||
| 
 | ||||
|  | ||||
| @ -38,10 +38,11 @@ type authDatabase struct { | ||||
| 	cache            cache.TokenModel | ||||
| 	accessSecret     string | ||||
| 	accessExpire     int64 | ||||
| 	multiLoginPolicy int | ||||
| } | ||||
| 
 | ||||
| func NewAuthDatabase(cache cache.TokenModel, accessSecret string, accessExpire int64) AuthDatabase { | ||||
| 	return &authDatabase{cache: cache, accessSecret: accessSecret, accessExpire: accessExpire} | ||||
| func NewAuthDatabase(cache cache.TokenModel, accessSecret string, accessExpire int64, policy int) AuthDatabase { | ||||
| 	return &authDatabase{cache: cache, accessSecret: accessSecret, accessExpire: accessExpire, multiLoginPolicy: policy} | ||||
| } | ||||
| 
 | ||||
| // If the result is empty. | ||||
| @ -55,15 +56,19 @@ func (a *authDatabase) SetTokenMapByUidPid(ctx context.Context, userID string, p | ||||
| 
 | ||||
| // Create Token. | ||||
| func (a *authDatabase) CreateToken(ctx context.Context, userID string, platformID int) (string, error) { | ||||
| 	// todo: get all platform token | ||||
| 	tokens, err := a.cache.GetTokensWithoutError(ctx, userID, platformID) | ||||
| 	if err != nil { | ||||
| 		return "", err | ||||
| 	} | ||||
| 	var deleteTokenKey []string | ||||
| 	var kickedTokenKey []string | ||||
| 	for k, v := range tokens { | ||||
| 		_, err = tokenverify.GetClaimFromToken(k, authverify.Secret(a.accessSecret)) | ||||
| 		t, err := tokenverify.GetClaimFromToken(k, authverify.Secret(a.accessSecret)) | ||||
| 		if err != nil || v != constant.NormalToken { | ||||
| 			deleteTokenKey = append(deleteTokenKey, k) | ||||
| 		} else if a.checkKickToken(ctx, platformID, t) { | ||||
| 			kickedTokenKey = append(kickedTokenKey, k) | ||||
| 		} | ||||
| 	} | ||||
| 	if len(deleteTokenKey) != 0 { | ||||
| @ -72,6 +77,14 @@ func (a *authDatabase) CreateToken(ctx context.Context, userID string, platformI | ||||
| 			return "", err | ||||
| 		} | ||||
| 	} | ||||
| 	if len(kickedTokenKey) != 0 { | ||||
| 		for _, k := range kickedTokenKey { | ||||
| 			err := a.cache.SetTokenFlagEx(ctx, userID, platformID, k, constant.KickedToken) | ||||
| 			if err != nil { | ||||
| 				return "", err | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	claims := tokenverify.BuildClaims(userID, platformID, a.accessExpire) | ||||
| 	token := jwt.NewWithClaims(jwt.SigningMethodHS256, claims) | ||||
| @ -85,3 +98,23 @@ func (a *authDatabase) CreateToken(ctx context.Context, userID string, platformI | ||||
| 	} | ||||
| 	return tokenString, nil | ||||
| } | ||||
| 
 | ||||
| func (a *authDatabase) checkKickToken(ctx context.Context, platformID int, token *tokenverify.Claims) bool { | ||||
| 	switch a.multiLoginPolicy { | ||||
| 	case constant.DefalutNotKick: | ||||
| 		return false | ||||
| 	case constant.PCAndOther: | ||||
| 		if constant.PlatformIDToClass(platformID) == constant.TerminalPC || | ||||
| 			constant.PlatformIDToClass(token.PlatformID) == constant.TerminalPC { | ||||
| 			return false | ||||
| 		} | ||||
| 		return true | ||||
| 	case constant.AllLoginButSameTermKick: | ||||
| 		if platformID == token.PlatformID { | ||||
| 			return true | ||||
| 		} | ||||
| 		return false | ||||
| 	default: | ||||
| 		return false | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @ -69,6 +69,8 @@ type ConversationDatabase interface { | ||||
| 	FindConversationUserVersion(ctx context.Context, userID string, version uint, limit int) (*relationtb.VersionLog, error) | ||||
| 	FindMaxConversationUserVersionCache(ctx context.Context, userID string) (*relationtb.VersionLog, error) | ||||
| 	GetOwnerConversation(ctx context.Context, ownerUserID string, pagination pagination.Pagination) (int64, []*relationtb.Conversation, error) | ||||
| 	// GetNotNotifyConversationIDs gets not notify conversationIDs by userID | ||||
| 	GetNotNotifyConversationIDs(ctx context.Context, userID string) ([]string, error) | ||||
| } | ||||
| 
 | ||||
| func NewConversationDatabase(conversation database.Conversation, cache cache.ConversationCache, tx tx.Tx) ConversationDatabase { | ||||
| @ -108,6 +110,7 @@ func (c *conversationDatabase) SetUsersConversationFieldTx(ctx context.Context, | ||||
| 			} | ||||
| 			if _, ok := fieldMap["recv_msg_opt"]; ok { | ||||
| 				cache = cache.DelConversationNotReceiveMessageUserIDs(conversation.ConversationID) | ||||
| 				cache = cache.DelConversationNotNotifyMessageUserIDs(userIDs...) | ||||
| 			} | ||||
| 			cache = cache.DelConversationVersionUserIDs(haveUserIDs...) | ||||
| 		} | ||||
| @ -144,6 +147,7 @@ func (c *conversationDatabase) UpdateUsersConversationField(ctx context.Context, | ||||
| 	cache = cache.DelUsersConversation(conversationID, userIDs...).DelConversationVersionUserIDs(userIDs...) | ||||
| 	if _, ok := args["recv_msg_opt"]; ok { | ||||
| 		cache = cache.DelConversationNotReceiveMessageUserIDs(conversationID) | ||||
| 		cache = cache.DelConversationNotNotifyMessageUserIDs(userIDs...) | ||||
| 	} | ||||
| 	return cache.ChainExecDel(ctx) | ||||
| } | ||||
| @ -152,14 +156,22 @@ func (c *conversationDatabase) CreateConversation(ctx context.Context, conversat | ||||
| 	if err := c.conversationDB.Create(ctx, conversations); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	var userIDs []string | ||||
| 	var ( | ||||
| 		userIDs          []string | ||||
| 		notNotifyUserIDs []string | ||||
| 	) | ||||
| 
 | ||||
| 	cache := c.cache.CloneConversationCache() | ||||
| 	for _, conversation := range conversations { | ||||
| 		cache = cache.DelConversations(conversation.OwnerUserID, conversation.ConversationID) | ||||
| 		cache = cache.DelConversationNotReceiveMessageUserIDs(conversation.ConversationID) | ||||
| 		userIDs = append(userIDs, conversation.OwnerUserID) | ||||
| 		if conversation.RecvMsgOpt == constant.ReceiveNotNotifyMessage { | ||||
| 			notNotifyUserIDs = append(notNotifyUserIDs, conversation.OwnerUserID) | ||||
| 		} | ||||
| 	return cache.DelConversationIDs(userIDs...).DelUserConversationIDsHash(userIDs...).DelConversationVersionUserIDs(userIDs...).ChainExecDel(ctx) | ||||
| 	} | ||||
| 	return cache.DelConversationIDs(userIDs...).DelUserConversationIDsHash(userIDs...).DelConversationVersionUserIDs(userIDs...). | ||||
| 		DelConversationNotNotifyMessageUserIDs(notNotifyUserIDs...).ChainExecDel(ctx) | ||||
| } | ||||
| 
 | ||||
| func (c *conversationDatabase) SyncPeerUserPrivateConversationTx(ctx context.Context, conversations []*relationtb.Conversation) error { | ||||
| @ -212,7 +224,8 @@ func (c *conversationDatabase) GetUserAllConversation(ctx context.Context, owner | ||||
| func (c *conversationDatabase) SetUserConversations(ctx context.Context, ownerUserID string, conversations []*relationtb.Conversation) error { | ||||
| 	return c.tx.Transaction(ctx, func(ctx context.Context) error { | ||||
| 		cache := c.cache.CloneConversationCache() | ||||
| 		cache = cache.DelConversationVersionUserIDs(ownerUserID) | ||||
| 		cache = cache.DelConversationVersionUserIDs(ownerUserID).DelConversationNotNotifyMessageUserIDs(ownerUserID) | ||||
| 
 | ||||
| 		groupIDs := datautil.Distinct(datautil.Filter(conversations, func(e *relationtb.Conversation) (string, bool) { | ||||
| 			return e.GroupID, e.GroupID != "" | ||||
| 		})) | ||||
| @ -353,3 +366,11 @@ func (c *conversationDatabase) GetOwnerConversation(ctx context.Context, ownerUs | ||||
| 	} | ||||
| 	return int64(len(conversationIDs)), conversations, nil | ||||
| } | ||||
| 
 | ||||
| func (c *conversationDatabase) GetNotNotifyConversationIDs(ctx context.Context, userID string) ([]string, error) { | ||||
| 	conversationIDs, err := c.cache.GetUserNotNotifyConversationIDs(ctx, userID) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return conversationIDs, nil | ||||
| } | ||||
|  | ||||
| @ -446,7 +446,7 @@ func (db *commonMsgDatabase) GetMsgBySeqsRange(ctx context.Context, userID strin | ||||
| 
 | ||||
| func (db *commonMsgDatabase) GetMsgBySeqs(ctx context.Context, userID string, conversationID string, seqs []int64) (int64, int64, []*sdkws.MsgData, error) { | ||||
| 	userMinSeq, err := db.seqUser.GetUserMinSeq(ctx, conversationID, userID) | ||||
| 	if err != nil && errs.Unwrap(err) != redis.Nil { | ||||
| 	if err != nil { | ||||
| 		return 0, 0, nil, err | ||||
| 	} | ||||
| 	minSeq, err := db.seqConversation.GetMinSeq(ctx, conversationID) | ||||
| @ -457,15 +457,28 @@ func (db *commonMsgDatabase) GetMsgBySeqs(ctx context.Context, userID string, co | ||||
| 	if err != nil { | ||||
| 		return 0, 0, nil, err | ||||
| 	} | ||||
| 	if userMinSeq < minSeq { | ||||
| 	userMaxSeq, err := db.seqUser.GetUserMaxSeq(ctx, conversationID, userID) | ||||
| 	if err != nil { | ||||
| 		return 0, 0, nil, err | ||||
| 	} | ||||
| 	if userMinSeq > minSeq { | ||||
| 		minSeq = userMinSeq | ||||
| 	} | ||||
| 	var newSeqs []int64 | ||||
| 	if userMaxSeq > 0 && userMaxSeq < maxSeq { | ||||
| 		maxSeq = userMaxSeq | ||||
| 	} | ||||
| 	newSeqs := make([]int64, 0, len(seqs)) | ||||
| 	for _, seq := range seqs { | ||||
| 		if seq <= 0 { | ||||
| 			continue | ||||
| 		} | ||||
| 		if seq >= minSeq && seq <= maxSeq { | ||||
| 			newSeqs = append(newSeqs, seq) | ||||
| 		} | ||||
| 	} | ||||
| 	if len(newSeqs) == 0 { | ||||
| 		return minSeq, maxSeq, nil, nil | ||||
| 	} | ||||
| 	successMsgs, failedSeqs, err := db.msg.GetMessagesBySeq(ctx, conversationID, newSeqs) | ||||
| 	if err != nil { | ||||
| 		if err != redis.Nil { | ||||
|  | ||||
| @ -27,6 +27,7 @@ type Conversation interface { | ||||
| 	Find(ctx context.Context, ownerUserID string, conversationIDs []string) (conversations []*model.Conversation, err error) | ||||
| 	FindUserID(ctx context.Context, userIDs []string, conversationIDs []string) ([]string, error) | ||||
| 	FindUserIDAllConversationID(ctx context.Context, userID string) ([]string, error) | ||||
| 	FindUserIDAllNotNotifyConversationID(ctx context.Context, userID string) ([]string, error) | ||||
| 	Take(ctx context.Context, userID, conversationID string) (conversation *model.Conversation, err error) | ||||
| 	FindConversationID(ctx context.Context, userID string, conversationIDs []string) (existConversationID []string, err error) | ||||
| 	FindUserIDAllConversations(ctx context.Context, userID string) (conversations []*model.Conversation, err error) | ||||
|  | ||||
| @ -124,6 +124,13 @@ func (c *ConversationMgo) FindUserIDAllConversationID(ctx context.Context, userI | ||||
| 	return mongoutil.Find[string](ctx, c.coll, bson.M{"owner_user_id": userID}, options.Find().SetProjection(bson.M{"_id": 0, "conversation_id": 1})) | ||||
| } | ||||
| 
 | ||||
| func (c *ConversationMgo) FindUserIDAllNotNotifyConversationID(ctx context.Context, userID string) ([]string, error) { | ||||
| 	return mongoutil.Find[string](ctx, c.coll, bson.M{ | ||||
| 		"owner_user_id": userID, | ||||
| 		"recv_msg_opt":  constant.ReceiveNotNotifyMessage, | ||||
| 	}, options.Find().SetProjection(bson.M{"_id": 0, "conversation_id": 1})) | ||||
| } | ||||
| 
 | ||||
| func (c *ConversationMgo) Take(ctx context.Context, userID, conversationID string) (conversation *model.Conversation, err error) { | ||||
| 	return mongoutil.FindOne[*model.Conversation](ctx, c.coll, bson.M{"owner_user_id": userID, "conversation_id": conversationID}) | ||||
| } | ||||
|  | ||||
| @ -109,12 +109,12 @@ func (u *UserRpcClient) GetUsersInfoMap(ctx context.Context, userIDs []string) ( | ||||
| func (u *UserRpcClient) GetPublicUserInfos( | ||||
| 	ctx context.Context, | ||||
| 	userIDs []string, | ||||
| 	complete bool, | ||||
| ) ([]*sdkws.PublicUserInfo, error) { | ||||
| 	users, err := u.GetUsersInfo(ctx, userIDs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	return datautil.Slice(users, func(e *sdkws.UserInfo) *sdkws.PublicUserInfo { | ||||
| 		return &sdkws.PublicUserInfo{ | ||||
| 			UserID:   e.UserID, | ||||
| @ -127,10 +127,11 @@ func (u *UserRpcClient) GetPublicUserInfos( | ||||
| 
 | ||||
| // GetPublicUserInfo retrieves public information for a single user based on the provided user ID. | ||||
| func (u *UserRpcClient) GetPublicUserInfo(ctx context.Context, userID string) (*sdkws.PublicUserInfo, error) { | ||||
| 	users, err := u.GetPublicUserInfos(ctx, []string{userID}, true) | ||||
| 	users, err := u.GetPublicUserInfos(ctx, []string{userID}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	return users[0], nil | ||||
| } | ||||
| 
 | ||||
| @ -138,12 +139,12 @@ func (u *UserRpcClient) GetPublicUserInfo(ctx context.Context, userID string) (* | ||||
| func (u *UserRpcClient) GetPublicUserInfoMap( | ||||
| 	ctx context.Context, | ||||
| 	userIDs []string, | ||||
| 	complete bool, | ||||
| ) (map[string]*sdkws.PublicUserInfo, error) { | ||||
| 	users, err := u.GetPublicUserInfos(ctx, userIDs, complete) | ||||
| 	users, err := u.GetPublicUserInfos(ctx, userIDs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	return datautil.SliceToMap(users, func(e *sdkws.PublicUserInfo) string { | ||||
| 		return e.UserID | ||||
| 	}), nil | ||||
|  | ||||
| @ -53,7 +53,6 @@ type User struct { | ||||
| 
 | ||||
| // UserRegisterRequest represents a request to register a user. | ||||
| type UserRegisterRequest struct { | ||||
| 	Secret string `json:"secret"` | ||||
| 	Users []User `json:"users"` | ||||
| } | ||||
| 
 | ||||
| @ -109,7 +108,6 @@ func RegisterUser(token, userID, nickname, faceURL string) error { | ||||
| 		FaceURL:  faceURL, | ||||
| 	} | ||||
| 	reqBody := UserRegisterRequest{ | ||||
| 		Secret: SecretKey, | ||||
| 		Users: []User{user}, | ||||
| 	} | ||||
| 	reqBytes, err := json.Marshal(reqBody) | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user