mirror of
				https://github.com/openimsdk/open-im-server.git
				synced 2025-10-26 05:02:11 +08:00 
			
		
		
		
	Merge branch 'main' into changelog-v3.8.3-patch.2
This commit is contained in:
		
						commit
						23a1626ac1
					
				
							
								
								
									
										2
									
								
								.env
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								.env
									
									
									
									
									
								
							| @ -2,7 +2,7 @@ MONGO_IMAGE=mongo:7.0 | |||||||
| REDIS_IMAGE=redis:7.0.0 | REDIS_IMAGE=redis:7.0.0 | ||||||
| KAFKA_IMAGE=bitnami/kafka:3.5.1 | KAFKA_IMAGE=bitnami/kafka:3.5.1 | ||||||
| MINIO_IMAGE=minio/minio:RELEASE.2024-01-11T07-46-16Z | MINIO_IMAGE=minio/minio:RELEASE.2024-01-11T07-46-16Z | ||||||
| ETCD_IMAGE=quay.io/coreos/etcd:v3.5.13 | ETCD_IMAGE=bitnami/etcd:3.5.13 | ||||||
| PROMETHEUS_IMAGE=prom/prometheus:v2.45.6 | 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 | ||||||
|  | |||||||
							
								
								
									
										2
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							| @ -4,7 +4,7 @@ contact_links: | |||||||
|   #   description: "Report a bug in the project" |   #   description: "Report a bug in the project" | ||||||
|   #   file: "bug-report.yml" |   #   file: "bug-report.yml" | ||||||
|   - name: 📢 Connect on slack |   - name: 📢 Connect on slack | ||||||
|     url: https://join.slack.com/t/openimsdk/shared_invite/zt-1tmoj26uf-_FDy3dowVHBiGvLk9e5Xkg |     url: https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A | ||||||
|     about: Support OpenIM-related requests or issues, get in touch with developers and help on slack |     about: Support OpenIM-related requests or issues, get in touch with developers and help on slack | ||||||
|   - name: 🌐 OpenIM Blog |   - name: 🌐 OpenIM Blog | ||||||
|     url: https://www.openim.io/ |     url: https://www.openim.io/ | ||||||
|  | |||||||
							
								
								
									
										3
									
								
								.github/ISSUE_TEMPLATE/other.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.github/ISSUE_TEMPLATE/other.yml
									
									
									
									
										vendored
									
									
								
							| @ -4,7 +4,6 @@ title: "[Other]: <give this problem a name>" | |||||||
| labels: ["other"] | labels: ["other"] | ||||||
| # assignees: [] | # assignees: [] | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| body: | body: | ||||||
|   - type: markdown |   - type: markdown | ||||||
|     attributes: |     attributes: | ||||||
| @ -26,5 +25,5 @@ body: | |||||||
|   - type: markdown |   - type: markdown | ||||||
|     attributes: |     attributes: | ||||||
|       value: | |       value: | | ||||||
|         You can also join our Discord community [here](https://join.slack.com/t/openimsdk/shared_invite/zt-1tmoj26uf-_FDy3dowVHBiGvLk9e5Xkg) |         You can also join our Discord community [here](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
|         Feel free to check out other cool repositories of the openim Community [here](https://github.com/openimsdk) |         Feel free to check out other cool repositories of the openim Community [here](https://github.com/openimsdk) | ||||||
|  | |||||||
							
								
								
									
										5
									
								
								.github/workflows/auto-invite-comment.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.github/workflows/auto-invite-comment.yml
									
									
									
									
										vendored
									
									
								
							| @ -11,7 +11,6 @@ jobs: | |||||||
|     permissions: |     permissions: | ||||||
|       issues: write |       issues: write | ||||||
|     steps: |     steps: | ||||||
| 
 |  | ||||||
|       - name: Invite user to join OpenIM Community |       - name: Invite user to join OpenIM Community | ||||||
|         uses: peter-evans/create-or-update-comment@v4 |         uses: peter-evans/create-or-update-comment@v4 | ||||||
|         with: |         with: | ||||||
| @ -20,11 +19,11 @@ jobs: | |||||||
|           body: | |           body: | | ||||||
|             We value close connections with our users, developers, and contributors here at Open-IM-Server. With a large community and maintainer team, we're always here to help and support you. Whether you're looking to join our community or have any questions or suggestions, we welcome you to get in touch with us. |             We value close connections with our users, developers, and contributors here at Open-IM-Server. With a large community and maintainer team, we're always here to help and support you. Whether you're looking to join our community or have any questions or suggestions, we welcome you to get in touch with us. | ||||||
| 
 | 
 | ||||||
|             Our most recommended way to get in touch is through [Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q). Even if you're in China, Slack is usually not blocked by firewalls, making it an easy way to connect with us. Our Slack community is the ideal place to discuss and share ideas and suggestions with other users and developers of Open-IM-Server. You can ask technical questions, seek help, or share your experiences with other users of Open-IM-Server. |             Our most recommended way to get in touch is through [Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A). Even if you're in China, Slack is usually not blocked by firewalls, making it an easy way to connect with us. Our Slack community is the ideal place to discuss and share ideas and suggestions with other users and developers of Open-IM-Server. You can ask technical questions, seek help, or share your experiences with other users of Open-IM-Server. | ||||||
| 
 | 
 | ||||||
|             In addition to Slack, we also offer the following ways to get in touch: |             In addition to Slack, we also offer the following ways to get in touch: | ||||||
| 
 | 
 | ||||||
|             + <a href="https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q" target="_blank"><img src="https://img.shields.io/badge/Slack-OpenIM%2B-blueviolet?logo=slack&logoColor=white"></a> We also have Slack channels for you to communicate and discuss. To join, visit https://slack.com/ and join our [👀 Open-IM-Server slack](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) team channel. |             + <a href="https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A" target="_blank"><img src="https://img.shields.io/badge/Slack-OpenIM%2B-blueviolet?logo=slack&logoColor=white"></a> We also have Slack channels for you to communicate and discuss. To join, visit https://slack.com/ and join our [👀 Open-IM-Server slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) team channel. | ||||||
|             + <a href="https://mail.google.com/mail/u/0/?fs=1&tf=cm&to=info@openim.io" target="_blank"><img src="https://img.shields.io/badge/gmail-%40OOpenIMSDKCore?style=social&logo=gmail"></a> Get in touch with us on [Gmail](https://mail.google.com/mail/u/0/?fs=1&tf=cm&to=winxu81@gmail.com). If you have any questions or issues that need resolving, or any suggestions and feedback for our open source projects, please feel free to contact us via email. |             + <a href="https://mail.google.com/mail/u/0/?fs=1&tf=cm&to=info@openim.io" target="_blank"><img src="https://img.shields.io/badge/gmail-%40OOpenIMSDKCore?style=social&logo=gmail"></a> Get in touch with us on [Gmail](https://mail.google.com/mail/u/0/?fs=1&tf=cm&to=winxu81@gmail.com). If you have any questions or issues that need resolving, or any suggestions and feedback for our open source projects, please feel free to contact us via email. | ||||||
|             + <a href="https://doc.rentsoft.cn/" target="_blank"><img src="https://img.shields.io/badge/%E5%8D%9A%E5%AE%A2-%40OpenIMSDKCore-blue?style=social&logo=Octopus%20Deploy"></a> Read our [blog](https://doc.rentsoft.cn/). Our blog is a great place to stay up-to-date with Open-IM-Server projects and trends. On the blog, we share our latest developments, tech trends, and other interesting information. |             + <a href="https://doc.rentsoft.cn/" target="_blank"><img src="https://img.shields.io/badge/%E5%8D%9A%E5%AE%A2-%40OpenIMSDKCore-blue?style=social&logo=Octopus%20Deploy"></a> Read our [blog](https://doc.rentsoft.cn/). Our blog is a great place to stay up-to-date with Open-IM-Server projects and trends. On the blog, we share our latest developments, tech trends, and other interesting information. | ||||||
|             + <a href="https://github.com/OpenIMSDK/OpenIM-Docs/blob/main/docs/images/WechatIMG20.jpeg" target="_blank"><img src="https://img.shields.io/badge/%E5%BE%AE%E4%BF%A1-OpenIMSDKCore-brightgreen?logo=wechat&style=flat-square"></a> Add [Wechat](https://github.com/OpenIMSDK/OpenIM-Docs/blob/main/docs/images/WechatIMG20.jpeg) and indicate that you are a user or developer of Open-IM-Server. We will process your request as soon as possible. |             + <a href="https://github.com/OpenIMSDK/OpenIM-Docs/blob/main/docs/images/WechatIMG20.jpeg" target="_blank"><img src="https://img.shields.io/badge/%E5%BE%AE%E4%BF%A1-OpenIMSDKCore-brightgreen?logo=wechat&style=flat-square"></a> Add [Wechat](https://github.com/OpenIMSDK/OpenIM-Docs/blob/main/docs/images/WechatIMG20.jpeg) and indicate that you are a user or developer of Open-IM-Server. We will process your request as soon as possible. | ||||||
|  | |||||||
							
								
								
									
										25
									
								
								.github/workflows/go-build-test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										25
									
								
								.github/workflows/go-build-test.yml
									
									
									
									
										vendored
									
									
								
							| @ -12,6 +12,10 @@ jobs: | |||||||
|   go-build: |   go-build: | ||||||
|     name: Test with go ${{ matrix.go_version }} on ${{ matrix.os }} |     name: Test with go ${{ matrix.go_version }} on ${{ matrix.os }} | ||||||
|     runs-on: ${{ matrix.os }} |     runs-on: ${{ matrix.os }} | ||||||
|  | 
 | ||||||
|  |     env: | ||||||
|  |       SHARE_CONFIG_PATH: config/share.yml | ||||||
|  | 
 | ||||||
|     permissions: |     permissions: | ||||||
|       contents: write |       contents: write | ||||||
|       pull-requests: write |       pull-requests: write | ||||||
| @ -40,6 +44,10 @@ jobs: | |||||||
|         with: |         with: | ||||||
|           compose-file: "./docker-compose.yml" |           compose-file: "./docker-compose.yml" | ||||||
| 
 | 
 | ||||||
|  |       - name: Modify Server Configuration | ||||||
|  |         run: | | ||||||
|  |           yq e '.secret = 123456' -i ${{ env.SHARE_CONFIG_PATH }} | ||||||
|  | 
 | ||||||
|       # - name: Get Internal IP Address |       # - name: Get Internal IP Address | ||||||
|       #   id: get-ip |       #   id: get-ip | ||||||
|       #   run: | |       #   run: | | ||||||
| @ -71,6 +79,11 @@ jobs: | |||||||
|           go mod download |           go mod download | ||||||
|           go install github.com/magefile/mage@latest |           go install github.com/magefile/mage@latest | ||||||
| 
 | 
 | ||||||
|  |       - name: Modify Chat Configuration | ||||||
|  |         run: | | ||||||
|  |           cd ${{ github.workspace }}/chat-repo | ||||||
|  |           yq e '.openIM.secret = 123456' -i ${{ env.SHARE_CONFIG_PATH }} | ||||||
|  | 
 | ||||||
|       - name: Build and test Chat Services |       - name: Build and test Chat Services | ||||||
|         run: | |         run: | | ||||||
|           cd ${{ github.workspace }}/chat-repo |           cd ${{ github.workspace }}/chat-repo | ||||||
| @ -132,7 +145,7 @@ jobs: | |||||||
| 
 | 
 | ||||||
|           # Test get admin token |           # Test get admin token | ||||||
|           get_admin_token_response=$(curl -X POST -H "Content-Type: application/json" -H "operationID: imAdmin" -d '{ |           get_admin_token_response=$(curl -X POST -H "Content-Type: application/json" -H "operationID: imAdmin" -d '{ | ||||||
|             "secret": "openIM123", |             "secret": "123456", | ||||||
|             "platformID": 2, |             "platformID": 2, | ||||||
|             "userID": "imAdmin" |             "userID": "imAdmin" | ||||||
|           }' http://127.0.0.1:10002/auth/get_admin_token) |           }' http://127.0.0.1:10002/auth/get_admin_token) | ||||||
| @ -169,7 +182,8 @@ jobs: | |||||||
|       contents: write |       contents: write | ||||||
|     env: |     env: | ||||||
|       SDK_DIR: openim-sdk-core |       SDK_DIR: openim-sdk-core | ||||||
|       CONFIG_PATH: config/notification.yml |       NOTIFICATION_CONFIG_PATH: config/notification.yml | ||||||
|  |       SHARE_CONFIG_PATH: config/share.yml | ||||||
| 
 | 
 | ||||||
|     strategy: |     strategy: | ||||||
|       matrix: |       matrix: | ||||||
| @ -184,7 +198,7 @@ jobs: | |||||||
|         uses: actions/checkout@v4 |         uses: actions/checkout@v4 | ||||||
|         with: |         with: | ||||||
|           repository: "openimsdk/openim-sdk-core" |           repository: "openimsdk/openim-sdk-core" | ||||||
|           ref: "release-v3.8" |           ref: "main" | ||||||
|           path: ${{ env.SDK_DIR }} |           path: ${{ env.SDK_DIR }} | ||||||
| 
 | 
 | ||||||
|       - name: Set up Go ${{ matrix.go_version }} |       - name: Set up Go ${{ matrix.go_version }} | ||||||
| @ -199,8 +213,9 @@ jobs: | |||||||
| 
 | 
 | ||||||
|       - name: Modify Server Configuration |       - name: Modify Server Configuration | ||||||
|         run: | |         run: | | ||||||
|           yq e '.groupCreated.isSendMsg = true' -i ${{ env.CONFIG_PATH }} |           yq e '.groupCreated.isSendMsg = true' -i ${{ env.NOTIFICATION_CONFIG_PATH }} | ||||||
|           yq e '.friendApplicationApproved.isSendMsg = true' -i ${{ env.CONFIG_PATH }} |           yq e '.friendApplicationApproved.isSendMsg = true' -i ${{ env.NOTIFICATION_CONFIG_PATH }} | ||||||
|  |           yq e '.secret = 123456' -i ${{ env.SHARE_CONFIG_PATH }} | ||||||
| 
 | 
 | ||||||
|       - name: Start Server Services |       - name: Start Server Services | ||||||
|         run: | |         run: | | ||||||
|  | |||||||
							
								
								
									
										2
									
								
								.github/workflows/help-comment-issue.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/help-comment-issue.yml
									
									
									
									
										vendored
									
									
								
							| @ -32,5 +32,5 @@ jobs: | |||||||
|           token: ${{ secrets.BOT_TOKEN }} |           token: ${{ secrets.BOT_TOKEN }} | ||||||
|           body: | |           body: | | ||||||
|             This issue is available for anyone to work on. **Make sure to reference this issue in your pull request.** :sparkles: Thank you for your contribution! :sparkles: |             This issue is available for anyone to work on. **Make sure to reference this issue in your pull request.** :sparkles: Thank you for your contribution! :sparkles: | ||||||
|             [Join slack 🤖](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) to connect and communicate with our developers. |             [Join slack 🤖](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) to connect and communicate with our developers. | ||||||
|             If you wish to accept this assignment, please leave a comment in the comments section: `/accept`.🎯 |             If you wish to accept this assignment, please leave a comment in the comments section: `/accept`.🎯 | ||||||
|  | |||||||
							
								
								
									
										20
									
								
								.github/workflows/merge-from-milestone.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										20
									
								
								.github/workflows/merge-from-milestone.yml
									
									
									
									
										vendored
									
									
								
							| @ -155,11 +155,27 @@ jobs: | |||||||
|                   '{title: $title, head: $head, base: $base, body: $body}')") |                   '{title: $title, head: $head, base: $base, body: $body}')") | ||||||
| 
 | 
 | ||||||
|               new_pr_number=$(echo "$response" | jq -r '.number') |               new_pr_number=$(echo "$response" | jq -r '.number') | ||||||
|               echo "Created PR #$new_pr_number" |  | ||||||
| 
 | 
 | ||||||
|               curl -s -X POST -H "Authorization: token $GITHUB_TOKEN" \ |               if [[ "$new_pr_number" == "null" || -z "$new_pr_number" ]]; then | ||||||
|  |                 echo "Failed to create PR. Response: $response" | ||||||
|  |                | ||||||
|  |                 git checkout $TARGET_BRANCH | ||||||
|  | 
 | ||||||
|  |                 git branch -D $cherry_pick_branch | ||||||
|  |                  | ||||||
|  |                 echo "Deleted branch: $cherry_pick_branch" | ||||||
|  |                 git push origin --delete $cherry_pick_branch | ||||||
|  |               else | ||||||
|  |                 echo "Created PR #$new_pr_number" | ||||||
|  | 
 | ||||||
|  |                 curl -s -X POST -H "Authorization: token $GITHUB_TOKEN" \ | ||||||
|                 -H "Accept: application/vnd.github+json" \ |                 -H "Accept: application/vnd.github+json" \ | ||||||
|                 -d '{"labels": ["milestone-merge"]}' \ |                 -d '{"labels": ["milestone-merge"]}' \ | ||||||
|                 "https://api.github.com/repos/${{ github.repository }}/issues/$new_pr_number/labels" |                 "https://api.github.com/repos/${{ github.repository }}/issues/$new_pr_number/labels" | ||||||
|  |               fi | ||||||
|  | 
 | ||||||
|  |               echo "" | ||||||
|  |               echo "----------------------------------------" | ||||||
|  |               echo "" | ||||||
|             fi |             fi | ||||||
|           done |           done | ||||||
|  | |||||||
| @ -15,12 +15,33 @@ jobs: | |||||||
|         uses: actions/checkout@v4 |         uses: actions/checkout@v4 | ||||||
|         with: |         with: | ||||||
|           fetch-depth: 0 |           fetch-depth: 0 | ||||||
|  |           # submodules: "recursive" | ||||||
|  | 
 | ||||||
|  |       - name: Safe submodule initialization | ||||||
|  |         run: | | ||||||
|  |           echo "Checking for submodules..." | ||||||
|  |           if [ -f .gitmodules ]; then | ||||||
|  |             if [ -s .gitmodules ]; then | ||||||
|  |               echo "Initializing submodules..." | ||||||
|  |               if git submodule sync --recursive 2>/dev/null; then | ||||||
|  |                 git submodule update --init --force --recursive || { | ||||||
|  |                   echo "Warning: Some submodules failed to initialize, continuing anyway..." | ||||||
|  |                 } | ||||||
|  |               else | ||||||
|  |                 echo "Warning: Submodule sync failed, continuing without submodules..." | ||||||
|  |               fi | ||||||
|  |             else | ||||||
|  |               echo ".gitmodules exists but is empty, skipping submodule initialization" | ||||||
|  |             fi | ||||||
|  |           else | ||||||
|  |             echo "No .gitmodules file found, no submodules to initialize" | ||||||
|  |           fi | ||||||
| 
 | 
 | ||||||
|       # Step 2: Set up Git with official account |       # Step 2: Set up Git with official account | ||||||
|       - name: Set up Git |       - name: Set up Git | ||||||
|         run: | |         run: | | ||||||
|           git config user.name "github-actions[bot]" |           git config --global user.name "github-actions[bot]" | ||||||
|           git config user.email "github-actions[bot]@users.noreply.github.com" |           git config --global user.email "github-actions[bot]@users.noreply.github.com" | ||||||
| 
 | 
 | ||||||
|       # Step 3: Check and delete existing tag |       # Step 3: Check and delete existing tag | ||||||
|       - name: Check and delete existing tag |       - name: Check and delete existing tag | ||||||
| @ -33,7 +54,8 @@ jobs: | |||||||
|       # Step 4: Update version file |       # Step 4: Update version file | ||||||
|       - name: Update version file |       - name: Update version file | ||||||
|         run: | |         run: | | ||||||
|           echo "${{ env.TAG_VERSION }}" > version/version |           mkdir -p version | ||||||
|  |           echo -n "${{ env.TAG_VERSION }}" > version/version | ||||||
| 
 | 
 | ||||||
|       # Step 5: Commit and push changes |       # Step 5: Commit and push changes | ||||||
|       - name: Commit and push changes |       - name: Commit and push changes | ||||||
| @ -42,43 +64,56 @@ jobs: | |||||||
|         run: | |         run: | | ||||||
|           git add version/version |           git add version/version | ||||||
|           git commit -m "Update version to ${{ env.TAG_VERSION }}" |           git commit -m "Update version to ${{ env.TAG_VERSION }}" | ||||||
|           git push origin HEAD:${{ github.ref }} |  | ||||||
| 
 | 
 | ||||||
|       # Step 6: Create and push tag |       # Step 6: Update tag | ||||||
|       - name: Create and push tag |       - name: Update tag | ||||||
|         env: |  | ||||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} |  | ||||||
|         run: | |         run: | | ||||||
|           git tag ${{ env.TAG_VERSION }} |           git tag -fa ${{ env.TAG_VERSION }} -m "Update version to ${{ env.TAG_VERSION }}" | ||||||
|           git push origin ${{ env.TAG_VERSION }} |           git push origin ${{ env.TAG_VERSION }} --force | ||||||
| 
 | 
 | ||||||
|       # Step 7: Find and Publish Draft Release |       # Step 7: Find and Publish Draft Release | ||||||
|       - name: Find and Publish Draft Release |       - name: Find and Publish Draft Release | ||||||
|         uses: actions/github-script@v6 |         uses: actions/github-script@v7 | ||||||
|         with: |         with: | ||||||
|           github-token: ${{ secrets.GITHUB_TOKEN }} |           github-token: ${{ secrets.GITHUB_TOKEN }} | ||||||
|           script: | |           script: | | ||||||
|             // Get the list of releases |             const { owner, repo } = context.repo; | ||||||
|             const releases = await github.rest.repos.listReleases({ |             const tagName = process.env.TAG_VERSION; | ||||||
|               owner: context.repo.owner, |  | ||||||
|               repo: context.repo.repo |  | ||||||
|             }); |  | ||||||
| 
 | 
 | ||||||
|             // Find the draft release where the title and tag_name are the same |             try { | ||||||
|             const draftRelease = releases.data.find(release =>  |               let release; | ||||||
|               release.draft && release.name === release.tag_name |               try { | ||||||
|             ); |                 const response = await github.rest.repos.getReleaseByTag({ | ||||||
|  |                   owner, | ||||||
|  |                   repo, | ||||||
|  |                   tag: tagName | ||||||
|  |                 }); | ||||||
|  |                 release = response.data; | ||||||
|  |               } catch (tagError) { | ||||||
|  |                 core.info(`Release not found by tag, searching all releases...`); | ||||||
|  |                 const releases = await github.rest.repos.listReleases({ | ||||||
|  |                   owner, | ||||||
|  |                   repo, | ||||||
|  |                   per_page: 100 | ||||||
|  |                 }); | ||||||
|  |                  | ||||||
|  |                 release = releases.data.find(r => r.draft && r.tag_name === tagName);             | ||||||
|  |                 if (!release) { | ||||||
|  |                   throw new Error(`No release found with tag ${tagName}`); | ||||||
|  |                 } | ||||||
|  |               } | ||||||
|                |                | ||||||
|             if (draftRelease) { |  | ||||||
|               // Publish the draft release using the release_id |  | ||||||
|               await github.rest.repos.updateRelease({ |               await github.rest.repos.updateRelease({ | ||||||
|                 owner: context.repo.owner, |                 owner, | ||||||
|                 repo: context.repo.repo, |                 repo, | ||||||
|                 release_id: draftRelease.id,  // Use release_id |                 release_id: release.id, | ||||||
|                 draft: false |                 draft: false, | ||||||
|  |                 prerelease: release.prerelease | ||||||
|               }); |               }); | ||||||
|                |                | ||||||
|               core.info(`Draft Release ${draftRelease.tag_name} published successfully.`); |               const status = release.draft ? "was draft" : "was already published"; | ||||||
|             } else { |               core.info(`Release ${tagName} ensured to be published (${status}).`); | ||||||
|               core.info("No matching draft release found."); |                | ||||||
|  |             } catch (error) { | ||||||
|  |               core.warning(`Could not find or update release for tag ${tagName}: ${error.message}`); | ||||||
|             } |             } | ||||||
							
								
								
									
										4
									
								
								.github/workflows/user-first-interaction.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/user-first-interaction.yml
									
									
									
									
										vendored
									
									
								
							| @ -22,7 +22,7 @@ jobs: | |||||||
| 
 | 
 | ||||||
|             If you are implementing a feature request, please check with the maintainers that the feature will be accepted first. |             If you are implementing a feature request, please check with the maintainers that the feature will be accepted first. | ||||||
| 
 | 
 | ||||||
|             [Join slack 🤖](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) to connect and communicate with our developers. |             [Join slack 🤖](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) to connect and communicate with our developers. | ||||||
| 
 | 
 | ||||||
|             Please leave your information in the [✨ discussions](https://github.com/orgs/OpenIMSDK/discussions/426), we expect anyone to join OpenIM developer community. |             Please leave your information in the [✨ discussions](https://github.com/orgs/OpenIMSDK/discussions/426), we expect anyone to join OpenIM developer community. | ||||||
| 
 | 
 | ||||||
| @ -31,5 +31,5 @@ jobs: | |||||||
| 
 | 
 | ||||||
|             If this is a bug report, please include relevant logs to help us debug the problem. |             If this is a bug report, please include relevant logs to help us debug the problem. | ||||||
| 
 | 
 | ||||||
|             [Join slack 🤖](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) to connect and communicate with our developers. |             [Join slack 🤖](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) to connect and communicate with our developers. | ||||||
|         continue-on-error: true |         continue-on-error: true | ||||||
| @ -28,6 +28,8 @@ run: | |||||||
|   #   - util |   #   - util | ||||||
|   #   - .*~ |   #   - .*~ | ||||||
|   #   - api/swagger/docs |   #   - api/swagger/docs | ||||||
|  | 
 | ||||||
|  |    | ||||||
|   #   - server/docs |   #   - server/docs | ||||||
|   #   - components/mnt/config/certs |   #   - components/mnt/config/certs | ||||||
|   #   - logs |   #   - logs | ||||||
|  | |||||||
| @ -5,3 +5,22 @@ | |||||||
| 
 | 
 | ||||||
| **Full Changelog**: [v3.8.3-patch.1...v3.8.3-patch.2](https://github.com/openimsdk/open-im-server/compare/v3.8.3-patch.1...v3.8.3-patch.2) | **Full Changelog**: [v3.8.3-patch.1...v3.8.3-patch.2](https://github.com/openimsdk/open-im-server/compare/v3.8.3-patch.1...v3.8.3-patch.2) | ||||||
| 
 | 
 | ||||||
|  | ## [v3.8.3-patch.1](https://github.com/openimsdk/open-im-server/releases/tag/v3.8.3-patch.1) 	(2025-02-25) | ||||||
|  | 
 | ||||||
|  | ### New Features | ||||||
|  | * feat: add backup volume && optimize log print [Created [#3121](https://github.com/openimsdk/open-im-server/pull/3121) | ||||||
|  | 
 | ||||||
|  | ### Bug Fixes | ||||||
|  | * fix: seq conversion failed without exiting [Created [#3120](https://github.com/openimsdk/open-im-server/pull/3120) | ||||||
|  | * fix: check error in BatchSetTokenMapByUidPid [Created [#3123](https://github.com/openimsdk/open-im-server/pull/3123) | ||||||
|  | * fix: DeleteDoc crash [Created [#3124](https://github.com/openimsdk/open-im-server/pull/3124) | ||||||
|  | * fix: the abnormal message has no sending time, causing the SDK to be abnormal [Created [#3126](https://github.com/openimsdk/open-im-server/pull/3126) | ||||||
|  | * fix: crash caused [#3127](https://github.com/openimsdk/open-im-server/pull/3127) | ||||||
|  | * fix: the user sets the conversation timer cleanup timestamp unit incorrectly [Created [#3128](https://github.com/openimsdk/open-im-server/pull/3128) | ||||||
|  | * fix: seq conversion not reading env in docker environment [Created [#3131](https://github.com/openimsdk/open-im-server/pull/3131) | ||||||
|  | 
 | ||||||
|  | ### Builds | ||||||
|  | * build: improve workflows contents. [Created [#3125](https://github.com/openimsdk/open-im-server/pull/3125) | ||||||
|  | 
 | ||||||
|  | **Full Changelog**: [v3.8.3-e-v1.1.5...v3.8.3-patch.1-e-v1.1.5](https://github.com/openimsdk/open-im-server-enterprise/compare/v3.8.3-e-v1.1.5...v3.8.3-patch.1-e-v1.1.5) | ||||||
|  | 
 | ||||||
|  | |||||||
							
								
								
									
										210
									
								
								LICENSE
									
									
									
									
									
								
							
							
						
						
									
										210
									
								
								LICENSE
									
									
									
									
									
								
							| @ -1,35 +1,201 @@ | |||||||
| # Open Source License |                                  Apache License | ||||||
|  |                            Version 2.0, January 2004 | ||||||
|  |                         http://www.apache.org/licenses/ | ||||||
| 
 | 
 | ||||||
| OpenIM is licensed under the Apache License 2.0, with the following additional conditions: |    TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION | ||||||
| 
 | 
 | ||||||
| 1. OpenIM may be utilized commercially, including as a backend service for other applications or as an application development platform for enterprises.  |    1. Definitions. | ||||||
| A commercial license must be obtained from the producer if: |  | ||||||
|     - Under no circumstances may you operate a multi-tenant or multi-business environment using the OpenIM source code, whether or not you have modified the repository code. In other words, a single instance of OpenIM may not simultaneously serve multiple enterprises, nor may it serve multiple lines of business within the same enterprise. |  | ||||||
|    - If you intend to operate in such a multi-tenant or multi-business manner, you must obtain a commercial license from the producer in advance. |  | ||||||
| 
 | 
 | ||||||
| 2. As a contributor, you should agree that: |       "License" shall mean the terms and conditions for use, reproduction, | ||||||
|  |       and distribution as defined by Sections 1 through 9 of this document. | ||||||
| 
 | 
 | ||||||
|    a. The producer can adjust the open-source agreement to be more strict or more relaxed as deemed necessary.   |       "Licensor" shall mean the copyright owner or entity authorized by | ||||||
|    b. Your contributed code may be used for commercial purposes, including but not limited to its cloud business operations. |       the copyright owner that is granting the License. | ||||||
| 
 | 
 | ||||||
| Apart from the specific conditions mentioned above, all other rights and restrictions follow the Apache License 2.0. Detailed information about the Apache License 2.0 can be found at http://www.apache.org/licenses/LICENSE-2.0. |       "Legal Entity" shall mean the union of the acting entity and all | ||||||
|  |       other entities that control, are controlled by, or are under common | ||||||
|  |       control with that entity. For the purposes of this definition, | ||||||
|  |       "control" means (i) the power, direct or indirect, to cause the | ||||||
|  |       direction or management of such entity, whether by contract or | ||||||
|  |       otherwise, or (ii) ownership of fifty percent (50%) or more of the | ||||||
|  |       outstanding shares, or (iii) beneficial ownership of such entity. | ||||||
| 
 | 
 | ||||||
|  |       "You" (or "Your") shall mean an individual or Legal Entity | ||||||
|  |       exercising permissions granted by this License. | ||||||
| 
 | 
 | ||||||
| For any licensing-related questions or to obtain a commercial license, please contact contact@openim.io. |       "Source" form shall mean the preferred form for making modifications, | ||||||
|  |       including but not limited to software source code, documentation | ||||||
|  |       source, and configuration files. | ||||||
| 
 | 
 | ||||||
|  |       "Object" form shall mean any form resulting from mechanical | ||||||
|  |       transformation or translation of a Source form, including but | ||||||
|  |       not limited to compiled object code, generated documentation, | ||||||
|  |       and conversions to other media types. | ||||||
| 
 | 
 | ||||||
| © 2024 OpenIMSDK |       "Work" shall mean the work of authorship, whether in Source or | ||||||
|  |       Object form, made available under the License, as indicated by a | ||||||
|  |       copyright notice that is included in or attached to the work | ||||||
|  |       (an example is provided in the Appendix below). | ||||||
| 
 | 
 | ||||||
| ---------- |       "Derivative Works" shall mean any work, whether in Source or Object | ||||||
|  |       form, that is based on (or derived from) the Work and for which the | ||||||
|  |       editorial revisions, annotations, elaborations, or other modifications | ||||||
|  |       represent, as a whole, an original work of authorship. For the purposes | ||||||
|  |       of this License, Derivative Works shall not include works that remain | ||||||
|  |       separable from, or merely link (or bind by name) to the interfaces of, | ||||||
|  |       the Work and Derivative Works thereof. | ||||||
| 
 | 
 | ||||||
| Licensed under the Apache License, Version 2.0 (the "License");   |       "Contribution" shall mean any work of authorship, including | ||||||
| you may not use this file except in compliance with the License.   |       the original version of the Work and any modifications or additions | ||||||
| You may obtain a copy of the License at |       to that Work or Derivative Works thereof, that is intentionally | ||||||
|  |       submitted to Licensor for inclusion in the Work by the copyright owner | ||||||
|  |       or by an individual or Legal Entity authorized to submit on behalf of | ||||||
|  |       the copyright owner. For the purposes of this definition, "submitted" | ||||||
|  |       means any form of electronic, verbal, or written communication sent | ||||||
|  |       to the Licensor or its representatives, including but not limited to | ||||||
|  |       communication on electronic mailing lists, source code control systems, | ||||||
|  |       and issue tracking systems that are managed by, or on behalf of, the | ||||||
|  |       Licensor for the purpose of discussing and improving the Work, but | ||||||
|  |       excluding communication that is conspicuously marked or otherwise | ||||||
|  |       designated in writing by the copyright owner as "Not a Contribution." | ||||||
| 
 | 
 | ||||||
|     http://www.apache.org/licenses/LICENSE-2.0 |       "Contributor" shall mean Licensor and any individual or Legal Entity | ||||||
|  |       on behalf of whom a Contribution has been received by Licensor and | ||||||
|  |       subsequently incorporated within the Work. | ||||||
| 
 | 
 | ||||||
| Unless required by applicable law or agreed to in writing, software   |    2. Grant of Copyright License. Subject to the terms and conditions of | ||||||
| distributed under the License is distributed on an "AS IS" BASIS,   |       this License, each Contributor hereby grants to You a perpetual, | ||||||
| WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.   |       worldwide, non-exclusive, no-charge, royalty-free, irrevocable | ||||||
| See the License for the specific language governing permissions and   |       copyright license to reproduce, prepare Derivative Works of, | ||||||
| limitations under the License. |       publicly display, publicly perform, sublicense, and distribute the | ||||||
|  |       Work and such Derivative Works in Source or Object form. | ||||||
|  | 
 | ||||||
|  |    3. Grant of Patent License. Subject to the terms and conditions of | ||||||
|  |       this License, each Contributor hereby grants to You a perpetual, | ||||||
|  |       worldwide, non-exclusive, no-charge, royalty-free, irrevocable | ||||||
|  |       (except as stated in this section) patent license to make, have made, | ||||||
|  |       use, offer to sell, sell, import, and otherwise transfer the Work, | ||||||
|  |       where such license applies only to those patent claims licensable | ||||||
|  |       by such Contributor that are necessarily infringed by their | ||||||
|  |       Contribution(s) alone or by combination of their Contribution(s) | ||||||
|  |       with the Work to which such Contribution(s) was submitted. If You | ||||||
|  |       institute patent litigation against any entity (including a | ||||||
|  |       cross-claim or counterclaim in a lawsuit) alleging that the Work | ||||||
|  |       or a Contribution incorporated within the Work constitutes direct | ||||||
|  |       or contributory patent infringement, then any patent licenses | ||||||
|  |       granted to You under this License for that Work shall terminate | ||||||
|  |       as of the date such litigation is filed. | ||||||
|  | 
 | ||||||
|  |    4. Redistribution. You may reproduce and distribute copies of the | ||||||
|  |       Work or Derivative Works thereof in any medium, with or without | ||||||
|  |       modifications, and in Source or Object form, provided that You | ||||||
|  |       meet the following conditions: | ||||||
|  | 
 | ||||||
|  |       (a) You must give any other recipients of the Work or | ||||||
|  |           Derivative Works a copy of this License; and | ||||||
|  | 
 | ||||||
|  |       (b) You must cause any modified files to carry prominent notices | ||||||
|  |           stating that You changed the files; and | ||||||
|  | 
 | ||||||
|  |       (c) You must retain, in the Source form of any Derivative Works | ||||||
|  |           that You distribute, all copyright, patent, trademark, and | ||||||
|  |           attribution notices from the Source form of the Work, | ||||||
|  |           excluding those notices that do not pertain to any part of | ||||||
|  |           the Derivative Works; and | ||||||
|  | 
 | ||||||
|  |       (d) If the Work includes a "NOTICE" text file as part of its | ||||||
|  |           distribution, then any Derivative Works that You distribute must | ||||||
|  |           include a readable copy of the attribution notices contained | ||||||
|  |           within such NOTICE file, excluding those notices that do not | ||||||
|  |           pertain to any part of the Derivative Works, in at least one | ||||||
|  |           of the following places: within a NOTICE text file distributed | ||||||
|  |           as part of the Derivative Works; within the Source form or | ||||||
|  |           documentation, if provided along with the Derivative Works; or, | ||||||
|  |           within a display generated by the Derivative Works, if and | ||||||
|  |           wherever such third-party notices normally appear. The contents | ||||||
|  |           of the NOTICE file are for informational purposes only and | ||||||
|  |           do not modify the License. You may add Your own attribution | ||||||
|  |           notices within Derivative Works that You distribute, alongside | ||||||
|  |           or as an addendum to the NOTICE text from the Work, provided | ||||||
|  |           that such additional attribution notices cannot be construed | ||||||
|  |           as modifying the License. | ||||||
|  | 
 | ||||||
|  |       You may add Your own copyright statement to Your modifications and | ||||||
|  |       may provide additional or different license terms and conditions | ||||||
|  |       for use, reproduction, or distribution of Your modifications, or | ||||||
|  |       for any such Derivative Works as a whole, provided Your use, | ||||||
|  |       reproduction, and distribution of the Work otherwise complies with | ||||||
|  |       the conditions stated in this License. | ||||||
|  | 
 | ||||||
|  |    5. Submission of Contributions. Unless You explicitly state otherwise, | ||||||
|  |       any Contribution intentionally submitted for inclusion in the Work | ||||||
|  |       by You to the Licensor shall be under the terms and conditions of | ||||||
|  |       this License, without any additional terms or conditions. | ||||||
|  |       Notwithstanding the above, nothing herein shall supersede or modify | ||||||
|  |       the terms of any separate license agreement you may have executed | ||||||
|  |       with Licensor regarding such Contributions. | ||||||
|  | 
 | ||||||
|  |    6. Trademarks. This License does not grant permission to use the trade | ||||||
|  |       names, trademarks, service marks, or product names of the Licensor, | ||||||
|  |       except as required for reasonable and customary use in describing the | ||||||
|  |       origin of the Work and reproducing the content of the NOTICE file. | ||||||
|  | 
 | ||||||
|  |    7. Disclaimer of Warranty. Unless required by applicable law or | ||||||
|  |       agreed to in writing, Licensor provides the Work (and each | ||||||
|  |       Contributor provides its Contributions) on an "AS IS" BASIS, | ||||||
|  |       WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or | ||||||
|  |       implied, including, without limitation, any warranties or conditions | ||||||
|  |       of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A | ||||||
|  |       PARTICULAR PURPOSE. You are solely responsible for determining the | ||||||
|  |       appropriateness of using or redistributing the Work and assume any | ||||||
|  |       risks associated with Your exercise of permissions under this License. | ||||||
|  | 
 | ||||||
|  |    8. Limitation of Liability. In no event and under no legal theory, | ||||||
|  |       whether in tort (including negligence), contract, or otherwise, | ||||||
|  |       unless required by applicable law (such as deliberate and grossly | ||||||
|  |       negligent acts) or agreed to in writing, shall any Contributor be | ||||||
|  |       liable to You for damages, including any direct, indirect, special, | ||||||
|  |       incidental, or consequential damages of any character arising as a | ||||||
|  |       result of this License or out of the use or inability to use the | ||||||
|  |       Work (including but not limited to damages for loss of goodwill, | ||||||
|  |       work stoppage, computer failure or malfunction, or any and all | ||||||
|  |       other commercial damages or losses), even if such Contributor | ||||||
|  |       has been advised of the possibility of such damages. | ||||||
|  | 
 | ||||||
|  |    9. Accepting Warranty or Additional Liability. While redistributing | ||||||
|  |       the Work or Derivative Works thereof, You may choose to offer, | ||||||
|  |       and charge a fee for, acceptance of support, warranty, indemnity, | ||||||
|  |       or other liability obligations and/or rights consistent with this | ||||||
|  |       License. However, in accepting such obligations, You may act only | ||||||
|  |       on Your own behalf and on Your sole responsibility, not on behalf | ||||||
|  |       of any other Contributor, and only if You agree to indemnify, | ||||||
|  |       defend, and hold each Contributor harmless for any liability | ||||||
|  |       incurred by, or claims asserted against, such Contributor by reason | ||||||
|  |       of your accepting any such warranty or additional liability. | ||||||
|  | 
 | ||||||
|  |    END OF TERMS AND CONDITIONS | ||||||
|  | 
 | ||||||
|  |    APPENDIX: How to apply the Apache License to your work. | ||||||
|  | 
 | ||||||
|  |       To apply the Apache License to your work, attach the following | ||||||
|  |       boilerplate notice, with the fields enclosed by brackets "[]" | ||||||
|  |       replaced with your own identifying information. (Don't include | ||||||
|  |       the brackets!)  The text should be enclosed in the appropriate | ||||||
|  |       comment syntax for the file format. We also recommend that a | ||||||
|  |       file or class name and description of purpose be included on the | ||||||
|  |       same "printed page" as the copyright notice for easier | ||||||
|  |       identification within third-party archives. | ||||||
|  | 
 | ||||||
|  |    Copyright [yyyy] [name of copyright owner] | ||||||
|  | 
 | ||||||
|  |    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. | ||||||
|  | |||||||
							
								
								
									
										40
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										40
									
								
								README.md
									
									
									
									
									
								
							| @ -12,13 +12,12 @@ | |||||||
| [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | ||||||
| [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | ||||||
| [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | ||||||
| [](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | [](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| [](https://www.bestpractices.dev/projects/8045) | [](https://www.bestpractices.dev/projects/8045) | ||||||
| [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | ||||||
| [](https://golang.org/) | [](https://golang.org/) | ||||||
| [](https://gurubase.io/g/openim) | [](https://gurubase.io/g/openim) | ||||||
| 
 | 
 | ||||||
|       |  | ||||||
| <p align="center"> | <p align="center"> | ||||||
|   <a href="./README.md">English</a> ·  |   <a href="./README.md">English</a> ·  | ||||||
|   <a href="./README_zh_CN.md">中文</a> ·  |   <a href="./README_zh_CN.md">中文</a> ·  | ||||||
| @ -47,16 +46,15 @@ | |||||||
|   <a href="./docs/readme/README_tr.md">Türkçe</a> |   <a href="./docs/readme/README_tr.md">Türkçe</a> | ||||||
| </p> | </p> | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| </div> | </div> | ||||||
| 
 | 
 | ||||||
| </p> | </p> | ||||||
| 
 | 
 | ||||||
| ## :busts_in_silhouette: Join Our Community | ## :busts_in_silhouette: Join Our Community | ||||||
| 
 | 
 | ||||||
| + 💬 [Follow us on Twitter](https://twitter.com/founder_im63606) | - 💬 [Follow us on Twitter](https://twitter.com/founder_im63606) | ||||||
| + 🚀 [Join our Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | - 🚀 [Join our Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| + :eyes: [Join our WeChat Group](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | - :eyes: [Join our WeChat Group](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | ||||||
| 
 | 
 | ||||||
| ## Ⓜ️ About OpenIM | ## Ⓜ️ About OpenIM | ||||||
| 
 | 
 | ||||||
| @ -68,13 +66,14 @@ Unlike standalone chat applications such as Telegram, Signal, and Rocket.Chat, O | |||||||
| 
 | 
 | ||||||
| **OpenIMSDK**, designed for **OpenIMServer**, is an IM SDK created specifically for integration into client applications. It supports various functionalities and modules: | **OpenIMSDK**, designed for **OpenIMServer**, is an IM SDK created specifically for integration into client applications. It supports various functionalities and modules: | ||||||
| 
 | 
 | ||||||
| + 🌟 Main Features: | - 🌟 Main Features: | ||||||
|  | 
 | ||||||
|   - 📦 Local Storage |   - 📦 Local Storage | ||||||
|   - 🔔 Listener Callbacks |   - 🔔 Listener Callbacks | ||||||
|   - 🛡️ API Wrapping |   - 🛡️ API Wrapping | ||||||
|   - 🌐 Connection Management |   - 🌐 Connection Management | ||||||
| 
 | 
 | ||||||
| + 📚 Main Modules: | - 📚 Main Modules: | ||||||
|   1. 🚀 Initialization and Login |   1. 🚀 Initialization and Login | ||||||
|   2. 👤 User Management |   2. 👤 User Management | ||||||
|   3. 👫 Friends Management |   3. 👫 Friends Management | ||||||
| @ -87,16 +86,16 @@ Built with Golang and supports cross-platform deployment to ensure a consistent | |||||||
| 
 | 
 | ||||||
| ## 🌐 Introduction to OpenIMServer | ## 🌐 Introduction to OpenIMServer | ||||||
| 
 | 
 | ||||||
| + **OpenIMServer** features include: | - **OpenIMServer** features include: | ||||||
|   - 🌐 Microservices Architecture: Supports cluster mode, including a gateway and multiple rpc services. |   - 🌐 Microservices Architecture: Supports cluster mode, including a gateway and multiple rpc services. | ||||||
|   - 🚀 Diverse Deployment Options: Supports source code, Kubernetes, or Docker deployment. |   - 🚀 Diverse Deployment Options: Supports source code, Kubernetes, or Docker deployment. | ||||||
|   - Massive User Support: Supports large-scale groups with hundreds of thousands, millions of users, and billions of messages. |   - Massive User Support: Supports large-scale groups with hundreds of thousands, millions of users, and billions of messages. | ||||||
| 
 | 
 | ||||||
| ### Enhanced Business Functions: | ### Enhanced Business Functions: | ||||||
| 
 | 
 | ||||||
| + **REST API**: Provides a REST API for business systems to enhance functionality, such as group creation and message pushing through backend interfaces. | - **REST API**: Provides a REST API for business systems to enhance functionality, such as group creation and message pushing through backend interfaces. | ||||||
| 
 | 
 | ||||||
| + **Webhooks**: Expands business forms through callbacks, sending requests to business servers before or after certain events. | - **Webhooks**: Expands business forms through callbacks, sending requests to business servers before or after certain events. | ||||||
| 
 | 
 | ||||||
|    |    | ||||||
| 
 | 
 | ||||||
| @ -108,8 +107,8 @@ Experience online for iOS/Android/H5/PC/Web: | |||||||
| 
 | 
 | ||||||
| To facilitate user experience, we offer various deployment solutions. You can choose your preferred deployment method from the list below: | To facilitate user experience, we offer various deployment solutions. You can choose your preferred deployment method from the list below: | ||||||
| 
 | 
 | ||||||
| + **[Source Code Deployment Guide](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | - **[Source Code Deployment Guide](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | ||||||
| + **[Docker Deployment Guide](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | - **[Docker Deployment Guide](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | ||||||
| 
 | 
 | ||||||
| ## System Support | ## System Support | ||||||
| 
 | 
 | ||||||
| @ -117,25 +116,22 @@ Supports Linux, Windows, Mac systems, and ARM and AMD CPU architectures. | |||||||
| 
 | 
 | ||||||
| ## :link: Links | ## :link: Links | ||||||
| 
 | 
 | ||||||
|   + **[Developer Manual](https://docs.openim.io/)** | - **[Developer Manual](https://docs.openim.io/)** | ||||||
|   + **[Changelog](https://github.com/openimsdk/open-im-server/blob/main/CHANGELOG.md)** | - **[Changelog](https://github.com/openimsdk/open-im-server/blob/main/CHANGELOG.md)** | ||||||
| 
 | 
 | ||||||
| ## :writing_hand: How to Contribute | ## :writing_hand: How to Contribute | ||||||
| 
 | 
 | ||||||
| We welcome contributions of any kind! Please make sure to read our [Contributor Documentation](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md) before submitting a Pull Request. | We welcome contributions of any kind! Please make sure to read our [Contributor Documentation](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md) before submitting a Pull Request. | ||||||
| 
 | 
 | ||||||
|   + **[Report a Bug](https://github.com/openimsdk/open-im-server/issues/new?assignees=&labels=bug&template=bug_report.md&title=)** | - **[Report a Bug](https://github.com/openimsdk/open-im-server/issues/new?assignees=&labels=bug&template=bug_report.md&title=)** | ||||||
|   + **[Suggest a Feature](https://github.com/openimsdk/open-im-server/issues/new?assignees=&labels=enhancement&template=feature_request.md&title=)** | - **[Suggest a Feature](https://github.com/openimsdk/open-im-server/issues/new?assignees=&labels=enhancement&template=feature_request.md&title=)** | ||||||
|   + **[Submit a Pull Request](https://github.com/openimsdk/open-im-server/pulls)** | - **[Submit a Pull Request](https://github.com/openimsdk/open-im-server/pulls)** | ||||||
| 
 | 
 | ||||||
| Thank you for contributing to building a powerful instant messaging solution! | Thank you for contributing to building a powerful instant messaging solution! | ||||||
| 
 | 
 | ||||||
| ## :closed_book: License | ## :closed_book: License | ||||||
| 
 | 
 | ||||||
| For more details, please refer to [here](./LICENSE). | This software is licensed under the Apache License 2.0 | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| 
 | 
 | ||||||
| ## 🔮 Thanks to our contributors! | ## 🔮 Thanks to our contributors! | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -12,12 +12,11 @@ | |||||||
| [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | ||||||
| [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | ||||||
| [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | ||||||
| [](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | [](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| [](https://www.bestpractices.dev/projects/8045) | [](https://www.bestpractices.dev/projects/8045) | ||||||
| [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | ||||||
| [](https://golang.org/) | [](https://golang.org/) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| <p align="center"> | <p align="center"> | ||||||
|   <a href="./README.md">English</a> ·  |   <a href="./README.md">English</a> ·  | ||||||
|   <a href="./README_zh_CN.md">中文</a> ·  |   <a href="./README_zh_CN.md">中文</a> ·  | ||||||
| @ -46,34 +45,34 @@ | |||||||
|   <a href="./docs/readme/README_tr.md">Türkçe</a> |   <a href="./docs/readme/README_tr.md">Türkçe</a> | ||||||
| </p> | </p> | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| </div> | </div> | ||||||
| 
 | 
 | ||||||
| </p> | </p> | ||||||
| 
 | 
 | ||||||
| ## :busts_in_silhouette: 加入我们的社区 | ## :busts_in_silhouette: 加入我们的社区 | ||||||
| 
 | 
 | ||||||
| + 💬 [关注我们的 Twitter](https://twitter.com/founder_im63606) | - 💬 [关注我们的 Twitter](https://twitter.com/founder_im63606) | ||||||
| + 🚀 [加入我们的 Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2hljfom5u-9ZuzP3NfEKW~BJKbpLm0Hw) | - 🚀 [加入我们的 Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2hljfom5u-9ZuzP3NfEKW~BJKbpLm0Hw) | ||||||
| + :eyes: [加入我们的微信群](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | - :eyes: [加入我们的微信群](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | ||||||
| 
 | 
 | ||||||
| ## Ⓜ️ 关于 OpenIM | ## Ⓜ️ 关于 OpenIM | ||||||
| 
 | 
 | ||||||
| 与Telegram、Signal、Rocket.Chat等独立聊天应用不同,OpenIM提供了专为开发者设计的开源即时通讯解决方案,而不是直接安装使用的独立聊天应用。OpenIM由OpenIM SDK和OpenIM Server两大部分组成,为开发者提供了一整套集成即时通讯功能的工具和服务,包括消息发送接收、用户管理和群组管理等。总体来说,OpenIM旨在为开发者提供必要的工具和框架,帮助他们在自己的应用中实现高效的即时通讯解决方案。 | 与 Telegram、Signal、Rocket.Chat 等独立聊天应用不同,OpenIM 提供了专为开发者设计的开源即时通讯解决方案,而不是直接安装使用的独立聊天应用。OpenIM 由 OpenIM SDK 和 OpenIM Server 两大部分组成,为开发者提供了一整套集成即时通讯功能的工具和服务,包括消息发送接收、用户管理和群组管理等。总体来说,OpenIM 旨在为开发者提供必要的工具和框架,帮助他们在自己的应用中实现高效的即时通讯解决方案。 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
| 
 | 
 | ||||||
| ## 🚀 OpenIMSDK 介绍 | ## 🚀 OpenIMSDK 介绍 | ||||||
| 
 | 
 | ||||||
| **OpenIMSDK** 是为 **OpenIMServer** 设计的IM SDK,专为集成到客户端应用而生。它支持多种功能和模块: | **OpenIMSDK** 是为 **OpenIMServer** 设计的 IM SDK,专为集成到客户端应用而生。它支持多种功能和模块: | ||||||
|  | 
 | ||||||
|  | - 🌟 主要功能: | ||||||
| 
 | 
 | ||||||
| + 🌟 主要功能: |  | ||||||
|   - 📦 本地存储 |   - 📦 本地存储 | ||||||
|   - 🔔 监听器回调 |   - 🔔 监听器回调 | ||||||
|   - 🛡️ API封装 |   - 🛡️ API 封装 | ||||||
|   - 🌐 连接管理 |   - 🌐 连接管理 | ||||||
| 
 | 
 | ||||||
| + 📚 主要模块: | - 📚 主要模块: | ||||||
|   1. 🚀 初始化及登录 |   1. 🚀 初始化及登录 | ||||||
|   2. 👤 用户管理 |   2. 👤 用户管理 | ||||||
|   3. 👫 好友管理 |   3. 👫 好友管理 | ||||||
| @ -86,31 +85,29 @@ | |||||||
| 
 | 
 | ||||||
| ## 🌐 OpenIMServer 介绍 | ## 🌐 OpenIMServer 介绍 | ||||||
| 
 | 
 | ||||||
| + **OpenIMServer** 的特点包括: | - **OpenIMServer** 的特点包括: | ||||||
|   - 🌐 微服务架构:支持集群模式,包括网关(gateway)和多个rpc服务。 |   - 🌐 微服务架构:支持集群模式,包括网关(gateway)和多个 rpc 服务。 | ||||||
|   - 🚀 多样的部署方式:支持源代码、Kubernetes或Docker部署。 |   - 🚀 多样的部署方式:支持源代码、Kubernetes 或 Docker 部署。 | ||||||
|   - 海量用户支持:支持十万级超大群组,千万级用户和百亿级消息。 |   - 海量用户支持:支持十万级超大群组,千万级用户和百亿级消息。 | ||||||
| 
 | 
 | ||||||
| ### 增强的业务功能: | ### 增强的业务功能: | ||||||
| 
 | 
 | ||||||
| + **REST API**:为业务系统提供REST API,增加群组创建、消息推送等后台接口功能。 | - **REST API**:为业务系统提供 REST API,增加群组创建、消息推送等后台接口功能。 | ||||||
| 
 | 
 | ||||||
| + **Webhooks**:通过事件前后的回调,向业务服务器发送请求,扩展更多的业务形态。 | - **Webhooks**:通过事件前后的回调,向业务服务器发送请求,扩展更多的业务形态。 | ||||||
| 
 | 
 | ||||||
|    |    | ||||||
| 
 | 
 | ||||||
|    |  | ||||||
| 
 |  | ||||||
| ## :rocket: 快速入门 | ## :rocket: 快速入门 | ||||||
| 
 | 
 | ||||||
| 在线体验iOS/Android/H5/PC/Web: | 在线体验 iOS/Android/H5/PC/Web: | ||||||
| 
 | 
 | ||||||
| 👉 **[OpenIM在线演示](https://www.openim.io/en/commercial)** | 👉 **[OpenIM 在线演示](https://www.openim.io/en/commercial)** | ||||||
| 
 | 
 | ||||||
| 为了便于用户体验,我们提供了多种部署解决方案,您可以根据以下列表选择适合您的部署方式: | 为了便于用户体验,我们提供了多种部署解决方案,您可以根据以下列表选择适合您的部署方式: | ||||||
| 
 | 
 | ||||||
| + **[源代码部署指南](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | - **[源代码部署指南](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | ||||||
| + **[Docker 部署指南](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | - **[Docker 部署指南](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | ||||||
| 
 | 
 | ||||||
| ## 系统支持 | ## 系统支持 | ||||||
| 
 | 
 | ||||||
| @ -118,22 +115,22 @@ | |||||||
| 
 | 
 | ||||||
| ## :link: 相关链接 | ## :link: 相关链接 | ||||||
| 
 | 
 | ||||||
|   + **[开发手册](https://docs.openim.io/)** | - **[开发手册](https://docs.openim.io/)** | ||||||
|   + **[更新日志](https://github.com/openimsdk/open-im-server/blob/main/CHANGELOG.md)** | - **[更新日志](https://github.com/openimsdk/open-im-server/blob/main/CHANGELOG.md)** | ||||||
| 
 | 
 | ||||||
| ## :writing_hand: 如何贡献 | ## :writing_hand: 如何贡献 | ||||||
| 
 | 
 | ||||||
| 我们欢迎任何形式的贡献!在提交 Pull Request 之前,请确保阅读我们的[贡献者文档](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md) | 我们欢迎任何形式的贡献!在提交 Pull Request 之前,请确保阅读我们的[贡献者文档](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md) | ||||||
| 
 | 
 | ||||||
|   + **[报告 Bug](https://github.com/openimsdk/open-im-server/issues/new?assignees=&labels=bug&template=bug_report.md&title=)** | - **[报告 Bug](https://github.com/openimsdk/open-im-server/issues/new?assignees=&labels=bug&template=bug_report.md&title=)** | ||||||
|   + **[提出新特性](https://github.com/openimsdk/open-im-server/issues/new?assignees=&labels=enhancement&template=feature_request.md&title=)** | - **[提出新特性](https://github.com/openimsdk/open-im-server/issues/new?assignees=&labels=enhancement&template=feature_request.md&title=)** | ||||||
|   + **[提交 Pull Request](https://github.com/openimsdk/open-im-server/pulls)** | - **[提交 Pull Request](https://github.com/openimsdk/open-im-server/pulls)** | ||||||
| 
 | 
 | ||||||
| 感谢您的贡献,一起来打造强大的即时通讯解决方案! | 感谢您的贡献,一起来打造强大的即时通讯解决方案! | ||||||
| 
 | 
 | ||||||
| ## :closed_book: 许可证 | ## :closed_book: 开源许可证 License | ||||||
| 
 | 
 | ||||||
|   OpenIMSDK 在 Apache License 2.0 许可下可用。查看[LICENSE 文件](https://github.com/openimsdk/open-im-server/blob/main/LICENSE)了解更多信息。 | This software is licensed under the Apache License 2.0 | ||||||
| 
 | 
 | ||||||
| ## 🔮 Thanks to our contributors! | ## 🔮 Thanks to our contributors! | ||||||
| 
 | 
 | ||||||
|  | |||||||
							
								
								
									
										17
									
								
								cmd/main.go
									
									
									
									
									
								
							
							
						
						
									
										17
									
								
								cmd/main.go
									
									
									
									
									
								
							| @ -3,7 +3,6 @@ package main | |||||||
| import ( | import ( | ||||||
| 	"bytes" | 	"bytes" | ||||||
| 	"context" | 	"context" | ||||||
| 	"encoding/json" |  | ||||||
| 	"flag" | 	"flag" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"net" | 	"net" | ||||||
| @ -39,7 +38,6 @@ import ( | |||||||
| 	"github.com/openimsdk/tools/log" | 	"github.com/openimsdk/tools/log" | ||||||
| 	"github.com/openimsdk/tools/system/program" | 	"github.com/openimsdk/tools/system/program" | ||||||
| 	"github.com/openimsdk/tools/utils/datautil" | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
| 	"github.com/openimsdk/tools/utils/network" |  | ||||||
| 	"github.com/spf13/viper" | 	"github.com/spf13/viper" | ||||||
| 	"google.golang.org/grpc" | 	"google.golang.org/grpc" | ||||||
| ) | ) | ||||||
| @ -250,23 +248,12 @@ func (x *cmds) run(ctx context.Context) error { | |||||||
| 				return err | 				return err | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		ip, err := network.GetLocalIP() |  | ||||||
| 		if err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 		listener, err := net.Listen("tcp", fmt.Sprintf(":%d", port)) | 		listener, err := net.Listen("tcp", fmt.Sprintf(":%d", port)) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return fmt.Errorf("prometheus listen %d error %w", port, err) | 			return fmt.Errorf("prometheus listen %d error %w", port, err) | ||||||
| 		} | 		} | ||||||
| 		defer listener.Close() | 		defer listener.Close() | ||||||
| 		log.ZDebug(ctx, "prometheus start", "addr", listener.Addr()) | 		log.ZDebug(ctx, "prometheus start", "addr", listener.Addr()) | ||||||
| 		target, err := json.Marshal(prommetrics.BuildDefaultTarget(ip, listener.Addr().(*net.TCPAddr).Port)) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 		if err := standalone.GetKeyValue().SetKey(ctx, prommetrics.BuildDiscoveryKey(prommetrics.APIKeyName), target); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 		go func() { | 		go func() { | ||||||
| 			err := prommetrics.Start(listener) | 			err := prommetrics.Start(listener) | ||||||
| 			if err == nil { | 			if err == nil { | ||||||
| @ -342,7 +329,7 @@ func (x *cmds) run(ctx context.Context) error { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func putCmd[C any](cmd *cmds, block bool, fn func(ctx context.Context, config *C, client discovery.Conn, server grpc.ServiceRegistrar) error) { | func putCmd[C any](cmd *cmds, block bool, fn func(ctx context.Context, config *C, client discovery.SvcDiscoveryRegistry, server grpc.ServiceRegistrar) error) { | ||||||
| 	name := path.Base(runtime.FuncForPC(reflect.ValueOf(fn).Pointer()).Name()) | 	name := path.Base(runtime.FuncForPC(reflect.ValueOf(fn).Pointer()).Name()) | ||||||
| 	if index := strings.Index(name, "."); index >= 0 { | 	if index := strings.Index(name, "."); index >= 0 { | ||||||
| 		name = name[:index] | 		name = name[:index] | ||||||
| @ -352,7 +339,7 @@ func putCmd[C any](cmd *cmds, block bool, fn func(ctx context.Context, config *C | |||||||
| 		if err := cmd.parseConf(&conf); err != nil { | 		if err := cmd.parseConf(&conf); err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| 		return fn(ctx, &conf, standalone.GetDiscoveryConn(), standalone.GetServiceRegistrar()) | 		return fn(ctx, &conf, standalone.GetSvcDiscoveryRegistry(), standalone.GetServiceRegistrar()) | ||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -1,9 +1,11 @@ | |||||||
| enable: etcd | enable: etcd | ||||||
| etcd: | etcd: | ||||||
|   rootDirectory: openim |   rootDirectory: openim | ||||||
|   address: [ localhost:12379 ] |   address: [localhost:12379] | ||||||
|   username: '' |   ## Attention: If you set auth in etcd | ||||||
|   password: '' |   ## you must also update the username and password in Chat project. | ||||||
|  |   username: | ||||||
|  |   password: | ||||||
| 
 | 
 | ||||||
| kubernetes: | kubernetes: | ||||||
|   namespace: default |   namespace: default | ||||||
|  | |||||||
| @ -1,13 +1,13 @@ | |||||||
| # Username for authentication | ## Kafka authentication | ||||||
| username: '' | username: | ||||||
| # Password for authentication | password: | ||||||
| password: '' | 
 | ||||||
| # Producer acknowledgment settings | # Producer acknowledgment settings | ||||||
| producerAck: | producerAck: | ||||||
| # Compression type to use (e.g., none, gzip, snappy) | # Compression type to use (e.g., none, gzip, snappy) | ||||||
| compressType: none | compressType: none | ||||||
| # List of Kafka broker addresses | # List of Kafka broker addresses | ||||||
| address: [ localhost:19094 ] | address: [localhost:19094] | ||||||
| # Kafka topic for Redis integration | # Kafka topic for Redis integration | ||||||
| toRedisTopic: toRedis | toRedisTopic: toRedis | ||||||
| # Kafka topic for MongoDB integration | # Kafka topic for MongoDB integration | ||||||
|  | |||||||
| @ -1,7 +1,14 @@ | |||||||
| address: [ localhost:16379 ] | address: [localhost:16379] | ||||||
| username: | username: | ||||||
| password: openIM123 | password: openIM123 | ||||||
| clusterMode: false | # redis Mode, including "standalone","cluster","sentinel" | ||||||
|  | redisMode: "standalone" | ||||||
| db: 0 | db: 0 | ||||||
| maxRetry: 10 | maxRetry: 10 | ||||||
| poolSize: 100 | poolSize: 100 | ||||||
|  | # Sentinel configuration (only used when redisMode is "sentinel") | ||||||
|  | sentinelMode: | ||||||
|  |   masterName: "redis-master" | ||||||
|  |   sentinelsAddrs: ["127.0.0.1:26379", "127.0.0.1:26380", "127.0.0.1:26381"] | ||||||
|  |   routeByLatency: true | ||||||
|  |   routeRandomly: true | ||||||
|  | |||||||
| @ -1,9 +1,20 @@ | |||||||
| secret: openIM123 | secret: openIM123 | ||||||
| 
 | 
 | ||||||
| imAdminUserID: [ imAdmin ] | # imAdminUser: Configuration for instant messaging system administrators | ||||||
|  | imAdminUser: | ||||||
|  |   # userIDs: List of administrator user IDs. | ||||||
|  |   # Each entry here corresponds by index to the matching entry in the nicknames list below. | ||||||
|  |   userIDs: [imAdmin] | ||||||
|  |   # nicknames: List of administrator display names. | ||||||
|  |   # Each entry here corresponds by index to the matching entry in the userIDs list above. | ||||||
|  |   nicknames: [superAdmin] | ||||||
| 
 | 
 | ||||||
| # 1: For Android, iOS, Windows, Mac, and web platforms, only one instance can be online at a time | # 1: For Android, iOS, Windows, Mac, and web platforms, only one instance can be online at a time | ||||||
| multiLogin: | multiLogin: | ||||||
|   policy: 1 |   policy: 1 | ||||||
|   # max num of tokens in one end |   # max num of tokens in one end | ||||||
|   maxNumOneEnd: 30 |   maxNumOneEnd: 30 | ||||||
|  | 
 | ||||||
|  | rpcMaxBodySize: | ||||||
|  |   requestMaxBodySize:  8388608 | ||||||
|  |   responseMaxBodySize: 8388608 | ||||||
|  | |||||||
| @ -7,16 +7,16 @@ beforeSendSingleMsg: | |||||||
|   # If not set, all contentType messages will through this filter. |   # If not set, all contentType messages will through this filter. | ||||||
|   deniedTypes: [] |   deniedTypes: [] | ||||||
| beforeUpdateUserInfoEx: | beforeUpdateUserInfoEx: | ||||||
|   enable:  false |   enable: false | ||||||
|   timeout: 5 |   timeout: 5 | ||||||
|   failedContinue: true |   failedContinue: true | ||||||
| afterUpdateUserInfoEx: | afterUpdateUserInfoEx: | ||||||
|   enable:  false |   enable: false | ||||||
|   timeout: 5 |   timeout: 5 | ||||||
| afterSendSingleMsg: | afterSendSingleMsg: | ||||||
|   enable: false |   enable: false | ||||||
|   timeout: 5 |   timeout: 5 | ||||||
|   # Only the recvID specified in attentionIds will send the callback |   # Only the recvIDs 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. |   # See beforeSendSingleMsg comment. | ||||||
| @ -36,7 +36,7 @@ beforeMsgModify: | |||||||
| afterSendGroupMsg: | afterSendGroupMsg: | ||||||
|   enable: false |   enable: false | ||||||
|   timeout: 5 |   timeout: 5 | ||||||
|   # Only the recvID specified in attentionIds will send the callback |   # Only the GroupIDs 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. |   # See beforeSendSingleMsg comment. | ||||||
| @ -181,3 +181,19 @@ afterImportFriends: | |||||||
| afterRemoveBlack: | afterRemoveBlack: | ||||||
|   enable: false |   enable: false | ||||||
|   timeout: 5 |   timeout: 5 | ||||||
|  | beforeCreateSingleChatConversations: | ||||||
|  |   enable: false | ||||||
|  |   timeout: 5 | ||||||
|  |   failedContinue: false | ||||||
|  | afterCreateSingleChatConversations: | ||||||
|  |   enable: false | ||||||
|  |   timeout: 5 | ||||||
|  |   failedContinue: false | ||||||
|  | beforeCreateGroupChatConversations: | ||||||
|  |   enable: false | ||||||
|  |   timeout: 5 | ||||||
|  |   failedContinue: false | ||||||
|  | afterCreateGroupChatConversations: | ||||||
|  |   enable: false | ||||||
|  |   timeout: 5 | ||||||
|  |   failedContinue: false | ||||||
|  | |||||||
| @ -83,8 +83,83 @@ services: | |||||||
|       - ETCD_INITIAL_CLUSTER=s1=http://0.0.0.0:2380 |       - ETCD_INITIAL_CLUSTER=s1=http://0.0.0.0:2380 | ||||||
|       - ETCD_INITIAL_CLUSTER_TOKEN=tkn |       - ETCD_INITIAL_CLUSTER_TOKEN=tkn | ||||||
|       - ETCD_INITIAL_CLUSTER_STATE=new |       - ETCD_INITIAL_CLUSTER_STATE=new | ||||||
|  |       - ALLOW_NONE_AUTHENTICATION=no | ||||||
|  | 
 | ||||||
|  |       ## Optional: Enable etcd authentication by setting the following credentials | ||||||
|  |       # - ETCD_ROOT_USER=root | ||||||
|  |       # - ETCD_ROOT_PASSWORD=openIM123 | ||||||
|  |       # - ETCD_USERNAME=openIM | ||||||
|  |       # - ETCD_PASSWORD=openIM123 | ||||||
|     volumes: |     volumes: | ||||||
|       - "${DATA_DIR}/components/etcd:/etcd-data" |       - "${DATA_DIR}/components/etcd:/etcd-data" | ||||||
|  |     command: > | ||||||
|  |       /bin/sh -c ' | ||||||
|  |         etcd & | ||||||
|  |         export ETCDCTL_API=3 | ||||||
|  |         echo "Waiting for etcd to become healthy..." | ||||||
|  |         until etcdctl --endpoints=http://127.0.0.1:2379 endpoint health &>/dev/null; do | ||||||
|  |           echo "Waiting for ETCD to start..." | ||||||
|  |           sleep 1 | ||||||
|  |         done | ||||||
|  | 
 | ||||||
|  |         echo "etcd is healthy." | ||||||
|  | 
 | ||||||
|  |         if [ -n "$${ETCD_ROOT_USER}" ] && [ -n "$${ETCD_ROOT_PASSWORD}" ] && [ -n "$${ETCD_USERNAME}" ] && [ -n "$${ETCD_PASSWORD}" ]; then | ||||||
|  |           echo "Authentication credentials provided. Setting up authentication..." | ||||||
|  | 
 | ||||||
|  |         echo "Checking authentication status..." | ||||||
|  |         if ! etcdctl --endpoints=http://127.0.0.1:2379 auth status | grep -q "Authentication Status: true"; then | ||||||
|  |           echo "Authentication is disabled. Creating users and enabling..." | ||||||
|  |            | ||||||
|  |           # Create users and setup permissions | ||||||
|  |           etcdctl --endpoints=http://127.0.0.1:2379 user add $${ETCD_ROOT_USER} --new-user-password=$${ETCD_ROOT_PASSWORD} || true | ||||||
|  |           etcdctl --endpoints=http://127.0.0.1:2379 user add $${ETCD_USERNAME} --new-user-password=$${ETCD_PASSWORD} || true | ||||||
|  |            | ||||||
|  |           etcdctl --endpoints=http://127.0.0.1:2379 role add openim-role || true | ||||||
|  |           etcdctl --endpoints=http://127.0.0.1:2379 role grant-permission openim-role --prefix=true readwrite / || true | ||||||
|  |           etcdctl --endpoints=http://127.0.0.1:2379 role grant-permission openim-role --prefix=true readwrite "" || true | ||||||
|  |           etcdctl --endpoints=http://127.0.0.1:2379 user grant-role $${ETCD_USERNAME} openim-role || true | ||||||
|  |            | ||||||
|  |           etcdctl --endpoints=http://127.0.0.1:2379 user grant-role $${ETCD_ROOT_USER} $${ETCD_USERNAME} root || true | ||||||
|  |            | ||||||
|  |           echo "Enabling authentication..." | ||||||
|  |           etcdctl --endpoints=http://127.0.0.1:2379 auth enable | ||||||
|  |           echo "Authentication enabled successfully" | ||||||
|  |         else | ||||||
|  |           echo "Authentication is already enabled. Checking OpenIM user..." | ||||||
|  |            | ||||||
|  |           # Check if openIM user exists and can perform operations | ||||||
|  |           if ! etcdctl --endpoints=http://127.0.0.1:2379 --user=$${ETCD_USERNAME}:$${ETCD_PASSWORD} put /test/auth "auth-check" &>/dev/null; then | ||||||
|  |             echo "OpenIM user test failed. Recreating user with root credentials..." | ||||||
|  |              | ||||||
|  |             # Try to create/update the openIM user using root credentials | ||||||
|  |             etcdctl --endpoints=http://127.0.0.1:2379 --user=$${ETCD_ROOT_USER}:$${ETCD_ROOT_PASSWORD} user add $${ETCD_USERNAME} --new-user-password=$${ETCD_PASSWORD} --no-password-file || true | ||||||
|  |             etcdctl --endpoints=http://127.0.0.1:2379 --user=$${ETCD_ROOT_USER}:$${ETCD_ROOT_PASSWORD} role add openim-role || true | ||||||
|  |             etcdctl --endpoints=http://127.0.0.1:2379 --user=$${ETCD_ROOT_USER}:$${ETCD_ROOT_PASSWORD} role grant-permission openim-role --prefix=true readwrite / || true | ||||||
|  |             etcdctl --endpoints=http://127.0.0.1:2379 --user=$${ETCD_ROOT_USER}:$${ETCD_ROOT_PASSWORD} role grant-permission openim-role --prefix=true readwrite "" || true | ||||||
|  |             etcdctl --endpoints=http://127.0.0.1:2379 --user=$${ETCD_ROOT_USER}:$${ETCD_ROOT_PASSWORD} user grant-role $${ETCD_USERNAME} openim-role || true | ||||||
|  |             etcdctl --endpoints=http://127.0.0.1:2379 user grant-role $${ETCD_ROOT_USER} $${ETCD_USERNAME} root || true | ||||||
|  |              | ||||||
|  |             echo "OpenIM user recreated with required permissions" | ||||||
|  |           else | ||||||
|  |             echo "OpenIM user exists and has correct permissions" | ||||||
|  |             etcdctl --endpoints=http://127.0.0.1:2379 --user=$${ETCD_USERNAME}:$${ETCD_PASSWORD} del /test/auth &>/dev/null | ||||||
|  |           fi | ||||||
|  |         fi | ||||||
|  |         echo "Testing authentication with OpenIM user..." | ||||||
|  |         if etcdctl --endpoints=http://127.0.0.1:2379 --user=$${ETCD_USERNAME}:$${ETCD_PASSWORD} put /test/auth "auth-works"; then | ||||||
|  |           echo "Authentication working properly" | ||||||
|  |           etcdctl --endpoints=http://127.0.0.1:2379 --user=$${ETCD_USERNAME}:$${ETCD_PASSWORD} del /test/auth | ||||||
|  |         else | ||||||
|  |           echo "WARNING: Authentication test failed" | ||||||
|  |           fi | ||||||
|  |         else | ||||||
|  |           echo "No authentication credentials provided. Running in no-auth mode." | ||||||
|  |           echo "To enable authentication, set ETCD_ROOT_USER, ETCD_ROOT_PASSWORD, ETCD_USERNAME, and ETCD_PASSWORD environment variables." | ||||||
|  |         fi | ||||||
|  |          | ||||||
|  |         tail -f /dev/null | ||||||
|  |       ' | ||||||
|     restart: always |     restart: always | ||||||
|     networks: |     networks: | ||||||
|       - openim |       - openim | ||||||
| @ -104,12 +179,38 @@ services: | |||||||
|       KAFKA_CFG_NODE_ID: 0 |       KAFKA_CFG_NODE_ID: 0 | ||||||
|       KAFKA_CFG_PROCESS_ROLES: controller,broker |       KAFKA_CFG_PROCESS_ROLES: controller,broker | ||||||
|       KAFKA_CFG_CONTROLLER_QUORUM_VOTERS: 0@kafka:9093 |       KAFKA_CFG_CONTROLLER_QUORUM_VOTERS: 0@kafka:9093 | ||||||
|       KAFKA_CFG_LISTENERS: PLAINTEXT://:9092,CONTROLLER://:9093,EXTERNAL://:9094 |  | ||||||
|       KAFKA_CFG_ADVERTISED_LISTENERS: PLAINTEXT://kafka:9092,EXTERNAL://localhost:19094 |  | ||||||
|       KAFKA_CFG_LISTENER_SECURITY_PROTOCOL_MAP: CONTROLLER:PLAINTEXT,EXTERNAL:PLAINTEXT,PLAINTEXT:PLAINTEXT |  | ||||||
|       KAFKA_CFG_CONTROLLER_LISTENER_NAMES: CONTROLLER |       KAFKA_CFG_CONTROLLER_LISTENER_NAMES: CONTROLLER | ||||||
|       KAFKA_NUM_PARTITIONS: 8 |       KAFKA_NUM_PARTITIONS: 8 | ||||||
|       KAFKA_CFG_AUTO_CREATE_TOPICS_ENABLE: "true" |       KAFKA_CFG_AUTO_CREATE_TOPICS_ENABLE: "true" | ||||||
|  | 
 | ||||||
|  |       KAFKA_CFG_LISTENERS: "PLAINTEXT://:9092,CONTROLLER://:9093,EXTERNAL://:9094" | ||||||
|  |       KAFKA_CFG_ADVERTISED_LISTENERS: "PLAINTEXT://kafka:9092,EXTERNAL://localhost:19094" | ||||||
|  |       KAFKA_CFG_LISTENER_SECURITY_PROTOCOL_MAP: "CONTROLLER:PLAINTEXT,EXTERNAL:PLAINTEXT,PLAINTEXT:PLAINTEXT" | ||||||
|  |       KAFKA_CFG_INTER_BROKER_LISTENER_NAME: "PLAINTEXT" | ||||||
|  | 
 | ||||||
|  |       # Authentication configuration variables - comment out to disable auth | ||||||
|  |       # KAFKA_USERNAME: "openIM" | ||||||
|  |       # KAFKA_PASSWORD: "openIM123" | ||||||
|  |     command: > | ||||||
|  |       /bin/sh -c ' | ||||||
|  |         if [ -n "$${KAFKA_USERNAME}" ] && [ -n "$${KAFKA_PASSWORD}" ]; then | ||||||
|  |           echo "=== Kafka SASL Authentication ENABLED ===" | ||||||
|  |           echo "Username: $${KAFKA_USERNAME}" | ||||||
|  |            | ||||||
|  |           # Set environment variables for SASL authentication | ||||||
|  |           export KAFKA_CFG_LISTENERS="SASL_PLAINTEXT://:9092,CONTROLLER://:9093,EXTERNAL://:9094" | ||||||
|  |           export KAFKA_CFG_ADVERTISED_LISTENERS="SASL_PLAINTEXT://kafka:9092,EXTERNAL://localhost:19094" | ||||||
|  |           export KAFKA_CFG_LISTENER_SECURITY_PROTOCOL_MAP="CONTROLLER:PLAINTEXT,EXTERNAL:SASL_PLAINTEXT,SASL_PLAINTEXT:SASL_PLAINTEXT" | ||||||
|  |           export KAFKA_CFG_SASL_ENABLED_MECHANISMS="PLAIN" | ||||||
|  |           export KAFKA_CFG_SASL_MECHANISM_INTER_BROKER_PROTOCOL="PLAIN" | ||||||
|  |           export KAFKA_CFG_INTER_BROKER_LISTENER_NAME="SASL_PLAINTEXT" | ||||||
|  |           export KAFKA_CLIENT_USERS="$${KAFKA_USERNAME}" | ||||||
|  |           export KAFKA_CLIENT_PASSWORDS="$${KAFKA_PASSWORD}" | ||||||
|  |         fi | ||||||
|  |          | ||||||
|  |         # Start Kafka with the configured environment | ||||||
|  |         exec /opt/bitnami/scripts/kafka/entrypoint.sh /opt/bitnami/scripts/kafka/run.sh | ||||||
|  |       ' | ||||||
|     networks: |     networks: | ||||||
|       - openim |       - openim | ||||||
| 
 | 
 | ||||||
| @ -161,9 +262,9 @@ services: | |||||||
|       - ./config/instance-down-rules.yml:/etc/prometheus/instance-down-rules.yml |       - ./config/instance-down-rules.yml:/etc/prometheus/instance-down-rules.yml | ||||||
|       - ${DATA_DIR}/components/prometheus/data:/prometheus |       - ${DATA_DIR}/components/prometheus/data:/prometheus | ||||||
|     command: |     command: | ||||||
|       - '--config.file=/etc/prometheus/prometheus.yml' |       - "--config.file=/etc/prometheus/prometheus.yml" | ||||||
|       - '--storage.tsdb.path=/prometheus' |       - "--storage.tsdb.path=/prometheus" | ||||||
|       - '--web.listen-address=:${PROMETHEUS_PORT}' |       - "--web.listen-address=:${PROMETHEUS_PORT}" | ||||||
|     network_mode: host |     network_mode: host | ||||||
| 
 | 
 | ||||||
|   alertmanager: |   alertmanager: | ||||||
| @ -176,8 +277,8 @@ services: | |||||||
|       - ./config/alertmanager.yml:/etc/alertmanager/alertmanager.yml |       - ./config/alertmanager.yml:/etc/alertmanager/alertmanager.yml | ||||||
|       - ./config/email.tmpl:/etc/alertmanager/email.tmpl |       - ./config/email.tmpl:/etc/alertmanager/email.tmpl | ||||||
|     command: |     command: | ||||||
|       - '--config.file=/etc/alertmanager/alertmanager.yml' |       - "--config.file=/etc/alertmanager/alertmanager.yml" | ||||||
|       - '--web.listen-address=:${ALERTMANAGER_PORT}' |       - "--web.listen-address=:${ALERTMANAGER_PORT}" | ||||||
|     network_mode: host |     network_mode: host | ||||||
| 
 | 
 | ||||||
|   grafana: |   grafana: | ||||||
| @ -209,9 +310,8 @@ services: | |||||||
|       - /sys:/host/sys:ro |       - /sys:/host/sys:ro | ||||||
|       - /:/rootfs:ro |       - /:/rootfs:ro | ||||||
|     command: |     command: | ||||||
|       - '--path.procfs=/host/proc' |       - "--path.procfs=/host/proc" | ||||||
|       - '--path.sysfs=/host/sys' |       - "--path.sysfs=/host/sys" | ||||||
|       - '--path.rootfs=/rootfs' |       - "--path.rootfs=/rootfs" | ||||||
|       - '--web.listen-address=:19100' |       - "--web.listen-address=:19100" | ||||||
|     network_mode: host |     network_mode: host | ||||||
| 
 |  | ||||||
|  | |||||||
| @ -12,12 +12,11 @@ | |||||||
| [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | ||||||
| [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | ||||||
| [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | ||||||
| [](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | [](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| [](https://www.bestpractices.dev/projects/8045) | [](https://www.bestpractices.dev/projects/8045) | ||||||
| [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | ||||||
| [](https://golang.org/) | [](https://golang.org/) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| <p align="center"> | <p align="center"> | ||||||
|   <a href="../../README.md">English</a> ·  |   <a href="../../README.md">English</a> ·  | ||||||
|   <a href="../../README_zh_CN.md">中文</a> ·  |   <a href="../../README_zh_CN.md">中文</a> ·  | ||||||
| @ -46,7 +45,6 @@ | |||||||
|   <a href="./README_tr.md">Türkçe</a> |   <a href="./README_tr.md">Türkçe</a> | ||||||
| </p> | </p> | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| </div> | </div> | ||||||
| 
 | 
 | ||||||
| </p> | </p> | ||||||
| @ -61,14 +59,14 @@ OpenIM je platforma služeb speciálně navržená pro integraci chatu, audio-vi | |||||||
| 
 | 
 | ||||||
| **OpenIMSDK** je IM SDK navržený pro**OpenIMServer**, vytvořený speciálně pro vkládání do klientských aplikací. Jeho hlavní vlastnosti a moduly jsou následující: | **OpenIMSDK** je IM SDK navržený pro**OpenIMServer**, vytvořený speciálně pro vkládání do klientských aplikací. Jeho hlavní vlastnosti a moduly jsou následující: | ||||||
| 
 | 
 | ||||||
| + 🌟 Hlavní vlastnosti: | - 🌟 Hlavní vlastnosti: | ||||||
| 
 | 
 | ||||||
|   - 📦 Místní úložiště |   - 📦 Místní úložiště | ||||||
|   - 🔔 Zpětná volání posluchačů |   - 🔔 Zpětná volání posluchačů | ||||||
|   - 🛡️ API obalování |   - 🛡️ API obalování | ||||||
|   - 🌐 Správa připojení |   - 🌐 Správa připojení | ||||||
| 
 | 
 | ||||||
| + 📚 hlavní moduly: | - 📚 hlavní moduly: | ||||||
| 
 | 
 | ||||||
|   1. 🚀 Inicializace a přihlášení |   1. 🚀 Inicializace a přihlášení | ||||||
|   2. 👤 Správa uživatelů |   2. 👤 Správa uživatelů | ||||||
| @ -82,15 +80,15 @@ Je postaven pomocí Golang a podporuje nasazení napříč platformami, což zaj | |||||||
| 
 | 
 | ||||||
| ## 🌐 O OpenIMServeru | ## 🌐 O OpenIMServeru | ||||||
| 
 | 
 | ||||||
| + **OpenIMServer** má následující vlastnosti: | - **OpenIMServer** má následující vlastnosti: | ||||||
|   - 🌐 Architektura mikroslužeb: Podporuje režim clusteru, včetně brány a více služeb RPC. |   - 🌐 Architektura mikroslužeb: Podporuje režim clusteru, včetně brány a více služeb RPC. | ||||||
|   - 🚀 Různé metody nasazení: Podporuje nasazení prostřednictvím zdrojového kódu, Kubernetes nebo Docker. |   - 🚀 Různé metody nasazení: Podporuje nasazení prostřednictvím zdrojového kódu, Kubernetes nebo Docker. | ||||||
|   - Podpora masivní uživatelské základny: Super velké skupiny se stovkami tisíc uživatelů, desítkami milionů uživatelů a miliardami zpráv. |   - Podpora masivní uživatelské základny: Super velké skupiny se stovkami tisíc uživatelů, desítkami milionů uživatelů a miliardami zpráv. | ||||||
| 
 | 
 | ||||||
| ### Vylepšené obchodní funkce: | ### Vylepšené obchodní funkce: | ||||||
| 
 | 
 | ||||||
| + **REST API**: OpenIMServer nabízí REST API pro podnikové systémy, jejichž cílem je poskytnout podnikům více funkcí, jako je vytváření skupin a odesílání push zpráv přes backendová rozhraní. | - **REST API**: OpenIMServer nabízí REST API pro podnikové systémy, jejichž cílem je poskytnout podnikům více funkcí, jako je vytváření skupin a odesílání push zpráv přes backendová rozhraní. | ||||||
| + **Webhooks**: OpenIMServer poskytuje možnosti zpětného volání pro rozšíření více obchodních formulářů. Zpětné volání znamená, že OpenIMServer odešle požadavek na obchodní server před nebo po určité události, jako jsou zpětná volání před nebo po odeslání zprávy. | - **Webhooks**: OpenIMServer poskytuje možnosti zpětného volání pro rozšíření více obchodních formulářů. Zpětné volání znamená, že OpenIMServer odešle požadavek na obchodní server před nebo po určité události, jako jsou zpětná volání před nebo po odeslání zprávy. | ||||||
| 
 | 
 | ||||||
| 👉 **[Další informace](https://docs.openim.io/guides/introduction/product)** | 👉 **[Další informace](https://docs.openim.io/guides/introduction/product)** | ||||||
| 
 | 
 | ||||||
| @ -100,7 +98,6 @@ Ponořte se do srdce funkčnosti Open-IM-Server s naším diagramem architektury | |||||||
| 
 | 
 | ||||||
|  |  | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## :rocket: Rychlý start | ## :rocket: Rychlý start | ||||||
| 
 | 
 | ||||||
| Podporujeme mnoho platforem. Zde jsou adresy pro rychlou práci na webové stránce: | Podporujeme mnoho platforem. Zde jsou adresy pro rychlou práci na webové stránce: | ||||||
| @ -109,10 +106,10 @@ Podporujeme mnoho platforem. Zde jsou adresy pro rychlou práci na webové strá | |||||||
| 
 | 
 | ||||||
| 🤲 Pro usnadnění uživatelské zkušenosti nabízíme různá řešení nasazení. Způsob nasazení si můžete vybrat ze seznamu níže: | 🤲 Pro usnadnění uživatelské zkušenosti nabízíme různá řešení nasazení. Způsob nasazení si můžete vybrat ze seznamu níže: | ||||||
| 
 | 
 | ||||||
| + **[Průvodce nasazením zdrojového kódu](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | - **[Průvodce nasazením zdrojového kódu](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | ||||||
| + **[Docker Deployment Guide](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | - **[Docker Deployment Guide](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | ||||||
| + **[Průvodce nasazením Kubernetes](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | - **[Průvodce nasazením Kubernetes](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | ||||||
| + **[Průvodce nasazením pro vývojáře Mac](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | - **[Průvodce nasazením pro vývojáře Mac](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | ||||||
| 
 | 
 | ||||||
| ## :hammer_and_wrench: Chcete-li začít vyvíjet OpenIM | ## :hammer_and_wrench: Chcete-li začít vyvíjet OpenIM | ||||||
| 
 | 
 | ||||||
| @ -122,7 +119,7 @@ OpenIM Naším cílem je vybudovat špičkovou open source komunitu. Máme soubo | |||||||
| 
 | 
 | ||||||
| Pokud byste chtěli přispět do tohoto úložiště Open-IM-Server, přečtěte si naši [dokumentaci pro přispěvatele](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md). | Pokud byste chtěli přispět do tohoto úložiště Open-IM-Server, přečtěte si naši [dokumentaci pro přispěvatele](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md). | ||||||
| 
 | 
 | ||||||
| Než začnete, ujistěte se, že jsou vaše změny vyžadovány. Nejlepší pro to je vytvořit [nová diskuze](https://github.com/openimsdk/open-im-server/discussions/new/choose) NEBO [Slack Communication](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q), nebo pokud narazíte na problém, [nahlásit jej](https://github.com/openimsdk/open-im-server/issues/new/choose) jako první. | Než začnete, ujistěte se, že jsou vaše změny vyžadovány. Nejlepší pro to je vytvořit [nová diskuze](https://github.com/openimsdk/open-im-server/discussions/new/choose) NEBO [Slack Communication](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A), nebo pokud narazíte na problém, [nahlásit jej](https://github.com/openimsdk/open-im-server/issues/new/choose) jako první. | ||||||
| 
 | 
 | ||||||
| - [OpenIM API Reference](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | - [OpenIM API Reference](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | ||||||
| - [Protokolování OpenIM Bash](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | - [Protokolování OpenIM Bash](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | ||||||
| @ -154,19 +151,18 @@ Než začnete, ujistěte se, že jsou vaše změny vyžadovány. Nejlepší pro | |||||||
| - [Spravovat backend a monitorovat nasazení](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | - [Spravovat backend a monitorovat nasazení](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | ||||||
| - [Průvodce nasazením pro vývojáře Mac pro OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | - [Průvodce nasazením pro vývojáře Mac pro OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## :busts_in_silhouette: Společenství | ## :busts_in_silhouette: Společenství | ||||||
| 
 | 
 | ||||||
| + 📚 [Komunita OpenIM](https://github.com/OpenIMSDK/community) | - 📚 [Komunita OpenIM](https://github.com/OpenIMSDK/community) | ||||||
| + 💕 [Zájmová skupina OpenIM](https://github.com/Openim-sigs) | - 💕 [Zájmová skupina OpenIM](https://github.com/Openim-sigs) | ||||||
| + 🚀 [Připojte se k naší komunitě Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | - 🚀 [Připojte se k naší komunitě Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| + :eyes: [Připojte se k našemu wechatu](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | - :eyes: [Připojte se k našemu wechatu](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | ||||||
| 
 | 
 | ||||||
| ## :calendar: Komunitní setkání | ## :calendar: Komunitní setkání | ||||||
| 
 | 
 | ||||||
| Chceme, aby se do naší komunity a přispívání kódu zapojil kdokoli, nabízíme dárky a odměny a vítáme vás, abyste se k nám připojili každý čtvrtek večer. | Chceme, aby se do naší komunity a přispívání kódu zapojil kdokoli, nabízíme dárky a odměny a vítáme vás, abyste se k nám připojili každý čtvrtek večer. | ||||||
| 
 | 
 | ||||||
| Naše konference je v [OpenIM Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) 🎯, pak můžete vyhledat kanál Open-IM-Server a připojit se | Naše konference je v [OpenIM Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) 🎯, pak můžete vyhledat kanál Open-IM-Server a připojit se | ||||||
| 
 | 
 | ||||||
| Zaznamenáváme si každou [dvoutýdenní schůzku](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting)do [diskuzí na GitHubu](https://github.com/openimsdk/open-im-server/discussions/categories/meeting), naše historické poznámky ze schůzek a také záznamy schůzek jsou k dispozici na [Dokumenty Google :bookmark_tabs:](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing). | Zaznamenáváme si každou [dvoutýdenní schůzku](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting)do [diskuzí na GitHubu](https://github.com/openimsdk/open-im-server/discussions/categories/meeting), naše historické poznámky ze schůzek a také záznamy schůzek jsou k dispozici na [Dokumenty Google :bookmark_tabs:](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing). | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -12,12 +12,11 @@ | |||||||
| [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | ||||||
| [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | ||||||
| [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | ||||||
| [](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | [](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| [](https://www.bestpractices.dev/projects/8045) | [](https://www.bestpractices.dev/projects/8045) | ||||||
| [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | ||||||
| [](https://golang.org/) | [](https://golang.org/) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| <p align="center"> | <p align="center"> | ||||||
|   <a href="../../README.md">English</a> ·  |   <a href="../../README.md">English</a> ·  | ||||||
|   <a href="../../README_zh_CN.md">中文</a> ·  |   <a href="../../README_zh_CN.md">中文</a> ·  | ||||||
| @ -46,35 +45,30 @@ | |||||||
|   <a href="./README_tr.md">Türkçe</a> |   <a href="./README_tr.md">Türkçe</a> | ||||||
| </p> | </p> | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| </div> | </div> | ||||||
| 
 | 
 | ||||||
| </p> | </p> | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## :busts_in_silhouette: Fællesskab | ## :busts_in_silhouette: Fællesskab | ||||||
| 
 | 
 | ||||||
| + 📚 [OpenIM-fællesskab](https://github.com/OpenIMSDK/community) | - 📚 [OpenIM-fællesskab](https://github.com/OpenIMSDK/community) | ||||||
| + 💕 [OpenIM-interessegruppe](https://github.com/Openim-sigs) | - 💕 [OpenIM-interessegruppe](https://github.com/Openim-sigs) | ||||||
| + 🚀 [Deltag i vores Slack-fællesskab](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | - 🚀 [Deltag i vores Slack-fællesskab](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| + :eyes: [Deltag i vores WeChat (微信群)](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | - :eyes: [Deltag i vores WeChat (微信群)](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | ||||||
| + 👫 [Deltag i vores Reddit](https://www.reddit.com/r/OpenIMessaging) | - 👫 [Deltag i vores Reddit](https://www.reddit.com/r/OpenIMessaging) | ||||||
| + 💬 [Følg vores Twitter-konto](https://twitter.com/openimsdk) | - 💬 [Følg vores Twitter-konto](https://twitter.com/openimsdk) | ||||||
| 
 |  | ||||||
| 
 | 
 | ||||||
| ## Ⓜ️ Om OpenIM | ## Ⓜ️ Om OpenIM | ||||||
| 
 | 
 | ||||||
| OpenIM er en serviceplatform designet specifikt til integration af chat, lyd-videoopkald, notifikationer og AI-chatbots i applikationer. Den tilbyder en række kraftfulde API'er og Webhooks, som gør det let for udviklere at integrere disse interaktive funktioner i deres applikationer. OpenIM er ikke en selvstændig chatapplikation, men fungerer snarere som en platform, der understøtter andre applikationer i at opnå omfattende kommunikationsfunktionaliteter. Følgende diagram illustrerer interaktionen mellem AppServer, AppClient, OpenIMServer og OpenIMSDK for at forklare detaljeret. | OpenIM er en serviceplatform designet specifikt til integration af chat, lyd-videoopkald, notifikationer og AI-chatbots i applikationer. Den tilbyder en række kraftfulde API'er og Webhooks, som gør det let for udviklere at integrere disse interaktive funktioner i deres applikationer. OpenIM er ikke en selvstændig chatapplikation, men fungerer snarere som en platform, der understøtter andre applikationer i at opnå omfattende kommunikationsfunktionaliteter. Følgende diagram illustrerer interaktionen mellem AppServer, AppClient, OpenIMServer og OpenIMSDK for at forklare detaljeret. | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|  |  | ||||||
| 
 | 
 | ||||||
| ## 🚀 Om OpenIMSDK | ## 🚀 Om OpenIMSDK | ||||||
| 
 | 
 | ||||||
|  **OpenIMSDK** er en IM SDK designet til **OpenIMServer**, skabt specifikt til indlejring i klientapplikationer. Dens vigtigste funktioner og moduler er som følger: | **OpenIMSDK** er en IM SDK designet til **OpenIMServer**, skabt specifikt til indlejring i klientapplikationer. Dens vigtigste funktioner og moduler er som følger: | ||||||
| 
 | 
 | ||||||
| + 🌟 Hovedfunktioner: | - 🌟 Hovedfunktioner: | ||||||
| 
 | 
 | ||||||
|   - 📦 Lokal lagring |   - 📦 Lokal lagring | ||||||
|   - 🔔 Lytter-callbacks |   - 🔔 Lytter-callbacks | ||||||
| @ -95,15 +89,15 @@ Det er bygget ved hjælp af Golang og understøtter tværplatformsudrulning, hvi | |||||||
| 
 | 
 | ||||||
| ## 🌐 Om OpenIMServer | ## 🌐 Om OpenIMServer | ||||||
| 
 | 
 | ||||||
| + **OpenIMServer** har følgende karakteristika: | - **OpenIMServer** har følgende karakteristika: | ||||||
|   - 🌐 Mikroservicarkitektur: Understøtter klyngetilstand, inklusive en gateway og flere rpc-tjenester. |   - 🌐 Mikroservicarkitektur: Understøtter klyngetilstand, inklusive en gateway og flere rpc-tjenester. | ||||||
|   - 🚀 Forskellige udrulningsmetoder: Understøtter udrulning via kildekode, Kubernetes eller Docker. |   - 🚀 Forskellige udrulningsmetoder: Understøtter udrulning via kildekode, Kubernetes eller Docker. | ||||||
|   - Støtte til massiv brugerbase: Super store grupper med hundredtusinder af brugere, titusinder af brugere og milliarder af beskeder. |   - Støtte til massiv brugerbase: Super store grupper med hundredtusinder af brugere, titusinder af brugere og milliarder af beskeder. | ||||||
| 
 | 
 | ||||||
| ### Forbedret forretningsfunktionalitet: | ### Forbedret forretningsfunktionalitet: | ||||||
| 
 | 
 | ||||||
| + **REST API**:OpenIMServer tilbyder REST API'er til forretningssystemer, med det formål at give virksomheder flere funktioner, såsom at oprette grupper og sende push-beskeder gennem backend-grænseflader. | - **REST API**:OpenIMServer tilbyder REST API'er til forretningssystemer, med det formål at give virksomheder flere funktioner, såsom at oprette grupper og sende push-beskeder gennem backend-grænseflader. | ||||||
| + **Webhooks**:OpenIMServer giver mulighed for callback-funktionalitet for at udvide flere forretningsformer. Et callback betyder, at OpenIMServer sender en anmodning til forretningsserveren før eller efter en bestemt begivenhed, som callbacks før eller efter at have sendt en besked. | - **Webhooks**:OpenIMServer giver mulighed for callback-funktionalitet for at udvide flere forretningsformer. Et callback betyder, at OpenIMServer sender en anmodning til forretningsserveren før eller efter en bestemt begivenhed, som callbacks før eller efter at have sendt en besked. | ||||||
| 
 | 
 | ||||||
| 👉 **[Lær mere](https://docs.openim.io/guides/introduction/product)** | 👉 **[Lær mere](https://docs.openim.io/guides/introduction/product)** | ||||||
| 
 | 
 | ||||||
| @ -121,10 +115,10 @@ Vi understøtter mange platforme. Her er adresserne for hurtig oplevelse på web | |||||||
| 
 | 
 | ||||||
| 🤲 For at lette brugeroplevelsen tilbyder vi forskellige udrulningsløsninger. Du kan vælge din udrulningsmetode fra listen nedenfor: | 🤲 For at lette brugeroplevelsen tilbyder vi forskellige udrulningsløsninger. Du kan vælge din udrulningsmetode fra listen nedenfor: | ||||||
| 
 | 
 | ||||||
| + **[Vejledning til udrulning af kildekode](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | - **[Vejledning til udrulning af kildekode](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | ||||||
| + **[Vejledning til Docker-udrulning](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | - **[Vejledning til Docker-udrulning](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | ||||||
| + **[Vejledning til Kubernetes-udrulning](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | - **[Vejledning til Kubernetes-udrulning](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | ||||||
| + **[Vejledning til Mac-udviklerudrulning](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | - **[Vejledning til Mac-udviklerudrulning](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | ||||||
| 
 | 
 | ||||||
| ## :hammer_and_wrench: For at starte udviklingen af OpenIM | ## :hammer_and_wrench: For at starte udviklingen af OpenIM | ||||||
| 
 | 
 | ||||||
| @ -134,7 +128,7 @@ OpenIM Vores mål er at bygge et topniveau åben kildekode-fællesskab. Vi har e | |||||||
| 
 | 
 | ||||||
| Hvis du gerne vil bidrage til dette Open-IM-Server-repositorium, bedes du læse vores [dokumentation for bidragydere](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md). | Hvis du gerne vil bidrage til dette Open-IM-Server-repositorium, bedes du læse vores [dokumentation for bidragydere](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md). | ||||||
| 
 | 
 | ||||||
| Før du starter, skal du sikre dig, at dine ændringer er efterspurgte. Det bedste for det er at oprette en [ny diskussion](https://github.com/openimsdk/open-im-server/discussions/new/choose) ELLER [Slack-kommunikation](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q), eller hvis du finder et problem, [rapportere det](https://github.com/openimsdk/open-im-server/issues/new/choose) først. | Før du starter, skal du sikre dig, at dine ændringer er efterspurgte. Det bedste for det er at oprette en [ny diskussion](https://github.com/openimsdk/open-im-server/discussions/new/choose) ELLER [Slack-kommunikation](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A), eller hvis du finder et problem, [rapportere det](https://github.com/openimsdk/open-im-server/issues/new/choose) først. | ||||||
| 
 | 
 | ||||||
| - [OpenIM API-referencer](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | - [OpenIM API-referencer](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | ||||||
| - [OpenIM Bash-logging](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | - [OpenIM Bash-logging](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | ||||||
| @ -166,12 +160,11 @@ Før du starter, skal du sikre dig, at dine ændringer er efterspurgte. Det beds | |||||||
| - [Administrer backend og overvåg udrulning](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | - [Administrer backend og overvåg udrulning](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | ||||||
| - [Mac-udviklerudrulningsguide for OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | - [Mac-udviklerudrulningsguide for OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## :calendar: Fællesskabsmøder | ## :calendar: Fællesskabsmøder | ||||||
| 
 | 
 | ||||||
| Vi ønsker, at alle involverer sig i vores fællesskab og bidrager med kode, vi tilbyder gaver og belønninger, og vi byder dig velkommen til at deltage hver torsdag aften. | Vi ønsker, at alle involverer sig i vores fællesskab og bidrager med kode, vi tilbyder gaver og belønninger, og vi byder dig velkommen til at deltage hver torsdag aften. | ||||||
| 
 | 
 | ||||||
| Vores konference er på [OpenIM Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) 🎯, derefter kan du søge Open-IM-Server pipeline for at deltage. | Vores konference er på [OpenIM Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) 🎯, derefter kan du søge Open-IM-Server pipeline for at deltage. | ||||||
| 
 | 
 | ||||||
| Vi tager [notater](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting) af hvert fjortendages møde i [GitHub-diskussioner](https://github.com/openimsdk/open-im-server/discussions/categories/meeting), Vores historiske mødenotater samt genudsendelser af møderne er tilgængelige på [Google Docs](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing) 📑. | Vi tager [notater](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting) af hvert fjortendages møde i [GitHub-diskussioner](https://github.com/openimsdk/open-im-server/discussions/categories/meeting), Vores historiske mødenotater samt genudsendelser af møderne er tilgængelige på [Google Docs](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing) 📑. | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -12,12 +12,11 @@ | |||||||
| [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | ||||||
| [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | ||||||
| [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | ||||||
| [](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | [](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| [](https://www.bestpractices.dev/projects/8045) | [](https://www.bestpractices.dev/projects/8045) | ||||||
| [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | ||||||
| [](https://golang.org/) | [](https://golang.org/) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| <p align="center"> | <p align="center"> | ||||||
|   <a href="../../README.md">English</a> ·  |   <a href="../../README.md">English</a> ·  | ||||||
|   <a href="../../README_zh_CN.md">中文</a> ·  |   <a href="../../README_zh_CN.md">中文</a> ·  | ||||||
| @ -46,7 +45,6 @@ | |||||||
|   <a href="./README_tr.md">Türkçe</a> |   <a href="./README_tr.md">Türkçe</a> | ||||||
| </p> | </p> | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| </div> | </div> | ||||||
| 
 | 
 | ||||||
| </p> | </p> | ||||||
| @ -61,14 +59,14 @@ | |||||||
| 
 | 
 | ||||||
| Το **OpenIMSDK** είναι ένα SDK για αμεση ανταλλαγή μηνυμάτων σχεδιασμένο για το **OpenIMServer**, δημιουργήθηκε ειδικά για ενσωμάτωση σε εφαρμογές πελατών. Οι κύριες δυνατότητες και μονάδες του είναι οι εξής: | Το **OpenIMSDK** είναι ένα SDK για αμεση ανταλλαγή μηνυμάτων σχεδιασμένο για το **OpenIMServer**, δημιουργήθηκε ειδικά για ενσωμάτωση σε εφαρμογές πελατών. Οι κύριες δυνατότητες και μονάδες του είναι οι εξής: | ||||||
| 
 | 
 | ||||||
| + 🌟 Κύριες Δυνατότητες: | - 🌟 Κύριες Δυνατότητες: | ||||||
| 
 | 
 | ||||||
|   - 📦 Τοπική αποθήκευση |   - 📦 Τοπική αποθήκευση | ||||||
|   - 🔔 Callbacks ακροατών |   - 🔔 Callbacks ακροατών | ||||||
|   - 🛡️ Περιτύλιγμα API |   - 🛡️ Περιτύλιγμα API | ||||||
|   - 🌐 Διαχείριση σύνδεσης |   - 🌐 Διαχείριση σύνδεσης | ||||||
| 
 | 
 | ||||||
| + 📚 Κύριες Μονάδες: | - 📚 Κύριες Μονάδες: | ||||||
| 
 | 
 | ||||||
|   1. 🚀 Αρχικοποίηση και Σύνδεση |   1. 🚀 Αρχικοποίηση και Σύνδεση | ||||||
|   2. 👤 Διαχείριση Χρηστών |   2. 👤 Διαχείριση Χρηστών | ||||||
| @ -82,15 +80,15 @@ | |||||||
| 
 | 
 | ||||||
| ## 🌐 Σχετικά με το OpenIMServer | ## 🌐 Σχετικά με το OpenIMServer | ||||||
| 
 | 
 | ||||||
| + Το **OpenIMServer** έχει τις ακόλουθες χαρακτηριστικές: | - Το **OpenIMServer** έχει τις ακόλουθες χαρακτηριστικές: | ||||||
|   - 🌐 Αρχιτεκτονική μικροϋπηρεσιών: Υποστηρίζει λειτουργία σε σύμπλεγμα, περιλαμβάνοντας έναν πύλη και πολλαπλές υπηρεσίες rpc. |   - 🌐 Αρχιτεκτονική μικροϋπηρεσιών: Υποστηρίζει λειτουργία σε σύμπλεγμα, περιλαμβάνοντας έναν πύλη και πολλαπλές υπηρεσίες rpc. | ||||||
|   - 🚀 Διάφοροι τρόποι ανάπτυξης: Υποστηρίζει ανάπτυξη μέσω πηγαίου κώδικα, Kubernetes, ή Docker. |   - 🚀 Διάφοροι τρόποι ανάπτυξης: Υποστηρίζει ανάπτυξη μέσω πηγαίου κώδικα, Kubernetes, ή Docker. | ||||||
|   - Υποστήριξη για τεράστια βάση χρηστών: Πολύ μεγάλες ομάδες με εκατοντάδες χιλιάδες χρήστες, δεκάδες εκατομμύρια χρήστες και δισεκατομμύρια μηνύματα. |   - Υποστήριξη για τεράστια βάση χρηστών: Πολύ μεγάλες ομάδες με εκατοντάδες χιλιάδες χρήστες, δεκάδες εκατομμύρια χρήστες και δισεκατομμύρια μηνύματα. | ||||||
| 
 | 
 | ||||||
| ### Ενισχυμένη Επιχειρηματική Λειτουργικότητα: | ### Ενισχυμένη Επιχειρηματική Λειτουργικότητα: | ||||||
| 
 | 
 | ||||||
| + **REST API**: Το OpenIMServer προσφέρει REST APIs για επιχειρηματικά συστήματα, με στόχο την ενδυνάμωση των επιχειρήσεων με περισσότερες λειτουργικότητες, όπως η δημιουργία ομάδων και η αποστολή μηνυμάτων push μέσω backend διεπαφών. | - **REST API**: Το OpenIMServer προσφέρει REST APIs για επιχειρηματικά συστήματα, με στόχο την ενδυνάμωση των επιχειρήσεων με περισσότερες λειτουργικότητες, όπως η δημιουργία ομάδων και η αποστολή μηνυμάτων push μέσω backend διεπαφών. | ||||||
| + **Webhooks**: Το OpenIMServer παρέχει δυνατότητες επανάκλησης για την επέκταση περισσότερων επιχειρηματικών μορφών. Μια επανάκληση σημαίνει ότι το OpenIMServer στέλνει ένα αίτημα στον επιχειρηματικό διακομιστή πριν ή μετά από ένα συγκεκριμένο γεγονός, όπως επανακλήσεις πριν ή μετά την αποστολή ενός μηνύματος. | - **Webhooks**: Το OpenIMServer παρέχει δυνατότητες επανάκλησης για την επέκταση περισσότερων επιχειρηματικών μορφών. Μια επανάκληση σημαίνει ότι το OpenIMServer στέλνει ένα αίτημα στον επιχειρηματικό διακομιστή πριν ή μετά από ένα συγκεκριμένο γεγονός, όπως επανακλήσεις πριν ή μετά την αποστολή ενός μηνύματος. | ||||||
| 
 | 
 | ||||||
| 👉 **[Μάθετε περισσότερα](https://docs.openim.io/guides/introduction/product)** | 👉 **[Μάθετε περισσότερα](https://docs.openim.io/guides/introduction/product)** | ||||||
| 
 | 
 | ||||||
| @ -100,7 +98,6 @@ | |||||||
| 
 | 
 | ||||||
|  |  | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## :rocket: Γρήγορη Εκκίνηση | ## :rocket: Γρήγορη Εκκίνηση | ||||||
| 
 | 
 | ||||||
| Υποστηρίζουμε πολλές πλατφόρμες. Εδώ είναι οι διευθύνσεις για γρήγορη εμπειρία στην πλευρά του διαδικτύου: | Υποστηρίζουμε πολλές πλατφόρμες. Εδώ είναι οι διευθύνσεις για γρήγορη εμπειρία στην πλευρά του διαδικτύου: | ||||||
| @ -109,10 +106,10 @@ | |||||||
| 
 | 
 | ||||||
| 🤲 Για να διευκολύνουμε την εμπειρία του χρήστη, προσφέρουμε διάφορες λύσεις ανάπτυξης. Μπορείτε να επιλέξετε τη μέθοδο ανάπτυξης σας από την παρακάτω λίστα: | 🤲 Για να διευκολύνουμε την εμπειρία του χρήστη, προσφέρουμε διάφορες λύσεις ανάπτυξης. Μπορείτε να επιλέξετε τη μέθοδο ανάπτυξης σας από την παρακάτω λίστα: | ||||||
| 
 | 
 | ||||||
| + **[Οδηγός Ανάπτυξης Κώδικα Πηγής](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | - **[Οδηγός Ανάπτυξης Κώδικα Πηγής](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | ||||||
| + **[δηγός Ανάπτυξης μέσω Docker](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | - **[δηγός Ανάπτυξης μέσω Docker](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | ||||||
| + **[Οδηγός Ανάπτυξης Kubernetes](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | - **[Οδηγός Ανάπτυξης Kubernetes](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | ||||||
| + **[Οδηγός Ανάπτυξης για Αναπτυξιακούς στο Mac](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | - **[Οδηγός Ανάπτυξης για Αναπτυξιακούς στο Mac](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | ||||||
| 
 | 
 | ||||||
| ## :hammer_and_wrench: Για να Αρχίσετε την Ανάπτυξη του OpenIM | ## :hammer_and_wrench: Για να Αρχίσετε την Ανάπτυξη του OpenIM | ||||||
| 
 | 
 | ||||||
| @ -122,7 +119,7 @@ OpenIM Στόχος μας είναι να δημιουργήσουμε μια  | |||||||
| 
 | 
 | ||||||
| Εάν θέλετε να συνεισφέρετε σε αυτό το αποθετήριο Open-IM-Server, παρακαλούμε διαβάστε την [τεκμηρίωση συνεισφέροντος](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md). | Εάν θέλετε να συνεισφέρετε σε αυτό το αποθετήριο Open-IM-Server, παρακαλούμε διαβάστε την [τεκμηρίωση συνεισφέροντος](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md). | ||||||
| 
 | 
 | ||||||
| Πριν ξεκινήσετε, παρακαλούμε βεβαιωθείτε ότι οι αλλαγές σας είναι ζητούμενες. Το καλύτερο για αυτό είναι να δημιουργήσετε ένα [νέα συζήτηση](https://github.com/openimsdk/open-im-server/discussions/new/choose) ή [Επικοινωνία Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q), ή αν βρείτε ένα ζήτημα, [αναφέρετέ το](https://github.com/openimsdk/open-im-server/issues/new/choose) πρώτα. | Πριν ξεκινήσετε, παρακαλούμε βεβαιωθείτε ότι οι αλλαγές σας είναι ζητούμενες. Το καλύτερο για αυτό είναι να δημιουργήσετε ένα [νέα συζήτηση](https://github.com/openimsdk/open-im-server/discussions/new/choose) ή [Επικοινωνία Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A), ή αν βρείτε ένα ζήτημα, [αναφέρετέ το](https://github.com/openimsdk/open-im-server/issues/new/choose) πρώτα. | ||||||
| 
 | 
 | ||||||
| - [Αναφορά API του OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | - [Αναφορά API του OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | ||||||
| - [Καταγραφή Bash του OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | - [Καταγραφή Bash του OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | ||||||
| @ -154,28 +151,28 @@ OpenIM Στόχος μας είναι να δημιουργήσουμε μια  | |||||||
| - [Διαχείριση backend και παρακολούθηση ανάπτυξης](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | - [Διαχείριση backend και παρακολούθηση ανάπτυξης](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | ||||||
| - [Οδηγός Ανάπτυξης για Προγραμματιστές Mac του OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | - [Οδηγός Ανάπτυξης για Προγραμματιστές Mac του OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## :busts_in_silhouette: Κοινότητα | ## :busts_in_silhouette: Κοινότητα | ||||||
| 
 | 
 | ||||||
| + 📚 [Κοινότητα OpenIM](https://github.com/OpenIMSDK/community) | - 📚 [Κοινότητα OpenIM](https://github.com/OpenIMSDK/community) | ||||||
| + 💕 [Ομάδα Ενδιαφέροντος OpenIM](https://github.com/Openim-sigs) | - 💕 [Ομάδα Ενδιαφέροντος OpenIM](https://github.com/Openim-sigs) | ||||||
| + 🚀 [Εγγραφείτε στην κοινότητα Slack μας](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | - 🚀 [Εγγραφείτε στην κοινότητα Slack μας](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| + :eyes: [γγραφείτε στην ομάδα μας wechat (微信群)](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | - :eyes: [γγραφείτε στην ομάδα μας wechat (微信群)](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | ||||||
| 
 | 
 | ||||||
| ## :calendar: Συναντήσεις της κοινότητας | ## :calendar: Συναντήσεις της κοινότητας | ||||||
| 
 | 
 | ||||||
| Θέλουμε οποιονδήποτε να εμπλακεί στην κοινότητά μας και να συνεισφέρει κώδικα. Προσφέρουμε δώρα και ανταμοιβές και σας καλωσορίζουμε να μας ενταχθείτε κάθε Πέμπτη βράδυ. | Θέλουμε οποιονδήποτε να εμπλακεί στην κοινότητά μας και να συνεισφέρει κώδικα. Προσφέρουμε δώρα και ανταμοιβές και σας καλωσορίζουμε να μας ενταχθείτε κάθε Πέμπτη βράδυ. | ||||||
| 
 | 
 | ||||||
| Η διάσκεψή μας είναι στο [OpenIM Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) 🎯, στη συνέχεια μπορείτε να αναζητήσετε τη διαδικασία Open-IM-Server για να συμμετάσχετε | Η διάσκεψή μας είναι στο [OpenIM Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) 🎯, στη συνέχεια μπορείτε να αναζητήσετε τη διαδικασία Open-IM-Server για να συμμετάσχετε | ||||||
| 
 | 
 | ||||||
| Κάνουμε σημειώσεις για κάθε μια [Σημειώνουμε κάθε διμηνιαία συνάντηση](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting) στις [συζητήσεις του GitHub](https://github.com/openimsdk/open-im-server/discussions/categories/meeting), Οι ιστορικές μας σημειώσεις συναντήσεων, καθώς και οι επαναλήψεις των συναντήσεων είναι διαθέσιμες στο[Έγγραφα της Google :bookmark_tabs:](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing). | Κάνουμε σημειώσεις για κάθε μια [Σημειώνουμε κάθε διμηνιαία συνάντηση](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting) στις [συζητήσεις του GitHub](https://github.com/openimsdk/open-im-server/discussions/categories/meeting), Οι ιστορικές μας σημειώσεις συναντήσεων, καθώς και οι επαναλήψεις των συναντήσεων είναι διαθέσιμες στο[Έγγραφα της Google :bookmark_tabs:](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing). | ||||||
| 
 | 
 | ||||||
| ## :eyes: Ποιοί Χρησιμοποιούν το OpenIM | ## :eyes: Ποιοί Χρησιμοποιούν το OpenIM | ||||||
| 
 | 
 | ||||||
| Ελέγξτε τη σελίδα με τις [μελέτες περίπτωσης χρήσης ](https://github.com/OpenIMSDK/community/blob/main/ADOPTERS.md) μας για μια λίστα των χρηστών του έργου. Μην διστάσετε να αφήσετε ένα[📝σχόλιο](https://github.com/openimsdk/open-im-server/issues/379) και να μοιραστείτε την περίπτωση χρήσης σας. | Ελέγξτε τη σελίδα με τις [μελέτες περίπτωσης χρήσης ](https://github.com/OpenIMSDK/community/blob/main/ADOPTERS.md) μας για μια λίστα των χρηστών του έργου. Μην διστάσετε να αφήσετε ένα[📝σχόλιο](https://github.com/openimsdk/open-im-server/issues/379) και να μοιραστείτε την περίπτωση χρήσης σας. | ||||||
|  | 
 | ||||||
| ## :page_facing_up: Άδεια Χρήσης | ## :page_facing_up: Άδεια Χρήσης | ||||||
| 
 | 
 | ||||||
| Το OpenIM διατίθεται υπό την άδεια Apache 2.0. Δείτε τη [ΑΔΕΙΑ ΧΡΗΣΗΣ](https://github.com/openimsdk/open-im-server/tree/main/LICENSE)  για το πλήρες κείμενο της άδειας. | Το OpenIM διατίθεται υπό την άδεια Apache 2.0. Δείτε τη [ΑΔΕΙΑ ΧΡΗΣΗΣ](https://github.com/openimsdk/open-im-server/tree/main/LICENSE) για το πλήρες κείμενο της άδειας. | ||||||
| 
 | 
 | ||||||
| Το λογότυπο του OpenIM, συμπεριλαμβανομένων των παραλλαγών και των κινούμενων εικόνων, που εμφανίζονται σε αυτό το αποθετήριο[OpenIM](https://github.com/openimsdk/open-im-server) υπό τις διευθύνσεις [assets/logo](../../assets/logo) και [assets/logo-gif](../../assets/logo-gif) προστατεύονται από τους νόμους περί πνευματικής ιδιοκτησίας. | Το λογότυπο του OpenIM, συμπεριλαμβανομένων των παραλλαγών και των κινούμενων εικόνων, που εμφανίζονται σε αυτό το αποθετήριο[OpenIM](https://github.com/openimsdk/open-im-server) υπό τις διευθύνσεις [assets/logo](../../assets/logo) και [assets/logo-gif](../../assets/logo-gif) προστατεύονται από τους νόμους περί πνευματικής ιδιοκτησίας. | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -12,12 +12,11 @@ | |||||||
| [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | ||||||
| [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | ||||||
| [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | ||||||
| [](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | [](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| [](https://www.bestpractices.dev/projects/8045) | [](https://www.bestpractices.dev/projects/8045) | ||||||
| [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | ||||||
| [](https://golang.org/) | [](https://golang.org/) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| <p align="center"> | <p align="center"> | ||||||
|   <a href="../../README.md">English</a> ·  |   <a href="../../README.md">English</a> ·  | ||||||
|   <a href="../../README_zh_CN.md">中文</a> ·  |   <a href="../../README_zh_CN.md">中文</a> ·  | ||||||
| @ -46,7 +45,6 @@ | |||||||
|   <a href="./README_tr.md">Türkçe</a> |   <a href="./README_tr.md">Türkçe</a> | ||||||
| </p> | </p> | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| </div> | </div> | ||||||
| 
 | 
 | ||||||
| </p> | </p> | ||||||
| @ -61,14 +59,14 @@ OpenIM es una plataforma de servicio diseñada específicamente para integrar ch | |||||||
| 
 | 
 | ||||||
| **OpenIMSDK** es un SDK de mensajería instantánea diseñado para **OpenIMServer**, creado específicamente para su incorporación en aplicaciones cliente. Sus principales características y módulos son los siguientes: | **OpenIMSDK** es un SDK de mensajería instantánea diseñado para **OpenIMServer**, creado específicamente para su incorporación en aplicaciones cliente. Sus principales características y módulos son los siguientes: | ||||||
| 
 | 
 | ||||||
| + 🌟 Características Principales: | - 🌟 Características Principales: | ||||||
| 
 | 
 | ||||||
|   - 📦 Almacenamiento local |   - 📦 Almacenamiento local | ||||||
|   - 🔔 Callbacks de escuchas |   - 🔔 Callbacks de escuchas | ||||||
|   - 🛡️ Envoltura de API |   - 🛡️ Envoltura de API | ||||||
|   - 🌐 Gestión de conexiones |   - 🌐 Gestión de conexiones | ||||||
| 
 | 
 | ||||||
| + 📚 Módulos Principales: | - 📚 Módulos Principales: | ||||||
| 
 | 
 | ||||||
|   1. 🚀 Inicialización y acceso |   1. 🚀 Inicialización y acceso | ||||||
|   2. 👤 Gestión de usuarios |   2. 👤 Gestión de usuarios | ||||||
| @ -82,17 +80,15 @@ Está construido con Golang y soporta despliegue multiplataforma, asegurando una | |||||||
| 
 | 
 | ||||||
| ## 🌐 Acerca de OpenIMServer | ## 🌐 Acerca de OpenIMServer | ||||||
| 
 | 
 | ||||||
| + **OpenIMServer** tiene las siguientes características: | - **OpenIMServer** tiene las siguientes características: | ||||||
|   - 🌐 Arquitectura de microservicios: Soporta modo cluster, incluyendo un gateway y múltiples servicios rpc. |   - 🌐 Arquitectura de microservicios: Soporta modo cluster, incluyendo un gateway y múltiples servicios rpc. | ||||||
|   - 🚀 Métodos de despliegue diversos: Soporta el despliegue a través de código fuente, Kubernetes o Docker. |   - 🚀 Métodos de despliegue diversos: Soporta el despliegue a través de código fuente, Kubernetes o Docker. | ||||||
|   - Soporte para una base de usuarios masiva: Grupos super grandes con cientos de miles de usuarios, decenas de millones de usuarios y miles de millones de mensajes. |   - Soporte para una base de usuarios masiva: Grupos super grandes con cientos de miles de usuarios, decenas de millones de usuarios y miles de millones de mensajes. | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| ### Funcionalidad Empresarial Mejorada: | ### Funcionalidad Empresarial Mejorada: | ||||||
| 
 | 
 | ||||||
| + **API REST**: OpenIMServer ofrece APIs REST para sistemas empresariales, destinadas a empoderar a las empresas con más funcionalidades, como la creación de grupos y el envío de mensajes push a través de interfaces de backend. | - **API REST**: OpenIMServer ofrece APIs REST para sistemas empresariales, destinadas a empoderar a las empresas con más funcionalidades, como la creación de grupos y el envío de mensajes push a través de interfaces de backend. | ||||||
| + **Webhooks**: OpenIMServer proporciona capacidades de callback para extender más formas de negocio. Un callback significa que OpenIMServer envía una solicitud al servidor empresarial antes o después de un cierto evento, como callbacks antes o después de enviar un mensaje. | - **Webhooks**: OpenIMServer proporciona capacidades de callback para extender más formas de negocio. Un callback significa que OpenIMServer envía una solicitud al servidor empresarial antes o después de un cierto evento, como callbacks antes o después de enviar un mensaje. | ||||||
| 
 | 
 | ||||||
| 👉 **[Aprende más](https://docs.openim.io/guides/introduction/product)** | 👉 **[Aprende más](https://docs.openim.io/guides/introduction/product)** | ||||||
| 
 | 
 | ||||||
| @ -102,10 +98,8 @@ Adéntrate en el corazón de la funcionalidad de Open-IM-Server con nuestro diag | |||||||
| 
 | 
 | ||||||
|  |  | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## :rocket: Inicio Rápido | ## :rocket: Inicio Rápido | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| :rocket: Inicio Rápido | :rocket: Inicio Rápido | ||||||
| Apoyamos muchas plataformas. Aquí están las direcciones para una experiencia rápida en el lado web: | Apoyamos muchas plataformas. Aquí están las direcciones para una experiencia rápida en el lado web: | ||||||
| 
 | 
 | ||||||
| @ -113,10 +107,10 @@ Apoyamos muchas plataformas. Aquí están las direcciones para una experiencia r | |||||||
| 
 | 
 | ||||||
| 🤲 Para facilitar la experiencia del usuario, ofrecemos varias soluciones de despliegue. Puedes elegir tu método de despliegue de la lista a continuación: | 🤲 Para facilitar la experiencia del usuario, ofrecemos varias soluciones de despliegue. Puedes elegir tu método de despliegue de la lista a continuación: | ||||||
| 
 | 
 | ||||||
| + **[Guía de Despliegue de Código Fuente](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | - **[Guía de Despliegue de Código Fuente](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | ||||||
| + **[Guía de Despliegue con Docker](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | - **[Guía de Despliegue con Docker](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | ||||||
| + **[Guía de Despliegue con Kubernetes](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | - **[Guía de Despliegue con Kubernetes](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | ||||||
| + **[Guía de Despliegue para Desarrolladores en Mac](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | - **[Guía de Despliegue para Desarrolladores en Mac](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | ||||||
| 
 | 
 | ||||||
| ## :hammer_and_wrench: Para Comenzar a Desarrollar en OpenIM | ## :hammer_and_wrench: Para Comenzar a Desarrollar en OpenIM | ||||||
| 
 | 
 | ||||||
| @ -127,8 +121,7 @@ en el [repositorio de la Comunidad.](https://github.com/OpenIMSDK/community). | |||||||
| 
 | 
 | ||||||
| Si te gustaría contribuir a este repositorio de Open-IM-Server, por favor lee nuestra [documentación para colaboradores](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md). | Si te gustaría contribuir a este repositorio de Open-IM-Server, por favor lee nuestra [documentación para colaboradores](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md). | ||||||
| 
 | 
 | ||||||
| 
 | Antes de comenzar, asegúrate de que tus cambios sean demandados. Lo mejor para eso es crear una [nueva discusión](https://github.com/openimsdk/open-im-server/discussions/new/choose) O [Comunicación en Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A), o si encuentras un problema, [repórtalo](https://github.com/openimsdk/open-im-server/issues/new/choose) primero. | ||||||
| Antes de comenzar, asegúrate de que tus cambios sean demandados. Lo mejor para eso es crear una [nueva discusión](https://github.com/openimsdk/open-im-server/discussions/new/choose) O [Comunicación en Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q), o si encuentras un problema, [repórtalo](https://github.com/openimsdk/open-im-server/issues/new/choose) primero. |  | ||||||
| 
 | 
 | ||||||
| - [Referencia de API de OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | - [Referencia de API de OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | ||||||
| - [Registro de Bash de OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | - [Registro de Bash de OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | ||||||
| @ -160,32 +153,31 @@ Antes de comenzar, asegúrate de que tus cambios sean demandados. Lo mejor para | |||||||
| - [Gestión de backend y despliegue de monitoreo](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | - [Gestión de backend y despliegue de monitoreo](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | ||||||
| - [Guía de Despliegue para Desarrolladores Mac de OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | - [Guía de Despliegue para Desarrolladores Mac de OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## :busts_in_silhouette: Comunidad | ## :busts_in_silhouette: Comunidad | ||||||
| 
 | 
 | ||||||
| + 📚 [Comunidad de OpenIM](https://github.com/OpenIMSDK/community) | - 📚 [Comunidad de OpenIM](https://github.com/OpenIMSDK/community) | ||||||
| + 💕 [Grupo de Interés de OpenIM](https://github.com/Openim-sigs) | - 💕 [Grupo de Interés de OpenIM](https://github.com/Openim-sigs) | ||||||
| + 🚀 [Únete a nuestra comunidad de Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | - 🚀 [Únete a nuestra comunidad de Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| + :eyes: [Únete a nuestro wechat (微信群)](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | - :eyes: [Únete a nuestro wechat (微信群)](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | ||||||
| 
 | 
 | ||||||
| ## :calendar: Reuniones de la Comunidad | ## :calendar: Reuniones de la Comunidad | ||||||
| 
 | 
 | ||||||
| Queremos que cualquiera se involucre en nuestra comunidad y contribuya con código, ofrecemos regalos y recompensas, y te damos la bienvenida para que te unas a nosotros cada jueves por la noche. | Queremos que cualquiera se involucre en nuestra comunidad y contribuya con código, ofrecemos regalos y recompensas, y te damos la bienvenida para que te unas a nosotros cada jueves por la noche. | ||||||
| 
 | 
 | ||||||
| Nuestra conferencia está en [OpenIM Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) 🎯, luego puedes buscar el pipeline de Open-IM-Server para unirte | Nuestra conferencia está en [OpenIM Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) 🎯, luego puedes buscar el pipeline de Open-IM-Server para unirte | ||||||
| 
 | 
 | ||||||
| Tomamos notas de cada [reunión quincenal](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting) en [discusiones de GitHub](https://github.com/openimsdk/open-im-server/discussions/categories/meeting), Nuestras notas de reuniones históricas, así como las repeticiones de las reuniones están disponibles en [Google Docs :bookmark_tabs:](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing). | Tomamos notas de cada [reunión quincenal](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting) en [discusiones de GitHub](https://github.com/openimsdk/open-im-server/discussions/categories/meeting), Nuestras notas de reuniones históricas, así como las repeticiones de las reuniones están disponibles en [Google Docs :bookmark_tabs:](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing). | ||||||
| 
 | 
 | ||||||
| ## :eyes: Quiénes Están Usando OpenIM | ## :eyes: Quiénes Están Usando OpenIM | ||||||
| 
 | 
 | ||||||
| Consulta nuestros [estudios de caso de usuarios](https://github.com/OpenIMSDK/community/blob/main/ADOPTERS.md) página para obtener una lista de los usuarios del proyecto. No dudes en dejar un [📝comentario](https://github.com/openimsdk/open-im-server/issues/379) y compartir tu caso de uso. | Consulta nuestros [estudios de caso de usuarios](https://github.com/OpenIMSDK/community/blob/main/ADOPTERS.md) página para obtener una lista de los usuarios del proyecto. No dudes en dejar un [📝comentario](https://github.com/openimsdk/open-im-server/issues/379) y compartir tu caso de uso. | ||||||
|  | 
 | ||||||
| ## :page_facing_up: Licencia | ## :page_facing_up: Licencia | ||||||
| 
 | 
 | ||||||
| 
 | OpenIM está bajo la licencia Apache 2.0. Consulta [LICENSE](https://github.com/openimsdk/open-im-server/tree/main/LICENSE) para ver el texto completo de la licencia. | ||||||
| OpenIM está bajo la licencia Apache 2.0. Consulta [LICENSE](https://github.com/openimsdk/open-im-server/tree/main/LICENSE)  para ver el texto completo de la licencia. |  | ||||||
| 
 |  | ||||||
| 
 | 
 | ||||||
| El logotipo de OpenIM, incluyendo sus variaciones y versiones animadas, que se muestran en este repositorio [OpenIM](https://github.com/openimsdk/open-im-server) en los directorios [assets/logo](../../assets/logo) y [assets/logo-gif](assets/logo-gif) están protegidos por las leyes de derechos de autor. | El logotipo de OpenIM, incluyendo sus variaciones y versiones animadas, que se muestran en este repositorio [OpenIM](https://github.com/openimsdk/open-im-server) en los directorios [assets/logo](../../assets/logo) y [assets/logo-gif](assets/logo-gif) están protegidos por las leyes de derechos de autor. | ||||||
|  | 
 | ||||||
| ## 🔮 iGracias a nuestros colaboradores! | ## 🔮 iGracias a nuestros colaboradores! | ||||||
| 
 | 
 | ||||||
| <a href="https://github.com/openimsdk/open-im-server/graphs/contributors"> | <a href="https://github.com/openimsdk/open-im-server/graphs/contributors"> | ||||||
|  | |||||||
| @ -12,12 +12,11 @@ | |||||||
| [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | ||||||
| [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | ||||||
| [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | ||||||
| [](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | [](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| [](https://www.bestpractices.dev/projects/8045) | [](https://www.bestpractices.dev/projects/8045) | ||||||
| [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | ||||||
| [](https://golang.org/) | [](https://golang.org/) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| <p align="center"> | <p align="center"> | ||||||
|   <a href="../../README.md">English</a> ·  |   <a href="../../README.md">English</a> ·  | ||||||
|   <a href="../../README_zh_CN.md">中文</a> ·  |   <a href="../../README_zh_CN.md">中文</a> ·  | ||||||
| @ -46,7 +45,6 @@ | |||||||
|   <a href="./README_tr.md">Türkçe</a> |   <a href="./README_tr.md">Türkçe</a> | ||||||
| </p> | </p> | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| </div> | </div> | ||||||
| 
 | 
 | ||||||
| </p> | </p> | ||||||
| @ -61,14 +59,14 @@ OpenIM یک پلتفرم خدماتی است که به طور خاص برای ا | |||||||
| 
 | 
 | ||||||
| **OpenIMSDK** یک IM SDK است که برای **OpenIMServer** طراحی شده است که به طور خاص برای جاسازی در برنامه های مشتری ایجاد شده است. ویژگی ها و ماژول های اصلی آن به شرح زیر است: | **OpenIMSDK** یک IM SDK است که برای **OpenIMServer** طراحی شده است که به طور خاص برای جاسازی در برنامه های مشتری ایجاد شده است. ویژگی ها و ماژول های اصلی آن به شرح زیر است: | ||||||
| 
 | 
 | ||||||
| + 🌟 ویژگی های اصلی: | - 🌟 ویژگی های اصلی: | ||||||
| 
 | 
 | ||||||
|   - 📦 ذخیره سازی محلی |   - 📦 ذخیره سازی محلی | ||||||
|   - 🔔 پاسخ تماس شنونده |   - 🔔 پاسخ تماس شنونده | ||||||
|   - 🛡️ بسته بندی API |   - 🛡️ بسته بندی API | ||||||
|   - 🌐 مدیریت اتصال |   - 🌐 مدیریت اتصال | ||||||
| 
 | 
 | ||||||
| + 📚 ماژول های اصلی: | - 📚 ماژول های اصلی: | ||||||
| 
 | 
 | ||||||
|   1. 🚀 مقداردهی اولیه و ورود |   1. 🚀 مقداردهی اولیه و ورود | ||||||
|   2. 👤 مدیریت کاربر |   2. 👤 مدیریت کاربر | ||||||
| @ -82,15 +80,15 @@ OpenIM یک پلتفرم خدماتی است که به طور خاص برای ا | |||||||
| 
 | 
 | ||||||
| ## 🌐 درباره OpenIMServer | ## 🌐 درباره OpenIMServer | ||||||
| 
 | 
 | ||||||
| + **OpenIMServer** دارای ویژگی های زیر است: | - **OpenIMServer** دارای ویژگی های زیر است: | ||||||
|   - 🌐 معماری Microservice: از حالت کلاستر، از جمله یک دروازه و چندین سرویس rpc پشتیبانی می کند. |   - 🌐 معماری Microservice: از حالت کلاستر، از جمله یک دروازه و چندین سرویس rpc پشتیبانی می کند. | ||||||
|   - 🚀 روشهای استقرار متنوع: از استقرار از طریق کد منبع، Kubernetes یا Docker پشتیبانی میکند. |   - 🚀 روشهای استقرار متنوع: از استقرار از طریق کد منبع، Kubernetes یا Docker پشتیبانی میکند. | ||||||
|   - پشتیبانی از پایگاه عظیم کاربران: گروه های فوق العاده بزرگ با صدها هزار کاربر، ده ها میلیون کاربر و میلیاردها پیام. |   - پشتیبانی از پایگاه عظیم کاربران: گروه های فوق العاده بزرگ با صدها هزار کاربر، ده ها میلیون کاربر و میلیاردها پیام. | ||||||
| 
 | 
 | ||||||
| ### عملکردهای تجاری پیشرفته: | ### عملکردهای تجاری پیشرفته: | ||||||
| 
 | 
 | ||||||
| + **REST API**: OpenIMServer APIهای REST را برای سیستمهای تجاری ارائه میکند، با هدف توانمندسازی کسبوکارها با قابلیتهای بیشتر، مانند ایجاد گروهها و ارسال پیامهای فشار از طریق رابطهای باطنی. | - **REST API**: OpenIMServer APIهای REST را برای سیستمهای تجاری ارائه میکند، با هدف توانمندسازی کسبوکارها با قابلیتهای بیشتر، مانند ایجاد گروهها و ارسال پیامهای فشار از طریق رابطهای باطنی. | ||||||
| + **Webhooks**: OpenIMServer قابلیت های پاسخ به تماس را برای گسترش بیشتر فرم های تجاری ارائه می دهد. پاسخ به تماس به این معنی است که OpenIMServer درخواستی را قبل یا بعد از یک رویداد خاص به سرور تجاری ارسال می کند، مانند تماس های قبل یا بعد از ارسال یک پیام. | - **Webhooks**: OpenIMServer قابلیت های پاسخ به تماس را برای گسترش بیشتر فرم های تجاری ارائه می دهد. پاسخ به تماس به این معنی است که OpenIMServer درخواستی را قبل یا بعد از یک رویداد خاص به سرور تجاری ارسال می کند، مانند تماس های قبل یا بعد از ارسال یک پیام. | ||||||
| 
 | 
 | ||||||
| 👉 **[بیشتر بدانید](https://docs.openim.io/guides/introduction/product)** | 👉 **[بیشتر بدانید](https://docs.openim.io/guides/introduction/product)** | ||||||
| 
 | 
 | ||||||
| @ -100,7 +98,6 @@ OpenIM یک پلتفرم خدماتی است که به طور خاص برای ا | |||||||
| 
 | 
 | ||||||
|  |  | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## :rocket: شروع سریع | ## :rocket: شروع سریع | ||||||
| 
 | 
 | ||||||
| ما از بسیاری از پلتفرم ها پشتیبانی می کنیم. در اینجا آدرس هایی برای تجربه سریع در سمت وب آمده است: | ما از بسیاری از پلتفرم ها پشتیبانی می کنیم. در اینجا آدرس هایی برای تجربه سریع در سمت وب آمده است: | ||||||
| @ -109,10 +106,10 @@ OpenIM یک پلتفرم خدماتی است که به طور خاص برای ا | |||||||
| 
 | 
 | ||||||
| 🤲 برای تسهیل تجربه کاربر، ما راه حل های مختلف استقرار را ارائه می دهیم. می توانید روش استقرار خود را از لیست زیر انتخاب کنید: | 🤲 برای تسهیل تجربه کاربر، ما راه حل های مختلف استقرار را ارائه می دهیم. می توانید روش استقرار خود را از لیست زیر انتخاب کنید: | ||||||
| 
 | 
 | ||||||
| + **[راهنمای استقرار کد منبع](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | - **[راهنمای استقرار کد منبع](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | ||||||
| + **[راهنمای استقرار داکر](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | - **[راهنمای استقرار داکر](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | ||||||
| + **[راهنمای استقرار Kubernetes](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | - **[راهنمای استقرار Kubernetes](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | ||||||
| + **[راهنمای استقرار توسعه دهنده مک](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | - **[راهنمای استقرار توسعه دهنده مک](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | ||||||
| 
 | 
 | ||||||
| ## :hammer_and_wrench: برای شروع توسعه OpenIM | ## :hammer_and_wrench: برای شروع توسعه OpenIM | ||||||
| 
 | 
 | ||||||
| @ -122,7 +119,7 @@ OpenIM هدف ما ایجاد یک جامعه منبع باز سطح بالا ا | |||||||
| 
 | 
 | ||||||
| اگر میخواهید در این مخزن Open-IM-Server مشارکت کنید، لطفاً [مستندات مشارکتکننده](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md) ما را بخوانید. | اگر میخواهید در این مخزن Open-IM-Server مشارکت کنید، لطفاً [مستندات مشارکتکننده](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md) ما را بخوانید. | ||||||
| 
 | 
 | ||||||
| قبل از شروع، لطفاً مطمئن شوید که تغییرات شما مورد تقاضا هستند. بهترین کار برای آن این است که یک [بحث جدید](https://github.com/openimsdk/open-im-server/discussions/new/choose) یا [ارتباط اسلک](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) ایجاد کنید، یا اگر مشکلی پیدا کردید، ابتدا [آن را گزارش کنید](https://github.com/openimsdk/open-im-server/issues/new/choose). | قبل از شروع، لطفاً مطمئن شوید که تغییرات شما مورد تقاضا هستند. بهترین کار برای آن این است که یک [بحث جدید](https://github.com/openimsdk/open-im-server/discussions/new/choose) یا [ارتباط اسلک](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) ایجاد کنید، یا اگر مشکلی پیدا کردید، ابتدا [آن را گزارش کنید](https://github.com/openimsdk/open-im-server/issues/new/choose). | ||||||
| 
 | 
 | ||||||
| - [مرجع OpenIM API](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | - [مرجع OpenIM API](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | ||||||
| - [OpenIM Bash Logging](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | - [OpenIM Bash Logging](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | ||||||
| @ -154,21 +151,20 @@ OpenIM هدف ما ایجاد یک جامعه منبع باز سطح بالا ا | |||||||
| - [مدیریت استقرار باطن و نظارت](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | - [مدیریت استقرار باطن و نظارت](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | ||||||
| - [راهنمای استقرار توسعه دهنده مک برای OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | - [راهنمای استقرار توسعه دهنده مک برای OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## :busts_in_silhouette: انجمن | ## :busts_in_silhouette: انجمن | ||||||
| 
 | 
 | ||||||
| + 📚 [انجمن OpenIM](https://github.com/OpenIMSDK/community) | - 📚 [انجمن OpenIM](https://github.com/OpenIMSDK/community) | ||||||
| + 💕 [گروه علاقه OpenIM](https://github.com/Openim-sigs) | - 💕 [گروه علاقه OpenIM](https://github.com/Openim-sigs) | ||||||
| + 🚀 [به انجمن Slack ما بپیوندید](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | - 🚀 [به انجمن Slack ما بپیوندید](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| + :eyes: [به وی چت ما بپیوندید](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | - :eyes: [به وی چت ما بپیوندید](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | ||||||
| 
 | 
 | ||||||
| ## :calendar: جلسات جامعه | ## :calendar: جلسات جامعه | ||||||
| 
 | 
 | ||||||
| ما میخواهیم هر کسی در انجمن ما مشارکت کند و در کد مشارکت کند، ما هدایا و جوایزی ارائه میکنیم، و از شما استقبال میکنیم که هر پنجشنبه شب به ما بپیوندید. | ما میخواهیم هر کسی در انجمن ما مشارکت کند و در کد مشارکت کند، ما هدایا و جوایزی ارائه میکنیم، و از شما استقبال میکنیم که هر پنجشنبه شب به ما بپیوندید. | ||||||
| 
 | 
 | ||||||
| کنفرانس ما در [OpenIM Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) 🎯 است، سپس می توانید خط لوله Open-IM-Server را برای پیوستن جستجو کنید. | کنفرانس ما در [OpenIM Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) 🎯 است، سپس می توانید خط لوله Open-IM-Server را برای پیوستن جستجو کنید. | ||||||
| 
 | 
 | ||||||
| ما از هر [جلسه دو هفتهای](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting) در  [بحثهای GitHub](https://github.com/openimsdk/open-im-server/discussions/categories/meeting) یادداشتبرداری میکنیم، یادداشتهای جلسه تاریخی ما، و همچنین بازپخش جلسات در  [Google Docs :bookmark_tabs:](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing) موجود است. | ما از هر [جلسه دو هفتهای](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting) در [بحثهای GitHub](https://github.com/openimsdk/open-im-server/discussions/categories/meeting) یادداشتبرداری میکنیم، یادداشتهای جلسه تاریخی ما، و همچنین بازپخش جلسات در [Google Docs :bookmark_tabs:](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing) موجود است. | ||||||
| 
 | 
 | ||||||
| ## :eyes: چه کسانی از OpenIM استفاده می کنند | ## :eyes: چه کسانی از OpenIM استفاده می کنند | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -12,12 +12,11 @@ | |||||||
| [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | ||||||
| [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | ||||||
| [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | ||||||
| [](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | [](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| [](https://www.bestpractices.dev/projects/8045) | [](https://www.bestpractices.dev/projects/8045) | ||||||
| [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | ||||||
| [](https://golang.org/) | [](https://golang.org/) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| <p align="center"> | <p align="center"> | ||||||
|   <a href="../../README.md">English</a> ·  |   <a href="../../README.md">English</a> ·  | ||||||
|   <a href="../../README_zh_CN.md">中文</a> ·  |   <a href="../../README_zh_CN.md">中文</a> ·  | ||||||
| @ -46,25 +45,21 @@ | |||||||
|   <a href="./README_tr.md">Türkçe</a> |   <a href="./README_tr.md">Türkçe</a> | ||||||
| </p> | </p> | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| </div> | </div> | ||||||
| 
 | 
 | ||||||
| </p> | </p> | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## Ⓜ️ À propos de OpenIM | ## Ⓜ️ À propos de OpenIM | ||||||
| 
 | 
 | ||||||
| OpenIM est une plateforme de services conçue spécifiquement pour intégrer des fonctionnalités de communication telles que le chat, les appels audio et vidéo, les notifications, ainsi que les robots de chat IA dans les applications. Elle offre une série d'API puissantes et de Webhooks, permettant aux développeurs d'incorporer facilement ces caractéristiques interactives dans leurs applications. OpenIM n'est pas en soi une application de chat autonome, mais sert de plateforme supportant d'autres applications pour réaliser des fonctionnalités de communication enrichies. L'image ci-dessous montre les relations d'interaction entre AppServer, AppClient, OpenIMServer et OpenIMSDK pour illustrer spécifiquement. | OpenIM est une plateforme de services conçue spécifiquement pour intégrer des fonctionnalités de communication telles que le chat, les appels audio et vidéo, les notifications, ainsi que les robots de chat IA dans les applications. Elle offre une série d'API puissantes et de Webhooks, permettant aux développeurs d'incorporer facilement ces caractéristiques interactives dans leurs applications. OpenIM n'est pas en soi une application de chat autonome, mais sert de plateforme supportant d'autres applications pour réaliser des fonctionnalités de communication enrichies. L'image ci-dessous montre les relations d'interaction entre AppServer, AppClient, OpenIMServer et OpenIMSDK pour illustrer spécifiquement. | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|  |  | ||||||
| 
 | 
 | ||||||
| ## 🚀 À propos de OpenIMSDK | ## 🚀 À propos de OpenIMSDK | ||||||
| 
 | 
 | ||||||
| **OpenIMSDK**  est un SDK IM conçu pour **OpenIMServer** spécialement créé pour être intégré dans les applications clientes. Ses principales fonctionnalités et modules comprennent : | **OpenIMSDK** est un SDK IM conçu pour **OpenIMServer** spécialement créé pour être intégré dans les applications clientes. Ses principales fonctionnalités et modules comprennent : | ||||||
| 
 | 
 | ||||||
| + 🌟 Fonctionnalités clés : | - 🌟 Fonctionnalités clés : | ||||||
| 
 | 
 | ||||||
|   - 📦 Stockage local |   - 📦 Stockage local | ||||||
|   - 🔔 Rappels de l'écouteur |   - 🔔 Rappels de l'écouteur | ||||||
| @ -85,15 +80,15 @@ Il est construit avec Golang et supporte le déploiement multiplateforme, assura | |||||||
| 
 | 
 | ||||||
| ## 🌐 À propos de OpenIMServer | ## 🌐 À propos de OpenIMServer | ||||||
| 
 | 
 | ||||||
| + **OpenIMServer** présente les caractéristiques suivantes : | - **OpenIMServer** présente les caractéristiques suivantes : | ||||||
|   - 🌐 Architecture microservices : prend en charge le mode cluster, incluant le gateway (passerelle) et plusieurs services rpc。 |   - 🌐 Architecture microservices : prend en charge le mode cluster, incluant le gateway (passerelle) et plusieurs services rpc。 | ||||||
|   - 🚀 Divers modes de déploiement : supporte le déploiement via le code source, Kubernetes ou Docker。 |   - 🚀 Divers modes de déploiement : supporte le déploiement via le code source, Kubernetes ou Docker。 | ||||||
|   - Support d'une masse d'utilisateurs : plus de cent mille pour les super grands groupes, des millions d'utilisateurs, et des milliards de messages。 |   - Support d'une masse d'utilisateurs : plus de cent mille pour les super grands groupes, des millions d'utilisateurs, et des milliards de messages。 | ||||||
| 
 | 
 | ||||||
| ### Fonctionnalités commerciales améliorées : | ### Fonctionnalités commerciales améliorées : | ||||||
| 
 | 
 | ||||||
| + **REST API**:OpenIMServer fournit une REST API pour les systèmes commerciaux, visant à accorder plus de fonctionnalités, telles que la création de groupes via l'interface backend, l'envoi de messages push, etc。 | - **REST API**:OpenIMServer fournit une REST API pour les systèmes commerciaux, visant à accorder plus de fonctionnalités, telles que la création de groupes via l'interface backend, l'envoi de messages push, etc。 | ||||||
| + **Webhooks**:OpenIMServer offre des capacités de rappel pour étendre davantage les formes d'entreprise. Un rappel signifie que OpenIMServer enverra une requête au serveur d'entreprise avant ou après qu'un événement se soit produit, comme un rappel avant ou après l'envoi d'un message。 | - **Webhooks**:OpenIMServer offre des capacités de rappel pour étendre davantage les formes d'entreprise. Un rappel signifie que OpenIMServer enverra une requête au serveur d'entreprise avant ou après qu'un événement se soit produit, comme un rappel avant ou après l'envoi d'un message。 | ||||||
| 
 | 
 | ||||||
| 👉 **[En savoir plus](https://docs.openim.io/guides/introduction/product)** | 👉 **[En savoir plus](https://docs.openim.io/guides/introduction/product)** | ||||||
| 
 | 
 | ||||||
| @ -103,7 +98,6 @@ Plongez dans le cœur de la fonctionnalité d'Open-IM-Server avec notre diagramm | |||||||
| 
 | 
 | ||||||
|  |  | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## :rocket: Démarrage rapide | ## :rocket: Démarrage rapide | ||||||
| 
 | 
 | ||||||
| Nous prenons en charge de nombreuses plateformes. Voici les adresses pour une expérience rapide du côté web : | Nous prenons en charge de nombreuses plateformes. Voici les adresses pour une expérience rapide du côté web : | ||||||
| @ -112,17 +106,18 @@ Nous prenons en charge de nombreuses plateformes. Voici les adresses pour une ex | |||||||
| 
 | 
 | ||||||
| 🤲 Pour faciliter l'expérience utilisateur, nous proposons plusieurs solutions de déploiement. Vous pouvez choisir votre méthode de déploiement selon la liste ci-dessous : | 🤲 Pour faciliter l'expérience utilisateur, nous proposons plusieurs solutions de déploiement. Vous pouvez choisir votre méthode de déploiement selon la liste ci-dessous : | ||||||
| 
 | 
 | ||||||
| + **[Guide de déploiement du code source](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | - **[Guide de déploiement du code source](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | ||||||
| + **[Guide de déploiement Docker](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | - **[Guide de déploiement Docker](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | ||||||
| + **[Guide de déploiement Kubernetes](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | - **[Guide de déploiement Kubernetes](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | ||||||
| + **[Guide de déploiement pour développeur Mac](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | - **[Guide de déploiement pour développeur Mac](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | ||||||
| 
 | 
 | ||||||
| ## :hammer_and_wrench: Commencer à développer avec  OpenIM | ## :hammer_and_wrench: Commencer à développer avec OpenIM | ||||||
| 
 | 
 | ||||||
| Chez OpenIM, notre objectif est de construire une communauté open source de premier plan. Nous avons un ensemble de standards, disponibles dans le[ dépôt communautaire](https://github.com/OpenIMSDK/community)。 | Chez OpenIM, notre objectif est de construire une communauté open source de premier plan. Nous avons un ensemble de standards, disponibles dans le[ dépôt communautaire](https://github.com/OpenIMSDK/community)。 | ||||||
| Si vous souhaitez contribuer à ce dépôt Open-IM-Server, veuillez lire notre[ document pour les contributeurs](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md)。 | Si vous souhaitez contribuer à ce dépôt Open-IM-Server, veuillez lire notre[ document pour les contributeurs](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md)。 | ||||||
| 
 | 
 | ||||||
| Avant de commencer, assurez-vous que vos modifications sont nécessaires. La meilleure manière est de créer une[ nouvelle discussion ](https://github.com/openimsdk/open-im-server/discussions/new/choose) ou une [ communication Slack,](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q),ou si vous identifiez un problème, de[ signaler d'abord ](https://github.com/openimsdk/open-im-server/issues/new/choose)。 | Avant de commencer, assurez-vous que vos modifications sont nécessaires. La meilleure manière est de créer une[ nouvelle discussion ](https://github.com/openimsdk/open-im-server/discussions/new/choose) ou une [ communication Slack,](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A),ou si vous identifiez un problème, de[ signaler d'abord ](https://github.com/openimsdk/open-im-server/issues/new/choose)。 | ||||||
|  | 
 | ||||||
| - [Référence de l'API OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | - [Référence de l'API OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | ||||||
| - [Journalisation Bash OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | - [Journalisation Bash OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | ||||||
| - [Actions CI/CD OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/cicd-actions.md) | - [Actions CI/CD OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/cicd-actions.md) | ||||||
| @ -153,13 +148,12 @@ Avant de commencer, assurez-vous que vos modifications sont nécessaires. La mei | |||||||
| - [Gérer le déploiement du backend et la surveillance](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | - [Gérer le déploiement du backend et la surveillance](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | ||||||
| - [Guide de déploiement pour développeur Mac pour OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | - [Guide de déploiement pour développeur Mac pour OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | ||||||
| 
 | 
 | ||||||
| 
 | > ## :calendar: Réunions de la Communauté | ||||||
| >## :calendar: Réunions de la Communauté |  | ||||||
| 
 | 
 | ||||||
| Nous voulons que tout le monde s'implique dans notre communauté et contribue au code, nous offrons des cadeaux et des récompenses, et nous vous invitons à nous rejoindre chaque jeudi soir. | Nous voulons que tout le monde s'implique dans notre communauté et contribue au code, nous offrons des cadeaux et des récompenses, et nous vous invitons à nous rejoindre chaque jeudi soir. | ||||||
| Notre conférence se trouve dans le [ Slack OpenIM ](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) 🎯,  ensuite vous pouvez rechercher le pipeline Open-IM-Server pour rejoindre | Notre conférence se trouve dans le [ Slack OpenIM ](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) 🎯, ensuite vous pouvez rechercher le pipeline Open-IM-Server pour rejoindre | ||||||
| 
 | 
 | ||||||
| Nous prenons des notes de chaque [réunion bihebdomadaire ](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting) dans les  [discussions GitHub](https://github.com/openimsdk/open-im-server/discussions/categories/meeting), Nos notes de réunion historiques, ainsi que les rediffusions des réunions sont disponibles sur [ Google Docs :bookmark_tabs:](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing). | Nous prenons des notes de chaque [réunion bihebdomadaire ](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting) dans les [discussions GitHub](https://github.com/openimsdk/open-im-server/discussions/categories/meeting), Nos notes de réunion historiques, ainsi que les rediffusions des réunions sont disponibles sur [ Google Docs :bookmark_tabs:](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing). | ||||||
| 
 | 
 | ||||||
| ## :eyes: Qui Utilise OpenIM | ## :eyes: Qui Utilise OpenIM | ||||||
| 
 | 
 | ||||||
| @ -167,9 +161,9 @@ Consultez notre page [ études de cas d'utilisateurs ](https://github.com/OpenIM | |||||||
| 
 | 
 | ||||||
| ## :page_facing_up: License | ## :page_facing_up: License | ||||||
| 
 | 
 | ||||||
| OpenIM est sous licence Apache 2.0. Voir  [LICENSE](https://github.com/openimsdk/open-im-server/tree/main/LICENSE) pour le texte complet de la licence. | OpenIM est sous licence Apache 2.0. Voir [LICENSE](https://github.com/openimsdk/open-im-server/tree/main/LICENSE) pour le texte complet de la licence. | ||||||
| 
 | 
 | ||||||
| Le logo OpenIM, y compris ses variations et versions animées, affiché dans ce dépôt[OpenIM](https://github.com/openimsdk/open-im-server) sous les répertoires  [assets/logo](../../assets/logo) et [assets/logo-gif](assets/logo-gif) sont protégés par les lois sur le droit d'auteur. | Le logo OpenIM, y compris ses variations et versions animées, affiché dans ce dépôt[OpenIM](https://github.com/openimsdk/open-im-server) sous les répertoires [assets/logo](../../assets/logo) et [assets/logo-gif](assets/logo-gif) sont protégés par les lois sur le droit d'auteur. | ||||||
| 
 | 
 | ||||||
| ## 🔮 Merci à nos contributeurs ! | ## 🔮 Merci à nos contributeurs ! | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -12,12 +12,11 @@ | |||||||
| [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | ||||||
| [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | ||||||
| [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | ||||||
| [](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | [](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| [](https://www.bestpractices.dev/projects/8045) | [](https://www.bestpractices.dev/projects/8045) | ||||||
| [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | ||||||
| [](https://golang.org/) | [](https://golang.org/) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| <p align="center"> | <p align="center"> | ||||||
|   <a href="../../README.md">English</a> ·  |   <a href="../../README.md">English</a> ·  | ||||||
|   <a href="../../README_zh_CN.md">中文</a> ·  |   <a href="../../README_zh_CN.md">中文</a> ·  | ||||||
| @ -46,7 +45,6 @@ | |||||||
|   <a href="./README_tr.md">Türkçe</a> |   <a href="./README_tr.md">Türkçe</a> | ||||||
| </p> | </p> | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| </div> | </div> | ||||||
| 
 | 
 | ||||||
| </p> | </p> | ||||||
| @ -61,14 +59,14 @@ Az OpenIM egy szolgáltatási platform, amelyet kifejezetten a csevegés, az aud | |||||||
| 
 | 
 | ||||||
| Az **OpenIMSDK** egy **OpenIMServer** számára készült azonnali üzenetküldő SDK, amelyet kifejezetten ügyfélalkalmazásokba való beágyazáshoz hoztak létre. Fő jellemzői és moduljai a következők: | Az **OpenIMSDK** egy **OpenIMServer** számára készült azonnali üzenetküldő SDK, amelyet kifejezetten ügyfélalkalmazásokba való beágyazáshoz hoztak létre. Fő jellemzői és moduljai a következők: | ||||||
| 
 | 
 | ||||||
| + 🌟 Főbb jellemzők: | - 🌟 Főbb jellemzők: | ||||||
| 
 | 
 | ||||||
|   - 📦 Helyi raktár |   - 📦 Helyi raktár | ||||||
|   - 🔔 Hallgatói visszahívások |   - 🔔 Hallgatói visszahívások | ||||||
|   - 🛡️ API-csomagolás |   - 🛡️ API-csomagolás | ||||||
|   - 🌐 Kapcsolatkezelés |   - 🌐 Kapcsolatkezelés | ||||||
| 
 | 
 | ||||||
| + 📚 Fő modulok: | - 📚 Fő modulok: | ||||||
| 
 | 
 | ||||||
|   1. 🚀 Inicializálás és bejelentkezés |   1. 🚀 Inicializálás és bejelentkezés | ||||||
|   2. 👤 Felhasználókezelés |   2. 👤 Felhasználókezelés | ||||||
| @ -82,15 +80,15 @@ Golang használatával készült, és támogatja a többplatformos telepítést, | |||||||
| 
 | 
 | ||||||
| ## 🌐 Az OpenIMServerről | ## 🌐 Az OpenIMServerről | ||||||
| 
 | 
 | ||||||
| + **OpenIMServer** a következő jellemzőkkel rendelkezik: | - **OpenIMServer** a következő jellemzőkkel rendelkezik: | ||||||
|   - 🌐 Mikroszolgáltatási architektúra: Támogatja a fürt módot, beleértve az átjárót és több rpc szolgáltatást. |   - 🌐 Mikroszolgáltatási architektúra: Támogatja a fürt módot, beleértve az átjárót és több rpc szolgáltatást. | ||||||
|   - 🚀 Változatos telepítési módszerek: Támogatja a forráskódon, Kubernetesen vagy Dockeren keresztül történő telepítést. |   - 🚀 Változatos telepítési módszerek: Támogatja a forráskódon, Kubernetesen vagy Dockeren keresztül történő telepítést. | ||||||
|   - Hatalmas felhasználói bázis támogatása: Szuper nagy csoportok több százezer felhasználóval, több tízmillió felhasználóval és több milliárd üzenettel. |   - Hatalmas felhasználói bázis támogatása: Szuper nagy csoportok több százezer felhasználóval, több tízmillió felhasználóval és több milliárd üzenettel. | ||||||
| 
 | 
 | ||||||
| ### Továbbfejlesztett üzleti funkcionalitás: | ### Továbbfejlesztett üzleti funkcionalitás: | ||||||
| 
 | 
 | ||||||
| + **REST API**: Az OpenIMServer REST API-kat kínál az üzleti rendszerek számára, amelyek célja, hogy a vállalkozásokat több funkcióval ruházza fel, mint például csoportok létrehozása és push üzenetek küldése háttérfelületeken keresztül. | - **REST API**: Az OpenIMServer REST API-kat kínál az üzleti rendszerek számára, amelyek célja, hogy a vállalkozásokat több funkcióval ruházza fel, mint például csoportok létrehozása és push üzenetek küldése háttérfelületeken keresztül. | ||||||
| + **Webhooks**: Az OpenIMServer visszahívási lehetőségeket biztosít több üzleti forma kiterjesztéséhez. A visszahívás azt jelenti, hogy az OpenIMServer kérelmet küld az üzleti szervernek egy bizonyos esemény előtt vagy után, például visszahívásokat üzenet küldése előtt vagy után. | - **Webhooks**: Az OpenIMServer visszahívási lehetőségeket biztosít több üzleti forma kiterjesztéséhez. A visszahívás azt jelenti, hogy az OpenIMServer kérelmet küld az üzleti szervernek egy bizonyos esemény előtt vagy után, például visszahívásokat üzenet küldése előtt vagy után. | ||||||
| 
 | 
 | ||||||
| 👉 **[Tudj meg többet](https://docs.openim.io/guides/introduction/product)** | 👉 **[Tudj meg többet](https://docs.openim.io/guides/introduction/product)** | ||||||
| 
 | 
 | ||||||
| @ -100,7 +98,6 @@ Merüljön el az Open-IM-Server funkcióinak szívében az architektúra diagram | |||||||
| 
 | 
 | ||||||
|  |  | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## :rocket: Gyors indítás | ## :rocket: Gyors indítás | ||||||
| 
 | 
 | ||||||
| Számos platformot támogatunk. Íme a címek a gyors weboldali használathoz: | Számos platformot támogatunk. Íme a címek a gyors weboldali használathoz: | ||||||
| @ -109,10 +106,10 @@ Számos platformot támogatunk. Íme a címek a gyors weboldali használathoz: | |||||||
| 
 | 
 | ||||||
| 🤲 A felhasználói élmény megkönnyítése érdekében különféle telepítési megoldásokat kínálunk. Az alábbi listából választhatja ki a telepítési módot: | 🤲 A felhasználói élmény megkönnyítése érdekében különféle telepítési megoldásokat kínálunk. Az alábbi listából választhatja ki a telepítési módot: | ||||||
| 
 | 
 | ||||||
| + **[Forráskód-telepítési útmutató](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | - **[Forráskód-telepítési útmutató](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | ||||||
| + **[Docker telepítési útmutató](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | - **[Docker telepítési útmutató](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | ||||||
| + **[Kubernetes telepítési útmutató](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | - **[Kubernetes telepítési útmutató](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | ||||||
| + **[Mac fejlesztői telepítési útmutató](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | - **[Mac fejlesztői telepítési útmutató](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | ||||||
| 
 | 
 | ||||||
| ## :hammer_and_wrench: Az OpenIM fejlesztésének megkezdéséhez | ## :hammer_and_wrench: Az OpenIM fejlesztésének megkezdéséhez | ||||||
| 
 | 
 | ||||||
| @ -122,7 +119,7 @@ OpenIM Célunk egy felső szintű nyílt forráskódú közösség felépítése | |||||||
| 
 | 
 | ||||||
| Ha hozzá szeretne járulni ehhez az Open-IM-Server adattárhoz, kérjük, olvassa el [közreműködői dokumentációnkat](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md). | Ha hozzá szeretne járulni ehhez az Open-IM-Server adattárhoz, kérjük, olvassa el [közreműködői dokumentációnkat](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md). | ||||||
| 
 | 
 | ||||||
| Mielőtt elkezdené, győződjön meg arról, hogy a változtatásokra van-e igény. Erre a legjobb egy [új beszélgetés](https://github.com/openimsdk/open-im-server/discussions/new/choose) VAGY [Slack Communication](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q)létrehozása, vagy ha problémát talál, először [jelentse](https://github.com/openimsdk/open-im-server/issues/new/choose) first. | Mielőtt elkezdené, győződjön meg arról, hogy a változtatásokra van-e igény. Erre a legjobb egy [új beszélgetés](https://github.com/openimsdk/open-im-server/discussions/new/choose) VAGY [Slack Communication](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A)létrehozása, vagy ha problémát talál, először [jelentse](https://github.com/openimsdk/open-im-server/issues/new/choose) first. | ||||||
| 
 | 
 | ||||||
| - [OpenIM API referencia](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | - [OpenIM API referencia](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | ||||||
| - [OpenIM Bash naplózás](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | - [OpenIM Bash naplózás](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | ||||||
| @ -154,19 +151,18 @@ Mielőtt elkezdené, győződjön meg arról, hogy a változtatásokra van-e ig | |||||||
| - [A háttérrendszer kezelése és a telepítés figyelése](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | - [A háttérrendszer kezelése és a telepítés figyelése](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | ||||||
| - [Mac Developer Deployment Guide for OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | - [Mac Developer Deployment Guide for OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## :busts_in_silhouette: Közösség | ## :busts_in_silhouette: Közösség | ||||||
| 
 | 
 | ||||||
| + 📚 [OpenIM közösség](https://github.com/OpenIMSDK/community) | - 📚 [OpenIM közösség](https://github.com/OpenIMSDK/community) | ||||||
| + 💕 [OpenIM érdeklődési csoport](https://github.com/Openim-sigs) | - 💕 [OpenIM érdeklődési csoport](https://github.com/Openim-sigs) | ||||||
| + 🚀 [Csatlakozz a Slack közösségünkhöz](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | - 🚀 [Csatlakozz a Slack közösségünkhöz](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| + :eyes: [Csatlakozz a wechathez](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | - :eyes: [Csatlakozz a wechathez](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | ||||||
| 
 | 
 | ||||||
| ## :calendar: Közösségi Találkozók | ## :calendar: Közösségi Találkozók | ||||||
| 
 | 
 | ||||||
| Szeretnénk, ha bárki bekapcsolódna közösségünkbe és hozzájárulna kódunkhoz, ajándékokat és jutalmakat kínálunk, és szeretettel várjuk, hogy csatlakozzon hozzánk minden csütörtök este. | Szeretnénk, ha bárki bekapcsolódna közösségünkbe és hozzájárulna kódunkhoz, ajándékokat és jutalmakat kínálunk, és szeretettel várjuk, hogy csatlakozzon hozzánk minden csütörtök este. | ||||||
| 
 | 
 | ||||||
| Konferenciánk az [OpenIM Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) 🎯alatt van, akkor kereshet az Open-IM-Server folyamatban a csatlakozáshoz | Konferenciánk az [OpenIM Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) 🎯alatt van, akkor kereshet az Open-IM-Server folyamatban a csatlakozáshoz | ||||||
| 
 | 
 | ||||||
| A [GitHub-beszélgetések](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting)minden [kéthetente történő megbeszélésről](https://github.com/openimsdk/open-im-server/discussions/categories/meeting) jegyzeteket készítünk. A találkozók történeti feljegyzései, valamint az értekezletek visszajátszásai a [Google Dokumentumok :bookmark_tabs:](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing) webhelyen érhetők el. | A [GitHub-beszélgetések](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting)minden [kéthetente történő megbeszélésről](https://github.com/openimsdk/open-im-server/discussions/categories/meeting) jegyzeteket készítünk. A találkozók történeti feljegyzései, valamint az értekezletek visszajátszásai a [Google Dokumentumok :bookmark_tabs:](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing) webhelyen érhetők el. | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -12,12 +12,11 @@ | |||||||
| [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | ||||||
| [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | ||||||
| [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | ||||||
| [](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | [](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| [](https://www.bestpractices.dev/projects/8045) | [](https://www.bestpractices.dev/projects/8045) | ||||||
| [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | ||||||
| [](https://golang.org/) | [](https://golang.org/) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| <p align="center"> | <p align="center"> | ||||||
|   <a href="../../README.md">English</a> ·  |   <a href="../../README.md">English</a> ·  | ||||||
|   <a href="../../README_zh_CN.md">中文</a> ·  |   <a href="../../README_zh_CN.md">中文</a> ·  | ||||||
| @ -46,29 +45,25 @@ | |||||||
|   <a href="./README_tr.md">Türkçe</a> |   <a href="./README_tr.md">Türkçe</a> | ||||||
| </p> | </p> | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| </div> | </div> | ||||||
| 
 | 
 | ||||||
| </p> | </p> | ||||||
| 
 | 
 | ||||||
|  | ## Ⓜ️ OpenIM について | ||||||
| 
 | 
 | ||||||
| ## Ⓜ️ OpenIMについて | OpenIM は、アプリケーション内でチャット、音声通話、通知、AI チャットボットなどの通信機能を統合するために特別に設計されたサービスプラットフォームです。一連の強力な API と Webhooks を提供することで、開発者はアプリケーションに簡単にこれらの通信機能を統合できます。OpenIM 自体は独立したチャットアプリではなく、アプリケーションにサポートを提供し、豊富な通信機能を実現するプラットフォームです。以下の図は、AppServer、AppClient、OpenIMServer、OpenIMSDK 間の相互作用を示しています。 | ||||||
| 
 |  | ||||||
| OpenIMは、アプリケーション内でチャット、音声通話、通知、AIチャットボットなどの通信機能を統合するために特別に設計されたサービスプラットフォームです。一連の強力なAPIとWebhooksを提供することで、開発者はアプリケーションに簡単にこれらの通信機能を統合できます。OpenIM自体は独立したチャットアプリではなく、アプリケーションにサポートを提供し、豊富な通信機能を実現するプラットフォームです。以下の図は、AppServer、AppClient、OpenIMServer、OpenIMSDK間の相互作用を示しています。 |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
| 
 | 
 | ||||||
| ## 🚀 OpenIMSDKについて | ## 🚀 OpenIMSDK について | ||||||
| 
 | 
 | ||||||
|  **OpenIMSDK**は、**OpenIMServer**用に設計されたIM SDKで、クライアントアプリケーションに組み込むためのものです。主な機能とモジュールは以下の通りです: | **OpenIMSDK**は、**OpenIMServer**用に設計された IM SDK で、クライアントアプリケーションに組み込むためのものです。主な機能とモジュールは以下の通りです: | ||||||
| 
 | 
 | ||||||
| + 🌟 主な機能: | - 🌟 主な機能: | ||||||
| 
 | 
 | ||||||
|   - 📦  ローカルストレージ |   - 📦 ローカルストレージ | ||||||
|   - 🔔 リスナーコールバック |   - 🔔 リスナーコールバック | ||||||
|   - 🛡️ APIのラッピング |   - 🛡️ API のラッピング | ||||||
|   - 🌐 接続管理 |   - 🌐 接続管理 | ||||||
| 
 | 
 | ||||||
|   ## 📚 主なモジュール: |   ## 📚 主なモジュール: | ||||||
| @ -79,54 +74,54 @@ OpenIMは、アプリケーション内でチャット、音声通話、通知 | |||||||
|   4. 🤖 グループ機能 |   4. 🤖 グループ機能 | ||||||
|   5. 💬 会話処理 |   5. 💬 会話処理 | ||||||
| 
 | 
 | ||||||
| Golangを使用して構築され、クロスプラットフォームの導入をサポートし、すべてのプラットフォームで一貫したアクセス体験を提供します。 | Golang を使用して構築され、クロスプラットフォームの導入をサポートし、すべてのプラットフォームで一貫したアクセス体験を提供します。 | ||||||
| 
 | 
 | ||||||
| 👉 **[GO SDKを探索する](https://github.com/openimsdk/openim-sdk-core)** | 👉 **[GO SDK を探索する](https://github.com/openimsdk/openim-sdk-core)** | ||||||
| 
 | 
 | ||||||
| ## 🌐 OpenIMServerについて | ## 🌐 OpenIMServer について | ||||||
| 
 | 
 | ||||||
| + **OpenIMServer** には以下の特徴があります: | - **OpenIMServer** には以下の特徴があります: | ||||||
|   - 🌐 マイクロサービスアーキテクチャ:クラスターモードをサポートし、ゲートウェイ(gateway)と複数のrpcサービスを含みます。 |   - 🌐 マイクロサービスアーキテクチャ:クラスターモードをサポートし、ゲートウェイ(gateway)と複数の rpc サービスを含みます。 | ||||||
|   - 🚀 多様なデプロイメント方法:ソースコード、kubernetes、またはdockerでのデプロイメントをサポートします。 |   - 🚀 多様なデプロイメント方法:ソースコード、kubernetes、または docker でのデプロイメントをサポートします。 | ||||||
|   - 海量ユーザーサポート:十万人規模の超大型グループ、千万人のユーザー、および百億のメッセージ |   - 海量ユーザーサポート:十万人規模の超大型グループ、千万人のユーザー、および百億のメッセージ | ||||||
| 
 | 
 | ||||||
| ### 強化されたビジネス機能: | ### 強化されたビジネス機能: | ||||||
| 
 | 
 | ||||||
| + **REST API**:OpenIMServerは、ビジネスシステム用のREST APIを提供しており、ビジネスにさらに多くの機能を提供することを目指しています。たとえば、バックエンドインターフェースを通じてグループを作成したり、プッシュメッセージを送信したりするなどです。 | - **REST API**:OpenIMServer は、ビジネスシステム用の REST API を提供しており、ビジネスにさらに多くの機能を提供することを目指しています。たとえば、バックエンドインターフェースを通じてグループを作成したり、プッシュメッセージを送信したりするなどです。 | ||||||
| + **Webhooks**:OpenIMServerは、より多くのビジネス形態を拡張するためのコールバック機能を提供しています。コールバックとは、特定のイベントが発生する前後に、OpenIMServerがビジネスサーバーにリクエストを送信することを意味します。例えば、メッセージ送信の前後のコールバックなどです。 | - **Webhooks**:OpenIMServer は、より多くのビジネス形態を拡張するためのコールバック機能を提供しています。コールバックとは、特定のイベントが発生する前後に、OpenIMServer がビジネスサーバーにリクエストを送信することを意味します。例えば、メッセージ送信の前後のコールバックなどです。 | ||||||
| 
 | 
 | ||||||
| 👉 **[もっと詳しく知る](https://docs.openim.io/guides/introduction/product)** | 👉 **[もっと詳しく知る](https://docs.openim.io/guides/introduction/product)** | ||||||
| 
 | 
 | ||||||
| ## :building_construction: 全体のアーキテクチャ | ## :building_construction: 全体のアーキテクチャ | ||||||
| 
 | 
 | ||||||
| Open-IM-Serverの機能の核心に迫るために、アーキテクチャダイアグラムをご覧ください。 | Open-IM-Server の機能の核心に迫るために、アーキテクチャダイアグラムをご覧ください。 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
| 
 | 
 | ||||||
| ## :rocket: クイックスタート | ## :rocket: クイックスタート | ||||||
| 
 | 
 | ||||||
| iOS/Android/H5/PC/Webでのオンライン体験: | iOS/Android/H5/PC/Web でのオンライン体験: | ||||||
| 
 | 
 | ||||||
| 👉 **[OpenIM online demo](https://www.openim.io/zh/commercial)** | 👉 **[OpenIM online demo](https://www.openim.io/zh/commercial)** | ||||||
| 
 | 
 | ||||||
| 🤲 ユーザー体験を容易にするために、私たちは様々なデプロイメントソリューションを提供しています。以下のリストから、ご自身のデプロイメント方法を選択できます: | 🤲 ユーザー体験を容易にするために、私たちは様々なデプロイメントソリューションを提供しています。以下のリストから、ご自身のデプロイメント方法を選択できます: | ||||||
| 
 | 
 | ||||||
| + **[ソースコードデプロイメントガイド](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | - **[ソースコードデプロイメントガイド](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | ||||||
| + **[Docker デプロイメントガイド](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | - **[Docker デプロイメントガイド](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | ||||||
| + **[Kubernetes デプロイメントガイド](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | - **[Kubernetes デプロイメントガイド](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | ||||||
| + **[Mac 開発者向けデプロイメントガイド](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | - **[Mac 開発者向けデプロイメントガイド](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | ||||||
| 
 | 
 | ||||||
| ## :hammer_and_wrench: OpenIMの開発を始める | ## :hammer_and_wrench: OpenIM の開発を始める | ||||||
| 
 | 
 | ||||||
| [](https://vscode.dev/github/openimsdk/open-im-server) | [](https://vscode.dev/github/openimsdk/open-im-server) | ||||||
| 
 | 
 | ||||||
| OpenIM 私たちの目標は、トップレベルのオープンソースコミュニティを構築することです。[コミュニティリポジトリ](https://github.com/OpenIMSDK/community)には一連の基準があります。 | OpenIM 私たちの目標は、トップレベルのオープンソースコミュニティを構築することです。[コミュニティリポジトリ](https://github.com/OpenIMSDK/community)には一連の基準があります。 | ||||||
| 
 | 
 | ||||||
| このOpen-IM-Serverリポジトリに貢献したい場合は、[貢献者ドキュメントをお読みください](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md)。 | この Open-IM-Server リポジトリに貢献したい場合は、[貢献者ドキュメントをお読みください](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md)。 | ||||||
| 
 | 
 | ||||||
| 始める前に、変更に必要があることを確認してください。最良の方法は、[新しいディスカッション](https://github.com/openimsdk/open-im-server/discussions/new/choose)や[Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q)での通信を作成すること、または問題を発見した場合は、まずそれを[報告](https://github.com/openimsdk/open-im-server/issues/new/choose)することです。 | 始める前に、変更に必要があることを確認してください。最良の方法は、[新しいディスカッション](https://github.com/openimsdk/open-im-server/discussions/new/choose)や[Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A)での通信を作成すること、または問題を発見した場合は、まずそれを[報告](https://github.com/openimsdk/open-im-server/issues/new/choose)することです。 | ||||||
| 
 | 
 | ||||||
| - [OpenIM APIリファレンス](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | - [OpenIM API リファレンス](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | ||||||
| - [OpenIM Bash ロギング](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | - [OpenIM Bash ロギング](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | ||||||
| - [OpenIM CI/CD アクション](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/cicd-actions.md) | - [OpenIM CI/CD アクション](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/cicd-actions.md) | ||||||
| - [OpenIM コード規約](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/code-conventions.md) | - [OpenIM コード規約](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/code-conventions.md) | ||||||
| @ -149,37 +144,35 @@ OpenIM 私たちの目標は、トップレベルのオープンソースコミ | |||||||
| - [OpenIM オフラインデプロイメント](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/offline-deployment.md) | - [OpenIM オフラインデプロイメント](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/offline-deployment.md) | ||||||
| - [OpenIM Protoc ツール](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/protoc-tools.md) | - [OpenIM Protoc ツール](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/protoc-tools.md) | ||||||
| - [OpenIM テスティングガイド](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/test.md) | - [OpenIM テスティングガイド](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/test.md) | ||||||
| - [OpenIM ユーティリティGo](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/util-go.md) | - [OpenIM ユーティリティ Go](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/util-go.md) | ||||||
| - [OpenIM Makefile ユーティリティ](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/util-makefile.md) | - [OpenIM Makefile ユーティリティ](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/util-makefile.md) | ||||||
| - [OpenIM スクリプトユーティリティ](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/util-scripts.md) | - [OpenIM スクリプトユーティリティ](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/util-scripts.md) | ||||||
| - [OpenIM バージョニング](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/version.md) | - [OpenIM バージョニング](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/version.md) | ||||||
| - [バックエンド管理とモニターデプロイメント](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | - [バックエンド管理とモニターデプロイメント](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | ||||||
| - [OpenIM用Mac開発者デプロイメントガイド](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | - [OpenIM 用 Mac 開発者デプロイメントガイド](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | ||||||
| 
 |  | ||||||
| 
 | 
 | ||||||
| ## :busts_in_silhouette: コミュニティ | ## :busts_in_silhouette: コミュニティ | ||||||
| 
 | 
 | ||||||
| + 📚 [OpenIM コミュニティ](https://github.com/OpenIMSDK/community) | - 📚 [OpenIM コミュニティ](https://github.com/OpenIMSDK/community) | ||||||
| + 💕 [OpenIM 興味グループ](https://github.com/Openim-sigs) | - 💕 [OpenIM 興味グループ](https://github.com/Openim-sigs) | ||||||
| + 🚀 [私たちのSlackコミュニティに参加する](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | - 🚀 [私たちの Slack コミュニティに参加する](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| + :eyes: [私たちのWeChat(微信群)に参加する](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | - :eyes: [私たちの WeChat(微信群)に参加する](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | ||||||
| 
 | 
 | ||||||
| ## :calendar: コミュニティミーティング | ## :calendar: コミュニティミーティング | ||||||
| 
 | 
 | ||||||
| 私たちは、誰もがコミュニティに参加し、コードに貢献してもらいたいと考えています。私たちは、ギフトや報酬を提供し、毎週木曜日の夜に参加していただくことを歓迎します。 | 私たちは、誰もがコミュニティに参加し、コードに貢献してもらいたいと考えています。私たちは、ギフトや報酬を提供し、毎週木曜日の夜に参加していただくことを歓迎します。 | ||||||
| 
 | 
 | ||||||
| 私たちの会議は[OpenIM Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q)🎯で行われます。そこでOpen-IM-Serverパイプラインを検索して参加できます。 | 私たちの会議は[OpenIM Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A)🎯 で行われます。そこで Open-IM-Server パイプラインを検索して参加できます。 | ||||||
| 
 | 
 | ||||||
|  | 私たちは[隔週の会議](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting)のメモを[GitHub ディスカッション](https://github.com/openimsdk/open-im-server/discussions/categories/meeting)に記録しています。歴史的な会議のメモや会議のリプレイは[Google Docs📑](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing)で利用可能です。 | ||||||
| 
 | 
 | ||||||
| 私たちは[隔週の会議](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting)のメモを[GitHubディスカッション](https://github.com/openimsdk/open-im-server/discussions/categories/meeting)に記録しています。歴史的な会議のメモや会議のリプレイは[Google Docs📑](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing)で利用可能です。 | ## :eyes: OpenIM を使用している人たち | ||||||
| 
 | 
 | ||||||
| ## :eyes: OpenIMを使用している人たち | プロジェクトユーザーのリストについては、[ユーザーケーススタディ](https://github.com/OpenIMSDK/community/blob/main/ADOPTERS.md)ページをご覧ください。[コメント 📝](https://github.com/openimsdk/open-im-server/issues/379)を残して、あなたの使用例を共有することを躊躇しないでください。 | ||||||
| 
 |  | ||||||
| プロジェクトユーザーのリストについては、[ユーザーケーススタディ](https://github.com/OpenIMSDK/community/blob/main/ADOPTERS.md)ページをご覧ください。[コメント📝](https://github.com/openimsdk/open-im-server/issues/379)を残して、あなたの使用例を共有することを躊躇しないでください。 |  | ||||||
| 
 | 
 | ||||||
| ## :page_facing_up: ライセンス | ## :page_facing_up: ライセンス | ||||||
| 
 | 
 | ||||||
| OpenIMはApache 2.0ライセンスの下でライセンスされています。完全なライセンステキストについては、[LICENSE](https://github.com/openimsdk/open-im-server/tree/main/LICENSE)を参照してください。 | OpenIM は Apache 2.0 ライセンスの下でライセンスされています。完全なライセンステキストについては、[LICENSE](https://github.com/openimsdk/open-im-server/tree/main/LICENSE)を参照してください。 | ||||||
| 
 | 
 | ||||||
| このリポジトリに表示される[OpenIM](https://github.com/openimsdk/open-im-server)ロゴ、そのバリエーション、およびアニメーションバージョン([assets/logo](./assets/logo)および[assets/logo-gif](assets/logo-gif)ディレクトリ内)は、著作権法によって保護されています。 | このリポジトリに表示される[OpenIM](https://github.com/openimsdk/open-im-server)ロゴ、そのバリエーション、およびアニメーションバージョン([assets/logo](./assets/logo)および[assets/logo-gif](assets/logo-gif)ディレクトリ内)は、著作権法によって保護されています。 | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -12,12 +12,11 @@ | |||||||
| [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | ||||||
| [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | ||||||
| [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | ||||||
| [](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | [](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| [](https://www.bestpractices.dev/projects/8045) | [](https://www.bestpractices.dev/projects/8045) | ||||||
| [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | ||||||
| [](https://golang.org/) | [](https://golang.org/) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| <p align="center"> | <p align="center"> | ||||||
|   <a href="../../README.md">English</a> ·  |   <a href="../../README.md">English</a> ·  | ||||||
|   <a href="../../README_zh_CN.md">中文</a> ·  |   <a href="../../README_zh_CN.md">中文</a> ·  | ||||||
| @ -46,26 +45,21 @@ | |||||||
|   <a href="./README_tr.md">Türkçe</a> |   <a href="./README_tr.md">Türkçe</a> | ||||||
| </p> | </p> | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| </div> | </div> | ||||||
| 
 | 
 | ||||||
| </p> | </p> | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## Ⓜ️ OpenIM에 대하여 | ## Ⓜ️ OpenIM에 대하여 | ||||||
| 
 | 
 | ||||||
| OpenIM은 채팅, 오디오-비디오 통화, 알림 및 AI 챗봇을 애플리케이션에 통합하기 위해 특별히 설계된 서비스 플랫폼입니다. 이 플랫폼은 강력한 API와 웹훅을 제공하여 개발자가 이러한 상호작용 기능을 애플리케이션에 쉽게 통합할 수 있게 합니다. OpenIM은 독립 실행형 채팅 애플리케이션이 아니라, 다른 애플리케이션들이 풍부한 커뮤니케이션 기능을 달성할 수 있도록 지원하는 플랫폼으로서의 역할을 합니다. 다음 다이어그램은 AppServer, AppClient, OpenIMServer, 및 OpenIMSDK 간의 상호작용을 자세히 설명하기 위해 제시되었습니다. | OpenIM은 채팅, 오디오-비디오 통화, 알림 및 AI 챗봇을 애플리케이션에 통합하기 위해 특별히 설계된 서비스 플랫폼입니다. 이 플랫폼은 강력한 API와 웹훅을 제공하여 개발자가 이러한 상호작용 기능을 애플리케이션에 쉽게 통합할 수 있게 합니다. OpenIM은 독립 실행형 채팅 애플리케이션이 아니라, 다른 애플리케이션들이 풍부한 커뮤니케이션 기능을 달성할 수 있도록 지원하는 플랫폼으로서의 역할을 합니다. 다음 다이어그램은 AppServer, AppClient, OpenIMServer, 및 OpenIMSDK 간의 상호작용을 자세히 설명하기 위해 제시되었습니다. | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|  |  | ||||||
| 
 | 
 | ||||||
| ## 🚀 OpenIMSDK에 대하여 | ## 🚀 OpenIMSDK에 대하여 | ||||||
| 
 | 
 | ||||||
|  **OpenIMSDK**는**OpenIMServer**를 위해 특별히 제작된 IM SDK로, 클라이언트 애플리케이션 내에 내장하기 위해 설계되었습니다. 그 주요 기능 및 모듈은 다음과 같습니다: | **OpenIMSDK**는**OpenIMServer**를 위해 특별히 제작된 IM SDK로, 클라이언트 애플리케이션 내에 내장하기 위해 설계되었습니다. 그 주요 기능 및 모듈은 다음과 같습니다: | ||||||
| 
 | 
 | ||||||
| 
 | - 🌟 주요 기능: | ||||||
| + 🌟 주요 기능: |  | ||||||
| 
 | 
 | ||||||
|   - 📦 로컬 스토리지 |   - 📦 로컬 스토리지 | ||||||
|   - 🔔 리스너 콜백 |   - 🔔 리스너 콜백 | ||||||
| @ -86,15 +80,15 @@ OpenIM은 채팅, 오디오-비디오 통화, 알림 및 AI 챗봇을 애플리 | |||||||
| 
 | 
 | ||||||
| ## 🌐 OpenIMServer에 대하여 | ## 🌐 OpenIMServer에 대하여 | ||||||
| 
 | 
 | ||||||
| + **OpenIMServer** 는 다음과 같은 특성을 가지고 있습니다: | - **OpenIMServer** 는 다음과 같은 특성을 가지고 있습니다: | ||||||
|   - 🌐 마이크로서비스 아키텍처: 게이트웨이 및 다수의 rpc 서비스를 포함하는 클러스터 모드를 지원합니다. |   - 🌐 마이크로서비스 아키텍처: 게이트웨이 및 다수의 rpc 서비스를 포함하는 클러스터 모드를 지원합니다. | ||||||
|   - 🚀  다양한 배포 방법: 소스 코드, 쿠버네티스 또는 도커를 통한 배포를 지원합니다. |   - 🚀 다양한 배포 방법: 소스 코드, 쿠버네티스 또는 도커를 통한 배포를 지원합니다. | ||||||
|   - 대규모 사용자 기반 지원: 수십만 명의 사용자를 포함하는 초대형 그룹, 수천만 명의 사용자 및 수십억 건의 메시지를 지원합니다. |   - 대규모 사용자 기반 지원: 수십만 명의 사용자를 포함하는 초대형 그룹, 수천만 명의 사용자 및 수십억 건의 메시지를 지원합니다. | ||||||
| 
 | 
 | ||||||
| ### 강화된 비즈니스 기능: | ### 강화된 비즈니스 기능: | ||||||
| 
 | 
 | ||||||
| + **REST API**:OpenIMServer는 비즈니스 시스템을 위한 REST API를 제공하여, 백엔드 인터페이스를 통해 그룹 생성 및 푸시 메시지 전송과 같은 더 많은 기능을 비즈니스에 제공하기 위해 설계되었습니다. | - **REST API**:OpenIMServer는 비즈니스 시스템을 위한 REST API를 제공하여, 백엔드 인터페이스를 통해 그룹 생성 및 푸시 메시지 전송과 같은 더 많은 기능을 비즈니스에 제공하기 위해 설계되었습니다. | ||||||
| + **Webhooks**:OpenIMServer는 더 많은 비즈니스 형태를 확장할 수 있는 콜백 기능을 제공합니다. 콜백이란 메시지 전송 전후와 같은 특정 이벤트 전후에 OpenIMServer가 비즈니스 서버로 요청을 보내는 것을 의미합니다. | - **Webhooks**:OpenIMServer는 더 많은 비즈니스 형태를 확장할 수 있는 콜백 기능을 제공합니다. 콜백이란 메시지 전송 전후와 같은 특정 이벤트 전후에 OpenIMServer가 비즈니스 서버로 요청을 보내는 것을 의미합니다. | ||||||
| 
 | 
 | ||||||
| 👉 **[더 알아보기](https://docs.openim.io/guides/introduction/product)** | 👉 **[더 알아보기](https://docs.openim.io/guides/introduction/product)** | ||||||
| 
 | 
 | ||||||
| @ -112,10 +106,10 @@ Open-IM-Server의 기능의 핵심으로 들어가 우리의 아키텍처 다이 | |||||||
| 
 | 
 | ||||||
| 🤲 사용자 경험을 용이하게 하기 위해, 다양한 배포 솔루션을 제공합니다. 아래 목록에서 배포 방법을 선택할 수 있습니다: | 🤲 사용자 경험을 용이하게 하기 위해, 다양한 배포 솔루션을 제공합니다. 아래 목록에서 배포 방법을 선택할 수 있습니다: | ||||||
| 
 | 
 | ||||||
| + **[소스 코드 배포 가이드](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | - **[소스 코드 배포 가이드](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | ||||||
| + **[docker 배포 가이드](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | - **[docker 배포 가이드](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | ||||||
| + **[Kubernetes 배포 가이드](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | - **[Kubernetes 배포 가이드](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | ||||||
| + **[Mac 개발자 배포 가이드](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | - **[Mac 개발자 배포 가이드](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | ||||||
| 
 | 
 | ||||||
| ## :hammer_and_wrench: OpenIM 개발 시작하기 | ## :hammer_and_wrench: OpenIM 개발 시작하기 | ||||||
| 
 | 
 | ||||||
| @ -125,7 +119,7 @@ OpenIM의 목표는 최상위 수준의 오픈 소스 커뮤니티를 구축하 | |||||||
| 
 | 
 | ||||||
| 이 Open-IM-Server 리포지토리에 기여하고 싶다면, 우리의 [기여자 문서](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md)를 읽어주세요. | 이 Open-IM-Server 리포지토리에 기여하고 싶다면, 우리의 [기여자 문서](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md)를 읽어주세요. | ||||||
| 
 | 
 | ||||||
| 시작하기 전에, 변경 사항이 필요한지 확인해 주세요. 가장 좋은 방법은 [새로운 토론](https://github.com/openimsdk/open-im-server/discussions/new/choose)을 생성하거나 [Slack 통신을](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) 하거나, 문제를 발견했다면 먼저 [보고](https://github.com/openimsdk/open-im-server/issues/new/choose)하는 것입니다. | 시작하기 전에, 변경 사항이 필요한지 확인해 주세요. 가장 좋은 방법은 [새로운 토론](https://github.com/openimsdk/open-im-server/discussions/new/choose)을 생성하거나 [Slack 통신을](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) 하거나, 문제를 발견했다면 먼저 [보고](https://github.com/openimsdk/open-im-server/issues/new/choose)하는 것입니다. | ||||||
| 
 | 
 | ||||||
| - [OpenIM API 참조](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | - [OpenIM API 참조](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | ||||||
| - [OpenIM Bash 로깅](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | - [OpenIM Bash 로깅](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | ||||||
| @ -157,23 +151,21 @@ OpenIM의 목표는 최상위 수준의 오픈 소스 커뮤니티를 구축하 | |||||||
| - [백엔드 관리 및 모니터 배포](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | - [백엔드 관리 및 모니터 배포](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | ||||||
| - [맥 개발자 배포 가이드 for OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | - [맥 개발자 배포 가이드 for OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## :busts_in_silhouette: 커뮤니티 | ## :busts_in_silhouette: 커뮤니티 | ||||||
| 
 | 
 | ||||||
| + 📚 [OpenIM 커뮤니티](https://github.com/OpenIMSDK/community) | - 📚 [OpenIM 커뮤니티](https://github.com/OpenIMSDK/community) | ||||||
| + 💕 [OpenIM 관심 그룹](https://github.com/Openim-sigs) | - 💕 [OpenIM 관심 그룹](https://github.com/Openim-sigs) | ||||||
| + 🚀 [우리의 Slack 커뮤니티에 가입하기](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | - 🚀 [우리의 Slack 커뮤니티에 가입하기](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| + :eyes: [우리의 위챗(微信群)에 가입하기](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | - :eyes: [우리의 위챗(微信群)에 가입하기](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | ||||||
| 
 | 
 | ||||||
| ## :calendar: 커뮤니티 미팅 | ## :calendar: 커뮤니티 미팅 | ||||||
| 
 | 
 | ||||||
| 우리는 누구나 커뮤니티에 참여하고 코드를 기여할 수 있도록 하며, 선물과 보상을 제공하며, 매주 목요일 밤에 여러분을 환영합니다. | 우리는 누구나 커뮤니티에 참여하고 코드를 기여할 수 있도록 하며, 선물과 보상을 제공하며, 매주 목요일 밤에 여러분을 환영합니다. | ||||||
| 
 | 
 | ||||||
| 우리의 회의는 [OpenIM Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) 🎯에서 이루어지며, Open-IM-Server 파이프라인을 검색하여 참여할 수 있습니다. | 우리의 회의는 [OpenIM Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) 🎯에서 이루어지며, Open-IM-Server 파이프라인을 검색하여 참여할 수 있습니다. | ||||||
| 
 | 
 | ||||||
| 우리는 격주 회의의 메모를 [GitHub 토론](https://github.com/openimsdk/open-im-server/discussions/categories/meeting)에서 기록하며, 우리의 역사적 [회의](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting) 노트와 회의 재생은 [Google Docs 📑](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing)에서 이용할 수 있습니다. | 우리는 격주 회의의 메모를 [GitHub 토론](https://github.com/openimsdk/open-im-server/discussions/categories/meeting)에서 기록하며, 우리의 역사적 [회의](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting) 노트와 회의 재생은 [Google Docs 📑](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing)에서 이용할 수 있습니다. | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## :eyes: OpenIM을 사용하는 사람들 | ## :eyes: OpenIM을 사용하는 사람들 | ||||||
| 
 | 
 | ||||||
| 프로젝트 사용자 목록을 위한 우리의 [사용자 사례 연구](https://github.com/OpenIMSDK/community/blob/main/ADOPTERS.md) 페이지를 확인하세요. 사용 사례를 공유하고 싶다면 주저하지 말고 [📝코멘트](https://github.com/openimsdk/open-im-server/issues/379)를 남겨주세요. | 프로젝트 사용자 목록을 위한 우리의 [사용자 사례 연구](https://github.com/OpenIMSDK/community/blob/main/ADOPTERS.md) 페이지를 확인하세요. 사용 사례를 공유하고 싶다면 주저하지 말고 [📝코멘트](https://github.com/openimsdk/open-im-server/issues/379)를 남겨주세요. | ||||||
| @ -184,7 +176,6 @@ OpenIM은 Apache 2.0 라이선스에 따라 라이선스가 부여됩니다. 전 | |||||||
| 
 | 
 | ||||||
| 이 리포지토리 [OpenIM](https://github.com/openimsdk/open-im-server)에 표시된 OpenIM 로고, 그 변형 및 애니메이션 버전은 [assets/logo](../../assets/logo) 및 [assets/logo-gif](../../assets/logo-gif) 디렉토리 아래에 있으며, 저작권 법에 의해 보호됩니다. | 이 리포지토리 [OpenIM](https://github.com/openimsdk/open-im-server)에 표시된 OpenIM 로고, 그 변형 및 애니메이션 버전은 [assets/logo](../../assets/logo) 및 [assets/logo-gif](../../assets/logo-gif) 디렉토리 아래에 있으며, 저작권 법에 의해 보호됩니다. | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## 🔮 우리의 기여자들에게 감사합니다! | ## 🔮 우리의 기여자들에게 감사합니다! | ||||||
| 
 | 
 | ||||||
| <a href="https://github.com/openimsdk/open-im-server/graphs/contributors"> | <a href="https://github.com/openimsdk/open-im-server/graphs/contributors"> | ||||||
|  | |||||||
| @ -12,12 +12,11 @@ | |||||||
| [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | ||||||
| [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | ||||||
| [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | ||||||
| [](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | [](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| [](https://www.bestpractices.dev/projects/8045) | [](https://www.bestpractices.dev/projects/8045) | ||||||
| [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | ||||||
| [](https://golang.org/) | [](https://golang.org/) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| <p align="center"> | <p align="center"> | ||||||
|   <a href="../../README.md">English</a> ·  |   <a href="../../README.md">English</a> ·  | ||||||
|   <a href="../../README_zh_CN.md">中文</a> ·  |   <a href="../../README_zh_CN.md">中文</a> ·  | ||||||
| @ -46,25 +45,21 @@ | |||||||
|   <a href="./README_tr.md">Türkçe</a> |   <a href="./README_tr.md">Türkçe</a> | ||||||
| </p> | </p> | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| </div> | </div> | ||||||
| 
 | 
 | ||||||
| </p> | </p> | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## Ⓜ️ OpenIM Hakkında | ## Ⓜ️ OpenIM Hakkında | ||||||
| 
 | 
 | ||||||
| OpenIM, uygulamalara sohbet, sesli-görüntülü aramalar, bildirimler ve AI sohbet robotları entegre etmek için özel olarak tasarlanmış bir hizmet platformudur. Güçlü API'ler ve Webhook'lar sunarak, geliştiricilerin bu etkileşimli özellikleri uygulamalarına kolayca dahil etmelerini sağlar. OpenIM bağımsız bir sohbet uygulaması değildir, ancak zengin iletişim işlevselliği sağlama amacıyla diğer uygulamaları destekleyen bir platform olarak hizmet verir. Aşağıdaki diyagram, AppServer, AppClient, OpenIMServer ve OpenIMSDK arasındaki etkileşimi detaylandırmak için açıklar. | OpenIM, uygulamalara sohbet, sesli-görüntülü aramalar, bildirimler ve AI sohbet robotları entegre etmek için özel olarak tasarlanmış bir hizmet platformudur. Güçlü API'ler ve Webhook'lar sunarak, geliştiricilerin bu etkileşimli özellikleri uygulamalarına kolayca dahil etmelerini sağlar. OpenIM bağımsız bir sohbet uygulaması değildir, ancak zengin iletişim işlevselliği sağlama amacıyla diğer uygulamaları destekleyen bir platform olarak hizmet verir. Aşağıdaki diyagram, AppServer, AppClient, OpenIMServer ve OpenIMSDK arasındaki etkileşimi detaylandırmak için açıklar. | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|  |  | ||||||
| 
 | 
 | ||||||
| ## 🚀 OpenIMSDK Hakkında | ## 🚀 OpenIMSDK Hakkında | ||||||
| 
 | 
 | ||||||
|  **OpenIMSDK**, müşteri uygulamalarına gömülmek üzere özel olarak oluşturulan **OpenIMServer** için tasarlanmış bir IM SDK'sıdır. Ana özellikleri ve modülleri aşağıdaki gibidir: | **OpenIMSDK**, müşteri uygulamalarına gömülmek üzere özel olarak oluşturulan **OpenIMServer** için tasarlanmış bir IM SDK'sıdır. Ana özellikleri ve modülleri aşağıdaki gibidir: | ||||||
| 
 | 
 | ||||||
| + 🌟 Ana Özellikler: | - 🌟 Ana Özellikler: | ||||||
| 
 | 
 | ||||||
|   - 📦 Yerel depolama |   - 📦 Yerel depolama | ||||||
|   - 🔔 Dinleyici geri çağırmaları |   - 🔔 Dinleyici geri çağırmaları | ||||||
| @ -85,15 +80,15 @@ Golang kullanılarak inşa edilmiş ve tüm platformlarda tutarlı bir erişim d | |||||||
| 
 | 
 | ||||||
| ## 🌐 OpenIMServer Hakkında | ## 🌐 OpenIMServer Hakkında | ||||||
| 
 | 
 | ||||||
| + **OpenIMServer** aşağıdaki özelliklere sahiptir: | - **OpenIMServer** aşağıdaki özelliklere sahiptir: | ||||||
|   - 🌐 Mikroservis mimarisi: Bir kapı ve çoklu rpc servisleri içeren küme modunu destekler. |   - 🌐 Mikroservis mimarisi: Bir kapı ve çoklu rpc servisleri içeren küme modunu destekler. | ||||||
|   - 🚀 Çeşitli dağıtım yöntemleri: Kaynak kodu, Kubernetes veya Docker aracılığıyla dağıtımı destekler. |   - 🚀 Çeşitli dağıtım yöntemleri: Kaynak kodu, Kubernetes veya Docker aracılığıyla dağıtımı destekler. | ||||||
|   - Büyük kullanıcı tabanı desteği: Yüz binlerce kullanıcısı olan süper büyük gruplar, on milyonlarca kullanıcı ve milyarlarca mesaj. |   - Büyük kullanıcı tabanı desteği: Yüz binlerce kullanıcısı olan süper büyük gruplar, on milyonlarca kullanıcı ve milyarlarca mesaj. | ||||||
| 
 | 
 | ||||||
| ### Geliştirilmiş İşlevsellik: | ### Geliştirilmiş İşlevsellik: | ||||||
| 
 | 
 | ||||||
| + **REST API**:OpenIMServer, işletmeleri gruplar oluşturma ve arka plan arayüzleri aracılığıyla itme mesajları gönderme gibi daha fazla işlevsellikle güçlendirmeyi amaçlayan iş sistemleri için REST API'leri sunar. | - **REST API**:OpenIMServer, işletmeleri gruplar oluşturma ve arka plan arayüzleri aracılığıyla itme mesajları gönderme gibi daha fazla işlevsellikle güçlendirmeyi amaçlayan iş sistemleri için REST API'leri sunar. | ||||||
| + **Webhooks**:OpenIMServer, daha fazla iş formunu genişletme yetenekleri sağlayan geri çağırma özellikleri sunar. Geri çağırma, OpenIMServer'ın belirli bir olaydan önce veya sonra, örneğin bir mesaj göndermeden önce veya sonra iş sunucusuna bir istek göndermesi anlamına gelir. | - **Webhooks**:OpenIMServer, daha fazla iş formunu genişletme yetenekleri sağlayan geri çağırma özellikleri sunar. Geri çağırma, OpenIMServer'ın belirli bir olaydan önce veya sonra, örneğin bir mesaj göndermeden önce veya sonra iş sunucusuna bir istek göndermesi anlamına gelir. | ||||||
| 
 | 
 | ||||||
| 👉 **[Daha fazla bilgi edinin](https://docs.openim.io/guides/introduction/product)** | 👉 **[Daha fazla bilgi edinin](https://docs.openim.io/guides/introduction/product)** | ||||||
| 
 | 
 | ||||||
| @ -111,10 +106,10 @@ Birçok platformu destekliyoruz. Web tarafında hızlı deneyim için adresler  | |||||||
| 
 | 
 | ||||||
| 🤲 Kullanıcı deneyimini kolaylaştırmak için çeşitli dağıtım çözümleri sunuyoruz. Aşağıdaki listeden dağıtım yönteminizi seçebilirsiniz: | 🤲 Kullanıcı deneyimini kolaylaştırmak için çeşitli dağıtım çözümleri sunuyoruz. Aşağıdaki listeden dağıtım yönteminizi seçebilirsiniz: | ||||||
| 
 | 
 | ||||||
| + **[Kaynak Kodu Dağıtım Kılavuzu](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | - **[Kaynak Kodu Dağıtım Kılavuzu](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | ||||||
| + **[Docker Dağıtım Kılavuzu](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | - **[Docker Dağıtım Kılavuzu](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | ||||||
| + **[Kubernetes Dağıtım Kılavuzu](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | - **[Kubernetes Dağıtım Kılavuzu](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | ||||||
| + **[Mac Geliştirici Dağıtım Kılavuzu](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | - **[Mac Geliştirici Dağıtım Kılavuzu](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | ||||||
| 
 | 
 | ||||||
| ## :hammer_and_wrench: OpenIM Geliştirmeye Başlamak | ## :hammer_and_wrench: OpenIM Geliştirmeye Başlamak | ||||||
| 
 | 
 | ||||||
| @ -124,7 +119,7 @@ OpenIM Amacımız, üst düzey bir açık kaynak topluluğu oluşturmaktır. [To | |||||||
| 
 | 
 | ||||||
| Bu Open-IM-Server deposuna katkıda bulunmak istiyorsanız, lütfen katkıda bulunanlar için [dokümantasyonumuzu](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md) okuyun. | Bu Open-IM-Server deposuna katkıda bulunmak istiyorsanız, lütfen katkıda bulunanlar için [dokümantasyonumuzu](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md) okuyun. | ||||||
| 
 | 
 | ||||||
| Başlamadan önce, lütfen değişikliklerinizin talep edildiğinden emin olun. Bunun için en iyisi, [yeni bir tartışma OLUŞTURMAK](https://github.com/openimsdk/open-im-server/discussions/new/choose) veya [Slack İletişimi](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) kurmak, ya da bir sorun bulursanız, önce bunu [rapor](https://github.com/openimsdk/open-im-server/issues/new/choose) etmektir. | Başlamadan önce, lütfen değişikliklerinizin talep edildiğinden emin olun. Bunun için en iyisi, [yeni bir tartışma OLUŞTURMAK](https://github.com/openimsdk/open-im-server/discussions/new/choose) veya [Slack İletişimi](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) kurmak, ya da bir sorun bulursanız, önce bunu [rapor](https://github.com/openimsdk/open-im-server/issues/new/choose) etmektir. | ||||||
| 
 | 
 | ||||||
| - [OpenIM API Referansı](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | - [OpenIM API Referansı](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | ||||||
| - [OpenIM Bash Günlüğü](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | - [OpenIM Bash Günlüğü](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | ||||||
| @ -156,19 +151,18 @@ Başlamadan önce, lütfen değişikliklerinizin talep edildiğinden emin olun. | |||||||
| - [Arka uç yönetimi ve izleme dağıtımı](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | - [Arka uç yönetimi ve izleme dağıtımı](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | ||||||
| - [Mac Geliştirici Dağıtım Kılavuzu for OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | - [Mac Geliştirici Dağıtım Kılavuzu for OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## :busts_in_silhouette: Topluluk | ## :busts_in_silhouette: Topluluk | ||||||
| 
 | 
 | ||||||
| + 📚 [OpenIM Topluluğu](https://github.com/OpenIMSDK/community) | - 📚 [OpenIM Topluluğu](https://github.com/OpenIMSDK/community) | ||||||
| + 💕 [OpenIM İlgi Grubu](https://github.com/Openim-sigs) | - 💕 [OpenIM İlgi Grubu](https://github.com/Openim-sigs) | ||||||
| + 🚀 [Slack topluluğumuza katılın](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | - 🚀 [Slack topluluğumuza katılın](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| + :eyes: [Wechat grubumuza katılın (微信群)](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | - :eyes: [Wechat grubumuza katılın (微信群)](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | ||||||
| 
 | 
 | ||||||
| ## :calendar: Topluluk Toplantıları | ## :calendar: Topluluk Toplantıları | ||||||
| 
 | 
 | ||||||
| Topluluğumuza herkesin katılmasını ve kod katkısında bulunmasını istiyoruz, hediyeler ve ödüller sunuyoruz ve sizi her Perşembe gecesi bize katılmaya davet ediyoruz. | Topluluğumuza herkesin katılmasını ve kod katkısında bulunmasını istiyoruz, hediyeler ve ödüller sunuyoruz ve sizi her Perşembe gecesi bize katılmaya davet ediyoruz. | ||||||
| 
 | 
 | ||||||
| Konferansımız [OpenIM Slack'te](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) 🎯, ardından Open-IM-Server boru hattını arayıp katılabilirsiniz. | Konferansımız [OpenIM Slack'te](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) 🎯, ardından Open-IM-Server boru hattını arayıp katılabilirsiniz. | ||||||
| 
 | 
 | ||||||
| İki haftada bir yapılan toplantının [notlarını](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting) [GitHub tartışmalarında alıyoruz](https://github.com/openimsdk/open-im-server/discussions/categories/meeting), Tarihi toplantı notlarımız ve toplantıların tekrarları [Google Docs'ta](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing) 📑 mevcut. | İki haftada bir yapılan toplantının [notlarını](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting) [GitHub tartışmalarında alıyoruz](https://github.com/openimsdk/open-im-server/discussions/categories/meeting), Tarihi toplantı notlarımız ve toplantıların tekrarları [Google Docs'ta](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing) 📑 mevcut. | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -12,12 +12,11 @@ | |||||||
| [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | ||||||
| [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | ||||||
| [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | ||||||
| [](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | [](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| [](https://www.bestpractices.dev/projects/8045) | [](https://www.bestpractices.dev/projects/8045) | ||||||
| [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | ||||||
| [](https://golang.org/) | [](https://golang.org/) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| <p align="center"> | <p align="center"> | ||||||
|   <a href="../../README.md">English</a> ·  |   <a href="../../README.md">English</a> ·  | ||||||
|   <a href="../../README_zh_CN.md">中文</a> ·  |   <a href="../../README_zh_CN.md">中文</a> ·  | ||||||
| @ -46,7 +45,6 @@ | |||||||
|   <a href="./README_tr.md">Türkçe</a> |   <a href="./README_tr.md">Türkçe</a> | ||||||
| </p> | </p> | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| </div> | </div> | ||||||
| 
 | 
 | ||||||
| </p> | </p> | ||||||
| @ -61,14 +59,14 @@ OpenIM — це сервісна платформа, спеціально роз | |||||||
| 
 | 
 | ||||||
| **OpenIMSDK** – це пакет IM SDK, розроблений для **OpenIMServer**, створений спеціально для вбудовування в клієнтські програми. Його основні функції та модулі такі: | **OpenIMSDK** – це пакет IM SDK, розроблений для **OpenIMServer**, створений спеціально для вбудовування в клієнтські програми. Його основні функції та модулі такі: | ||||||
| 
 | 
 | ||||||
| + 🌟 Основні характеристики: | - 🌟 Основні характеристики: | ||||||
| 
 | 
 | ||||||
|   - 📦 Локальне сховище |   - 📦 Локальне сховище | ||||||
|   - 🔔 Зворотні виклики слухача |   - 🔔 Зворотні виклики слухача | ||||||
|   - 🛡️ Обгортка API |   - 🛡️ Обгортка API | ||||||
|   - 🌐 Керування підключенням |   - 🌐 Керування підключенням | ||||||
| 
 | 
 | ||||||
| + 📚 Основні модулі: | - 📚 Основні модулі: | ||||||
| 
 | 
 | ||||||
|   1. 🚀 Ініціалізація та вхід |   1. 🚀 Ініціалізація та вхід | ||||||
|   2. 👤 Керування користувачами |   2. 👤 Керування користувачами | ||||||
| @ -82,15 +80,15 @@ OpenIM — це сервісна платформа, спеціально роз | |||||||
| 
 | 
 | ||||||
| ## 🌐 Про OpenIMServer | ## 🌐 Про OpenIMServer | ||||||
| 
 | 
 | ||||||
| + **OpenIMServer** має такі характеристики: | - **OpenIMServer** має такі характеристики: | ||||||
|   - 🌐 Архітектура мікросервісу: підтримує режим кластера, включаючи шлюз і кілька служб rpc. |   - 🌐 Архітектура мікросервісу: підтримує режим кластера, включаючи шлюз і кілька служб rpc. | ||||||
|   - 🚀 Різноманітні методи розгортання: підтримує розгортання через вихідний код, Kubernetes або Docker. |   - 🚀 Різноманітні методи розгортання: підтримує розгортання через вихідний код, Kubernetes або Docker. | ||||||
|   - Підтримка величезної бази користувачів: надвеликі групи із сотнями тисяч користувачів, десятками мільйонів користувачів і мільярдами повідомлень. |   - Підтримка величезної бази користувачів: надвеликі групи із сотнями тисяч користувачів, десятками мільйонів користувачів і мільярдами повідомлень. | ||||||
| 
 | 
 | ||||||
| ### Розширена бізнес-функціональність: | ### Розширена бізнес-функціональність: | ||||||
| 
 | 
 | ||||||
| + **REST API**: OpenIMServer пропонує REST API для бізнес-систем, спрямованих на надання компаніям додаткових можливостей, таких як створення груп і надсилання push-повідомлень через серверні інтерфейси. | - **REST API**: OpenIMServer пропонує REST API для бізнес-систем, спрямованих на надання компаніям додаткових можливостей, таких як створення груп і надсилання push-повідомлень через серверні інтерфейси. | ||||||
| + **Веб-перехоплення**: OpenIMServer надає можливості зворотного виклику, щоб розширити більше бізнес-форм. Зворотний виклик означає, що OpenIMServer надсилає запит на бізнес-сервер до або після певної події, як зворотні виклики до або після надсилання повідомлення. | - **Веб-перехоплення**: OpenIMServer надає можливості зворотного виклику, щоб розширити більше бізнес-форм. Зворотний виклик означає, що OpenIMServer надсилає запит на бізнес-сервер до або після певної події, як зворотні виклики до або після надсилання повідомлення. | ||||||
| 
 | 
 | ||||||
| 👉 **[Докладніше](https://docs.openim.io/guides/introduction/product)** | 👉 **[Докладніше](https://docs.openim.io/guides/introduction/product)** | ||||||
| 
 | 
 | ||||||
| @ -100,7 +98,6 @@ OpenIM — це сервісна платформа, спеціально роз | |||||||
| 
 | 
 | ||||||
|  |  | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## :rocket: Швидкий початок | ## :rocket: Швидкий початок | ||||||
| 
 | 
 | ||||||
| Ми підтримуємо багато платформ. Ось адреси для швидкого використання веб-сайту: | Ми підтримуємо багато платформ. Ось адреси для швидкого використання веб-сайту: | ||||||
| @ -109,10 +106,10 @@ OpenIM — це сервісна платформа, спеціально роз | |||||||
| 
 | 
 | ||||||
| 🤲 Щоб полегшити роботу користувача, ми пропонуємо різні рішення для розгортання. Ви можете вибрати спосіб розгортання зі списку нижче: | 🤲 Щоб полегшити роботу користувача, ми пропонуємо різні рішення для розгортання. Ви можете вибрати спосіб розгортання зі списку нижче: | ||||||
| 
 | 
 | ||||||
| + **[Посібник із розгортання вихідного коду](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | - **[Посібник із розгортання вихідного коду](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | ||||||
| + **[Посібник із розгортання Docker](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | - **[Посібник із розгортання Docker](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | ||||||
| + **[Посібник із розгортання Kubernetes](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | - **[Посібник із розгортання Kubernetes](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | ||||||
| + **[Посібник із розгортання розробника Mac](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | - **[Посібник із розгортання розробника Mac](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | ||||||
| 
 | 
 | ||||||
| ## :hammer_and_wrench: Щоб розпочати розробку OpenIM | ## :hammer_and_wrench: Щоб розпочати розробку OpenIM | ||||||
| 
 | 
 | ||||||
| @ -122,7 +119,7 @@ OpenIM. Наша мета — побудувати спільноту з від | |||||||
| 
 | 
 | ||||||
| Якщо ви хочете внести свій внесок у це сховище Open-IM-Server, прочитайте нашу [документацію для учасників](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md). | Якщо ви хочете внести свій внесок у це сховище Open-IM-Server, прочитайте нашу [документацію для учасників](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md). | ||||||
| 
 | 
 | ||||||
| Перш ніж почати, переконайтеся, що ваші зміни затребувані. Найкраще для цього створити [нове обговорення](https://github.com/openimsdk/open-im-server/discussions/new/choose) АБО [Нездійснене спілкування](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q)або, якщо ви виявите проблему, спершу [повідомити про неї](https://github.com/openimsdk/open-im-server/issues/new/choose). | Перш ніж почати, переконайтеся, що ваші зміни затребувані. Найкраще для цього створити [нове обговорення](https://github.com/openimsdk/open-im-server/discussions/new/choose) АБО [Нездійснене спілкування](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A)або, якщо ви виявите проблему, спершу [повідомити про неї](https://github.com/openimsdk/open-im-server/issues/new/choose). | ||||||
| 
 | 
 | ||||||
| - [Довідка щодо API OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | - [Довідка щодо API OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | ||||||
| - [Ведення журналу OpenIM Bash](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | - [Ведення журналу OpenIM Bash](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | ||||||
| @ -154,19 +151,18 @@ OpenIM. Наша мета — побудувати спільноту з від | |||||||
| - [Керування серверною частиною та моніторинг розгортання](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | - [Керування серверною частиною та моніторинг розгортання](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | ||||||
| - [Посібник із розгортання розробника Mac для OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | - [Посібник із розгортання розробника Mac для OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## :busts_in_silhouette: Спільнота | ## :busts_in_silhouette: Спільнота | ||||||
| 
 | 
 | ||||||
| + 📚 [Спільнота OpenIM](https://github.com/OpenIMSDK/community) | - 📚 [Спільнота OpenIM](https://github.com/OpenIMSDK/community) | ||||||
| + 💕 [Група інтересів OpenIM](https://github.com/Openim-sigs) | - 💕 [Група інтересів OpenIM](https://github.com/Openim-sigs) | ||||||
| + 🚀 [Приєднайтеся до нашої спільноти Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | - 🚀 [Приєднайтеся до нашої спільноти Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| + :eyes: [Приєднайтеся до нашого wechat](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | - :eyes: [Приєднайтеся до нашого wechat](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | ||||||
| 
 | 
 | ||||||
| ## :calendar: Збори громади | ## :calendar: Збори громади | ||||||
| 
 | 
 | ||||||
| Ми хочемо, щоб будь-хто долучився до нашої спільноти та додав код, ми пропонуємо подарунки та нагороди, і ми запрошуємо вас приєднатися до нас щочетверга ввечері. | Ми хочемо, щоб будь-хто долучився до нашої спільноти та додав код, ми пропонуємо подарунки та нагороди, і ми запрошуємо вас приєднатися до нас щочетверга ввечері. | ||||||
| 
 | 
 | ||||||
| Наша конференція знаходиться в [OpenIM Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) 🎯, тоді ви можете шукати конвеєр Open-IM-Server, щоб приєднатися. | Наша конференція знаходиться в [OpenIM Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) 🎯, тоді ви можете шукати конвеєр Open-IM-Server, щоб приєднатися. | ||||||
| 
 | 
 | ||||||
| Ми робимо нотатки про кожну [двотижневу зустріч](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting)в [обговореннях GitHub](https://github.com/openimsdk/open-im-server/discussions/categories/meeting). Наші історичні нотатки зустрічей, а також повтори зустрічей доступні в[Google Docs :bookmark_tabs:](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing). | Ми робимо нотатки про кожну [двотижневу зустріч](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting)в [обговореннях GitHub](https://github.com/openimsdk/open-im-server/discussions/categories/meeting). Наші історичні нотатки зустрічей, а також повтори зустрічей доступні в[Google Docs :bookmark_tabs:](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing). | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -12,12 +12,11 @@ | |||||||
| [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | [](https://goreportcard.com/report/github.com/openimsdk/open-im-server) | ||||||
| [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | [](https://pkg.go.dev/github.com/openimsdk/open-im-server/v3) | ||||||
| [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | [](https://github.com/openimsdk/open-im-server/blob/main/LICENSE) | ||||||
| [](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | [](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| [](https://www.bestpractices.dev/projects/8045) | [](https://www.bestpractices.dev/projects/8045) | ||||||
| [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | [](https://github.com/openimsdk/open-im-server/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc+label%3A%22good+first+issue%22) | ||||||
| [](https://golang.org/) | [](https://golang.org/) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| <p align="center"> | <p align="center"> | ||||||
|   <a href="../../README.md">English</a> ·  |   <a href="../../README.md">English</a> ·  | ||||||
|   <a href="../../README_zh_CN.md">中文</a> ·  |   <a href="../../README_zh_CN.md">中文</a> ·  | ||||||
| @ -46,7 +45,6 @@ | |||||||
|   <a href="./README_tr.md">Türkçe</a> |   <a href="./README_tr.md">Türkçe</a> | ||||||
| </p> | </p> | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| </div> | </div> | ||||||
| 
 | 
 | ||||||
| </p> | </p> | ||||||
| @ -61,14 +59,14 @@ OpenIM là một nền tảng dịch vụ được thiết kế đặc biệt ch | |||||||
| 
 | 
 | ||||||
| **OpenIMSDK** là một SDK IM được thiết kế cho **OpenIMServer**, được tạo ra đặc biệt để nhúng vào các ứng dụng khách. Các tính năng chính và các mô-đun của nó như sau: | **OpenIMSDK** là một SDK IM được thiết kế cho **OpenIMServer**, được tạo ra đặc biệt để nhúng vào các ứng dụng khách. Các tính năng chính và các mô-đun của nó như sau: | ||||||
| 
 | 
 | ||||||
| + 🌟 Các Tính Năng Chính: | - 🌟 Các Tính Năng Chính: | ||||||
| 
 | 
 | ||||||
|   - 📦 Lưu trữ cục bộ |   - 📦 Lưu trữ cục bộ | ||||||
|   - 🔔 Gọi lại sự kiện (Listener callbacks) |   - 🔔 Gọi lại sự kiện (Listener callbacks) | ||||||
|   - 🛡️ Bọc API |   - 🛡️ Bọc API | ||||||
|   - 🌐 Quản lý kết nối |   - 🌐 Quản lý kết nối | ||||||
| 
 | 
 | ||||||
| + 📚 Các Mô-đun Chính: | - 📚 Các Mô-đun Chính: | ||||||
| 
 | 
 | ||||||
|   1. 🚀 Khởi tạo và Đăng nhập |   1. 🚀 Khởi tạo và Đăng nhập | ||||||
|   2. 👤 Quản lý Người dùng |   2. 👤 Quản lý Người dùng | ||||||
| @ -82,25 +80,24 @@ Nó được xây dựng bằng Golang và hỗ trợ triển khai đa nền t | |||||||
| 
 | 
 | ||||||
| ## 🌐 Về OpenIMServer | ## 🌐 Về OpenIMServer | ||||||
| 
 | 
 | ||||||
| + **OpenIMServer** có những đặc điểm sau: | - **OpenIMServer** có những đặc điểm sau: | ||||||
|   - 🌐 Kiến trúc vi dịch vụ: Hỗ trợ chế độ cluster, bao gồm một gateway và nhiều dịch vụ rpc. |   - 🌐 Kiến trúc vi dịch vụ: Hỗ trợ chế độ cluster, bao gồm một gateway và nhiều dịch vụ rpc. | ||||||
|   - 🚀 Phương pháp triển khai đa dạng: Hỗ trợ triển khai qua mã nguồn, Kubernetes hoặc Docker. |   - 🚀 Phương pháp triển khai đa dạng: Hỗ trợ triển khai qua mã nguồn, Kubernetes hoặc Docker. | ||||||
|   - Hỗ trợ cho cơ sở người dùng lớn: Nhóm siêu lớn với hàng trăm nghìn người dùng, hàng chục triệu người dùng và hàng tỷ tin nhắn. |   - Hỗ trợ cho cơ sở người dùng lớn: Nhóm siêu lớn với hàng trăm nghìn người dùng, hàng chục triệu người dùng và hàng tỷ tin nhắn. | ||||||
| 
 | 
 | ||||||
| ### Tăng cường Chức năng Kinh doanh: | ### Tăng cường Chức năng Kinh doanh: | ||||||
| 
 | 
 | ||||||
| + **REST API**: OpenIMServer cung cấp REST APIs cho các hệ thống kinh doanh, nhằm tăng cường khả năng cho doanh nghiệp với nhiều chức năng hơn, như tạo nhóm và gửi tin nhắn đẩy qua giao diện backend. | - **REST API**: OpenIMServer cung cấp REST APIs cho các hệ thống kinh doanh, nhằm tăng cường khả năng cho doanh nghiệp với nhiều chức năng hơn, như tạo nhóm và gửi tin nhắn đẩy qua giao diện backend. | ||||||
| + **Webhooks**: OpenIMServer cung cấp khả năng gọi lại để mở rộng thêm hình thức kinh doanh. Một gọi lại có nghĩa là OpenIMServer gửi một yêu cầu đến máy chủ kinh doanh trước hoặc sau một sự kiện nhất định, giống như gọi lại trước hoặc sau khi gửi một tin nhắn. | - **Webhooks**: OpenIMServer cung cấp khả năng gọi lại để mở rộng thêm hình thức kinh doanh. Một gọi lại có nghĩa là OpenIMServer gửi một yêu cầu đến máy chủ kinh doanh trước hoặc sau một sự kiện nhất định, giống như gọi lại trước hoặc sau khi gửi một tin nhắn. | ||||||
| 
 | 
 | ||||||
| 👉 **[Learn more](https://docs.openim.io/guides/introduction/product)** | 👉 **[Learn more](https://docs.openim.io/guides/introduction/product)** | ||||||
| 
 | 
 | ||||||
| ## :building_construction: Kiến trúc tổng thể | ## :building_construction: Kiến trúc tổng thể | ||||||
| 
 | 
 | ||||||
|  Làm sâu sắc vào trái tim của chức năng Open-IM-Server với sơ đồ kiến trúc của chúng tôi. | Làm sâu sắc vào trái tim của chức năng Open-IM-Server với sơ đồ kiến trúc của chúng tôi. | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## :rocket: Bắt đầu nhanh | ## :rocket: Bắt đầu nhanh | ||||||
| 
 | 
 | ||||||
| Chúng tôi hỗ trợ nhiều nền tảng. Dưới đây là các địa chỉ để trải nghiệm nhanh trên phía web: | Chúng tôi hỗ trợ nhiều nền tảng. Dưới đây là các địa chỉ để trải nghiệm nhanh trên phía web: | ||||||
| @ -109,12 +106,12 @@ Chúng tôi hỗ trợ nhiều nền tảng. Dưới đây là các địa chỉ | |||||||
| 
 | 
 | ||||||
| 🤲 Để tạo thuận lợi cho trải nghiệm người dùng, chúng tôi cung cấp các giải pháp triển khai đa dạng. Bạn có thể chọn phương thức triển khai từ danh sách dưới đây: | 🤲 Để tạo thuận lợi cho trải nghiệm người dùng, chúng tôi cung cấp các giải pháp triển khai đa dạng. Bạn có thể chọn phương thức triển khai từ danh sách dưới đây: | ||||||
| 
 | 
 | ||||||
| + **[Hướng dẫn Triển khai Mã Nguồn](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | - **[Hướng dẫn Triển khai Mã Nguồn](https://docs.openim.io/guides/gettingStarted/imSourceCodeDeployment)** | ||||||
| + **[Hướng dẫn Triển khai Docker](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | - **[Hướng dẫn Triển khai Docker](https://docs.openim.io/guides/gettingStarted/dockerCompose)** | ||||||
| + **[Hướng dẫn Triển khai Kubernetes](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | - **[Hướng dẫn Triển khai Kubernetes](https://docs.openim.io/guides/gettingStarted/k8s-deployment)** | ||||||
| + **[Hướng dẫn Triển khai cho Nhà Phát Triển Mac](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | - **[Hướng dẫn Triển khai cho Nhà Phát Triển Mac](https://docs.openim.io/guides/gettingstarted/mac-deployment-guide)** | ||||||
| 
 | 
 | ||||||
| ## :hammer_and_wrench:  Để Bắt Đầu Phát Triển OpenIM | ## :hammer_and_wrench: Để Bắt Đầu Phát Triển OpenIM | ||||||
| 
 | 
 | ||||||
| [](https://vscode.dev/github/openimsdk/open-im-server) | [](https://vscode.dev/github/openimsdk/open-im-server) | ||||||
| 
 | 
 | ||||||
| @ -122,8 +119,7 @@ Mục tiêu của OpenIM là xây dựng một cộng đồng mã nguồn mở c | |||||||
| 
 | 
 | ||||||
| Nếu bạn muốn đóng góp cho kho lưu trữ Open-IM-Server này, vui lòng đọc [tài liệu hướng dẫn cho người đóng góp](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md). | Nếu bạn muốn đóng góp cho kho lưu trữ Open-IM-Server này, vui lòng đọc [tài liệu hướng dẫn cho người đóng góp](https://github.com/openimsdk/open-im-server/blob/main/CONTRIBUTING.md). | ||||||
| 
 | 
 | ||||||
| 
 | Trước khi bạn bắt đầu, hãy chắc chắn rằng các thay đổi của bạn được yêu cầu. Cách tốt nhất là tạo một [cuộc thảo luận mới](https://github.com/openimsdk/open-im-server/discussions/new/choose) hoặc [Giao tiếp Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A), hoặc nếu bạn tìm thấy một vấn đề, [báo cáo nó ](https://github.com/openimsdk/open-im-server/issues/new/choose) trước. | ||||||
| Trước khi bạn bắt đầu, hãy chắc chắn rằng các thay đổi của bạn được yêu cầu. Cách tốt nhất là tạo một [cuộc thảo luận mới](https://github.com/openimsdk/open-im-server/discussions/new/choose) hoặc [Giao tiếp Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q), hoặc nếu bạn tìm thấy một vấn đề, [báo cáo nó ](https://github.com/openimsdk/open-im-server/issues/new/choose) trước. |  | ||||||
| 
 | 
 | ||||||
| - [Tham khảo API OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | - [Tham khảo API OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/api.md) | ||||||
| - [Nhật ký Bash OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | - [Nhật ký Bash OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/bash-log.md) | ||||||
| @ -155,19 +151,18 @@ Trước khi bạn bắt đầu, hãy chắc chắn rằng các thay đổi củ | |||||||
| - [Quản lý triển khai và giám sát backend](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | - [Quản lý triển khai và giám sát backend](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/prometheus-grafana.md) | ||||||
| - [Hướng dẫn Triển khai cho Nhà Phát triển Mac OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | - [Hướng dẫn Triển khai cho Nhà Phát triển Mac OpenIM](https://github.com/openimsdk/open-im-server/tree/main/docs/contrib/mac-developer-deployment-guide.md) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| ## :busts_in_silhouette: Cộng đồng | ## :busts_in_silhouette: Cộng đồng | ||||||
| 
 | 
 | ||||||
| + 📚 [Cộng đồng OpenIM](https://github.com/OpenIMSDK/community) | - 📚 [Cộng đồng OpenIM](https://github.com/OpenIMSDK/community) | ||||||
| + 💕 [Nhóm Quan tâm OpenIM](https://github.com/Openim-sigs) | - 💕 [Nhóm Quan tâm OpenIM](https://github.com/Openim-sigs) | ||||||
| + 🚀 [Tham gia cộng đồng Slack của chúng tôi](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) | - 🚀 [Tham gia cộng đồng Slack của chúng tôi](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) | ||||||
| + :eyes: [Tham gia nhóm WeChat của chúng tôi (微信群)](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | - :eyes: [Tham gia nhóm WeChat của chúng tôi (微信群)](https://openim-1253691595.cos.ap-nanjing.myqcloud.com/WechatIMG20.jpeg) | ||||||
| 
 | 
 | ||||||
| ## :calendar: Cuộc họp Cộng đồng | ## :calendar: Cuộc họp Cộng đồng | ||||||
| 
 | 
 | ||||||
| Chúng tôi muốn bất kỳ ai cũng có thể tham gia cộng đồng và đóng góp mã nguồn, chúng tôi cung cấp quà tặng và phần thưởng, và chúng tôi chào đón bạn tham gia cùng chúng tôi mỗi tối thứ Năm. | Chúng tôi muốn bất kỳ ai cũng có thể tham gia cộng đồng và đóng góp mã nguồn, chúng tôi cung cấp quà tặng và phần thưởng, và chúng tôi chào đón bạn tham gia cùng chúng tôi mỗi tối thứ Năm. | ||||||
| 
 | 
 | ||||||
| Hội nghị của chúng tôi được tổ chức trên Slack của [OpenIM Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q) 🎯, sau đó bạn có thể tìm kiếm pipeline Open-IM-Server để tham gia | Hội nghị của chúng tôi được tổ chức trên Slack của [OpenIM Slack](https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A) 🎯, sau đó bạn có thể tìm kiếm pipeline Open-IM-Server để tham gia | ||||||
| 
 | 
 | ||||||
| Chúng tôi ghi chú mỗi [cuộc họp hai tuần một lần](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting) trong [các cuộc thảo luận GitHub](https://github.com/openimsdk/open-im-server/discussions/categories/meeting), ghi chú cuộc họp lịch sử của chúng tôi cũng như các bản ghi lại của cuộc họp có sẵn tại [Google Docs :bookmark_tabs:](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing). | Chúng tôi ghi chú mỗi [cuộc họp hai tuần một lần](https://github.com/orgs/OpenIMSDK/discussions/categories/meeting) trong [các cuộc thảo luận GitHub](https://github.com/openimsdk/open-im-server/discussions/categories/meeting), ghi chú cuộc họp lịch sử của chúng tôi cũng như các bản ghi lại của cuộc họp có sẵn tại [Google Docs :bookmark_tabs:](https://docs.google.com/document/d/1nx8MDpuG74NASx081JcCpxPgDITNTpIIos0DS6Vr9GU/edit?usp=sharing). | ||||||
| 
 | 
 | ||||||
|  | |||||||
							
								
								
									
										6
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										6
									
								
								go.mod
									
									
									
									
									
								
							| @ -12,8 +12,8 @@ require ( | |||||||
| 	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.78 | 	github.com/openimsdk/protocol v0.0.73-alpha.12 | ||||||
| 	github.com/openimsdk/tools v0.0.50-alpha.74 | 	github.com/openimsdk/tools v0.0.50-alpha.92 | ||||||
| 	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 | ||||||
| @ -34,7 +34,7 @@ require ( | |||||||
| 	github.com/hashicorp/golang-lru/v2 v2.0.7 | 	github.com/hashicorp/golang-lru/v2 v2.0.7 | ||||||
| 	github.com/kelindar/bitmap v1.5.2 | 	github.com/kelindar/bitmap v1.5.2 | ||||||
| 	github.com/likexian/gokit v0.25.13 | 	github.com/likexian/gokit v0.25.13 | ||||||
| 	github.com/openimsdk/gomake v0.0.15-alpha.2 | 	github.com/openimsdk/gomake v0.0.15-alpha.11 | ||||||
| 	github.com/redis/go-redis/v9 v9.4.0 | 	github.com/redis/go-redis/v9 v9.4.0 | ||||||
| 	github.com/robfig/cron/v3 v3.0.1 | 	github.com/robfig/cron/v3 v3.0.1 | ||||||
| 	github.com/shirou/gopsutil v3.21.11+incompatible | 	github.com/shirou/gopsutil v3.21.11+incompatible | ||||||
|  | |||||||
							
								
								
									
										12
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								go.sum
									
									
									
									
									
								
							| @ -345,12 +345,12 @@ github.com/onsi/ginkgo/v2 v2.19.0 h1:9Cnnf7UHo57Hy3k6/m5k3dRfGTMXGvxhHFvkDTCTpvA | |||||||
| github.com/onsi/ginkgo/v2 v2.19.0/go.mod h1:rlwLi9PilAFJ8jCg9UE1QP6VBpd6/xj3SRC0d6TU0To= | github.com/onsi/ginkgo/v2 v2.19.0/go.mod h1:rlwLi9PilAFJ8jCg9UE1QP6VBpd6/xj3SRC0d6TU0To= | ||||||
| github.com/onsi/gomega v1.25.0 h1:Vw7br2PCDYijJHSfBOWhov+8cAnUf8MfMaIOV323l6Y= | 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.15-alpha.2 h1:5Q8yl8ezy2yx+q8/ucU/t4kJnDfCzNOrkXcDACCqtyM= | github.com/openimsdk/gomake v0.0.15-alpha.11 h1:PQudYDRESYeYlUYrrLLJhYIlUPO5x7FAx+o5El9U/Bw= | ||||||
| github.com/openimsdk/gomake v0.0.15-alpha.2/go.mod h1:PndCozNc2IsQIciyn9mvEblYWZwJmAI+06z94EY+csI= | github.com/openimsdk/gomake v0.0.15-alpha.11/go.mod h1:PndCozNc2IsQIciyn9mvEblYWZwJmAI+06z94EY+csI= | ||||||
| github.com/openimsdk/protocol v0.0.72-alpha.78 h1:n9HVj5olMPiGLF3Z4apPvvYzn2yOpyrsn2/YiAaIsxw= | github.com/openimsdk/protocol v0.0.73-alpha.12 h1:2NYawXeHChYUeSme6QJ9pOLh+Empce2WmwEtbP4JvKk= | ||||||
| github.com/openimsdk/protocol v0.0.72-alpha.78/go.mod h1:WF7EuE55vQvpyUAzDXcqg+B+446xQyEba0X35lTINmw= | github.com/openimsdk/protocol v0.0.73-alpha.12/go.mod h1:WF7EuE55vQvpyUAzDXcqg+B+446xQyEba0X35lTINmw= | ||||||
| github.com/openimsdk/tools v0.0.50-alpha.74 h1:yh10SiMiivMEjicEQg+QAsH4pvaO+4noMPdlw+ew0Kc= | github.com/openimsdk/tools v0.0.50-alpha.92 h1:hWfykMhmi7EQEiwgQccJqbgggIuhun/PrVkBnjmj9Ec= | ||||||
| github.com/openimsdk/tools v0.0.50-alpha.74/go.mod h1:n2poR3asX1e1XZce4O+MOWAp+X02QJRFvhcLCXZdzRo= | github.com/openimsdk/tools v0.0.50-alpha.92/go.mod h1:n2poR3asX1e1XZce4O+MOWAp+X02QJRFvhcLCXZdzRo= | ||||||
| github.com/pelletier/go-toml/v2 v2.2.2 h1:aYUidT7k73Pcl9nb2gScu7NSrKCSHIDE89b3+6Wq+LM= | github.com/pelletier/go-toml/v2 v2.2.2 h1:aYUidT7k73Pcl9nb2gScu7NSrKCSHIDE89b3+6Wq+LM= | ||||||
| github.com/pelletier/go-toml/v2 v2.2.2/go.mod h1:1t835xjRzz80PqgE6HHgN2JOsmgYu/h4qDAS4n929Rs= | github.com/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= | ||||||
|  | |||||||
| @ -531,7 +531,7 @@ O:::::::OOO:::::::O p:::::ppppp:::::::pe::::::::e            n::::n    n::::nII: | |||||||
|     # Set text color to yellow for the Slack link |     # Set text color to yellow for the Slack link | ||||||
|     echo -e "\033[1;33m" |     echo -e "\033[1;33m" | ||||||
| 
 | 
 | ||||||
|     print_with_delay "Join our developer community on Slack: https://join.slack.com/t/openimsdk/shared_invite/zt-22720d66b-o_FvKxMTGXtcnnnHiMqe9Q" 0.01 |     print_with_delay "Join our developer community on Slack: https://join.slack.com/t/openimsdk/shared_invite/zt-2ijy1ys1f-O0aEDCr7ExRZ7mwsHAVg9A" 0.01 | ||||||
| 
 | 
 | ||||||
|     # Reset text color back to normal |     # Reset text color back to normal | ||||||
|     echo -e "\033[0m" |     echo -e "\033[0m" | ||||||
|  | |||||||
| @ -15,6 +15,7 @@ import ( | |||||||
| 	"github.com/openimsdk/tools/apiresp" | 	"github.com/openimsdk/tools/apiresp" | ||||||
| 	"github.com/openimsdk/tools/errs" | 	"github.com/openimsdk/tools/errs" | ||||||
| 	"github.com/openimsdk/tools/log" | 	"github.com/openimsdk/tools/log" | ||||||
|  | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
| 	"github.com/openimsdk/tools/utils/runtimeenv" | 	"github.com/openimsdk/tools/utils/runtimeenv" | ||||||
| 	clientv3 "go.etcd.io/etcd/client/v3" | 	clientv3 "go.etcd.io/etcd/client/v3" | ||||||
| ) | ) | ||||||
| @ -43,7 +44,7 @@ func NewConfigManager(IMAdminUserID []string, cfg *config.AllConfig, client *cli | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (cm *ConfigManager) CheckAdmin(c *gin.Context) { | func (cm *ConfigManager) CheckAdmin(c *gin.Context) { | ||||||
| 	if err := authverify.CheckAdmin(c, cm.imAdminUserID); err != nil { | 	if err := authverify.CheckAdmin(c); err != nil { | ||||||
| 		apiresp.GinError(c, err) | 		apiresp.GinError(c, err) | ||||||
| 		c.Abort() | 		c.Abort() | ||||||
| 	} | 	} | ||||||
| @ -144,6 +145,110 @@ func (cm *ConfigManager) SetConfig(c *gin.Context) { | |||||||
| 	apiresp.GinSuccess(c, nil) | 	apiresp.GinSuccess(c, nil) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func (cm *ConfigManager) SetConfigs(c *gin.Context) { | ||||||
|  | 	if cm.config.Discovery.Enable != config.ETCD { | ||||||
|  | 		apiresp.GinError(c, errs.New("only etcd support set config").Wrap()) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	var req apistruct.SetConfigsReq | ||||||
|  | 	if err := c.BindJSON(&req); err != nil { | ||||||
|  | 		apiresp.GinError(c, errs.ErrArgs.WithDetail(err.Error()).Wrap()) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	var ( | ||||||
|  | 		err error | ||||||
|  | 		ops []*clientv3.Op | ||||||
|  | 	) | ||||||
|  | 
 | ||||||
|  | 	for _, cf := range req.Configs { | ||||||
|  | 		var op *clientv3.Op | ||||||
|  | 		switch cf.ConfigName { | ||||||
|  | 		case cm.config.Discovery.GetConfigFileName(): | ||||||
|  | 			op, err = compareAndOp[config.Discovery](c, cm.config.Name2Config(cf.ConfigName), &cf, cm) | ||||||
|  | 		case cm.config.Kafka.GetConfigFileName(): | ||||||
|  | 			op, err = compareAndOp[config.Kafka](c, cm.config.Name2Config(cf.ConfigName), &cf, cm) | ||||||
|  | 		case cm.config.LocalCache.GetConfigFileName(): | ||||||
|  | 			op, err = compareAndOp[config.LocalCache](c, cm.config.Name2Config(cf.ConfigName), &cf, cm) | ||||||
|  | 		case cm.config.Log.GetConfigFileName(): | ||||||
|  | 			op, err = compareAndOp[config.Log](c, cm.config.Name2Config(cf.ConfigName), &cf, cm) | ||||||
|  | 		case cm.config.Minio.GetConfigFileName(): | ||||||
|  | 			op, err = compareAndOp[config.Minio](c, cm.config.Name2Config(cf.ConfigName), &cf, cm) | ||||||
|  | 		case cm.config.Mongo.GetConfigFileName(): | ||||||
|  | 			op, err = compareAndOp[config.Mongo](c, cm.config.Name2Config(cf.ConfigName), &cf, cm) | ||||||
|  | 		case cm.config.Notification.GetConfigFileName(): | ||||||
|  | 			op, err = compareAndOp[config.Notification](c, cm.config.Name2Config(cf.ConfigName), &cf, cm) | ||||||
|  | 		case cm.config.API.GetConfigFileName(): | ||||||
|  | 			op, err = compareAndOp[config.API](c, cm.config.Name2Config(cf.ConfigName), &cf, cm) | ||||||
|  | 		case cm.config.CronTask.GetConfigFileName(): | ||||||
|  | 			op, err = compareAndOp[config.CronTask](c, cm.config.Name2Config(cf.ConfigName), &cf, cm) | ||||||
|  | 		case cm.config.MsgGateway.GetConfigFileName(): | ||||||
|  | 			op, err = compareAndOp[config.MsgGateway](c, cm.config.Name2Config(cf.ConfigName), &cf, cm) | ||||||
|  | 		case cm.config.MsgTransfer.GetConfigFileName(): | ||||||
|  | 			op, err = compareAndOp[config.MsgTransfer](c, cm.config.Name2Config(cf.ConfigName), &cf, cm) | ||||||
|  | 		case cm.config.Push.GetConfigFileName(): | ||||||
|  | 			op, err = compareAndOp[config.Push](c, cm.config.Name2Config(cf.ConfigName), &cf, cm) | ||||||
|  | 		case cm.config.Auth.GetConfigFileName(): | ||||||
|  | 			op, err = compareAndOp[config.Auth](c, cm.config.Name2Config(cf.ConfigName), &cf, cm) | ||||||
|  | 		case cm.config.Conversation.GetConfigFileName(): | ||||||
|  | 			op, err = compareAndOp[config.Conversation](c, cm.config.Name2Config(cf.ConfigName), &cf, cm) | ||||||
|  | 		case cm.config.Friend.GetConfigFileName(): | ||||||
|  | 			op, err = compareAndOp[config.Friend](c, cm.config.Name2Config(cf.ConfigName), &cf, cm) | ||||||
|  | 		case cm.config.Group.GetConfigFileName(): | ||||||
|  | 			op, err = compareAndOp[config.Group](c, cm.config.Name2Config(cf.ConfigName), &cf, cm) | ||||||
|  | 		case cm.config.Msg.GetConfigFileName(): | ||||||
|  | 			op, err = compareAndOp[config.Msg](c, cm.config.Name2Config(cf.ConfigName), &cf, cm) | ||||||
|  | 		case cm.config.Third.GetConfigFileName(): | ||||||
|  | 			op, err = compareAndOp[config.Third](c, cm.config.Name2Config(cf.ConfigName), &cf, cm) | ||||||
|  | 		case cm.config.User.GetConfigFileName(): | ||||||
|  | 			op, err = compareAndOp[config.User](c, cm.config.Name2Config(cf.ConfigName), &cf, cm) | ||||||
|  | 		case cm.config.Redis.GetConfigFileName(): | ||||||
|  | 			op, err = compareAndOp[config.Redis](c, cm.config.Name2Config(cf.ConfigName), &cf, cm) | ||||||
|  | 		case cm.config.Share.GetConfigFileName(): | ||||||
|  | 			op, err = compareAndOp[config.Share](c, cm.config.Name2Config(cf.ConfigName), &cf, cm) | ||||||
|  | 		case cm.config.Webhooks.GetConfigFileName(): | ||||||
|  | 			op, err = compareAndOp[config.Webhooks](c, cm.config.Name2Config(cf.ConfigName), &cf, cm) | ||||||
|  | 		default: | ||||||
|  | 			apiresp.GinError(c, errs.ErrArgs.Wrap()) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 		if err != nil { | ||||||
|  | 			apiresp.GinError(c, errs.ErrArgs.WithDetail(err.Error()).Wrap()) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 		if op != nil { | ||||||
|  | 			ops = append(ops, op) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	if len(ops) > 0 { | ||||||
|  | 		tx := cm.client.Txn(c) | ||||||
|  | 		if _, err = tx.Then(datautil.Batch(func(op *clientv3.Op) clientv3.Op { return *op }, ops)...).Commit(); err != nil { | ||||||
|  | 			apiresp.GinError(c, errs.WrapMsg(err, "save to etcd failed")) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	apiresp.GinSuccess(c, nil) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func compareAndOp[T any](c *gin.Context, old any, req *apistruct.SetConfigReq, cm *ConfigManager) (*clientv3.Op, error) { | ||||||
|  | 	conf := new(T) | ||||||
|  | 	err := json.Unmarshal([]byte(req.Data), &conf) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, errs.ErrArgs.WithDetail(err.Error()).Wrap() | ||||||
|  | 	} | ||||||
|  | 	eq := reflect.DeepEqual(old, conf) | ||||||
|  | 	if eq { | ||||||
|  | 		return nil, nil | ||||||
|  | 	} | ||||||
|  | 	data, err := json.Marshal(conf) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, errs.ErrArgs.WithDetail(err.Error()).Wrap() | ||||||
|  | 	} | ||||||
|  | 	op := clientv3.OpPut(etcd.BuildKey(req.ConfigName), string(data)) | ||||||
|  | 	return &op, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func compareAndSave[T any](c *gin.Context, old any, req *apistruct.SetConfigReq, cm *ConfigManager) error { | func compareAndSave[T any](c *gin.Context, old any, req *apistruct.SetConfigReq, cm *ConfigManager) error { | ||||||
| 	conf := new(T) | 	conf := new(T) | ||||||
| 	err := json.Unmarshal([]byte(req.Data), &conf) | 	err := json.Unmarshal([]byte(req.Data), &conf) | ||||||
|  | |||||||
| @ -16,6 +16,7 @@ package api | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
|  | 
 | ||||||
| 	"github.com/openimsdk/protocol/conversation" | 	"github.com/openimsdk/protocol/conversation" | ||||||
| 	"github.com/openimsdk/tools/a2r" | 	"github.com/openimsdk/tools/a2r" | ||||||
| ) | ) | ||||||
| @ -48,9 +49,9 @@ func (o *ConversationApi) SetConversations(c *gin.Context) { | |||||||
| 	a2r.Call(c, conversation.ConversationClient.SetConversations, o.Client) | 	a2r.Call(c, conversation.ConversationClient.SetConversations, o.Client) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (o *ConversationApi) GetConversationOfflinePushUserIDs(c *gin.Context) { | //func (o *ConversationApi) GetConversationOfflinePushUserIDs(c *gin.Context) { | ||||||
| 	a2r.Call(c, conversation.ConversationClient.GetConversationOfflinePushUserIDs, o.Client) | //	a2r.Call(c, conversation.ConversationClient.GetConversationOfflinePushUserIDs, o.Client) | ||||||
| } | //} | ||||||
| 
 | 
 | ||||||
| func (o *ConversationApi) GetFullOwnerConversationIDs(c *gin.Context) { | func (o *ConversationApi) GetFullOwnerConversationIDs(c *gin.Context) { | ||||||
| 	a2r.Call(c, conversation.ConversationClient.GetFullOwnerConversationIDs, o.Client) | 	a2r.Call(c, conversation.ConversationClient.GetFullOwnerConversationIDs, o.Client) | ||||||
| @ -71,3 +72,7 @@ func (o *ConversationApi) GetNotNotifyConversationIDs(c *gin.Context) { | |||||||
| func (o *ConversationApi) GetPinnedConversationIDs(c *gin.Context) { | func (o *ConversationApi) GetPinnedConversationIDs(c *gin.Context) { | ||||||
| 	a2r.Call(c, conversation.ConversationClient.GetPinnedConversationIDs, o.Client) | 	a2r.Call(c, conversation.ConversationClient.GetPinnedConversationIDs, o.Client) | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | func (o *ConversationApi) UpdateConversationsByUser(c *gin.Context) { | ||||||
|  | 	a2r.Call(c, conversation.ConversationClient.UpdateConversationsByUser, o.Client) | ||||||
|  | } | ||||||
|  | |||||||
| @ -114,3 +114,7 @@ func (o *FriendApi) GetIncrementalBlacks(c *gin.Context) { | |||||||
| func (o *FriendApi) GetFullFriendUserIDs(c *gin.Context) { | func (o *FriendApi) GetFullFriendUserIDs(c *gin.Context) { | ||||||
| 	a2r.Call(c, relation.FriendClient.GetFullFriendUserIDs, o.Client) | 	a2r.Call(c, relation.FriendClient.GetFullFriendUserIDs, o.Client) | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | func (o *FriendApi) GetSelfUnhandledApplyCount(c *gin.Context) { | ||||||
|  | 	a2r.Call(c, relation.FriendClient.GetSelfUnhandledApplyCount, o.Client) | ||||||
|  | } | ||||||
|  | |||||||
| @ -165,3 +165,7 @@ func (o *GroupApi) GetFullGroupMemberUserIDs(c *gin.Context) { | |||||||
| func (o *GroupApi) GetFullJoinGroupIDs(c *gin.Context) { | func (o *GroupApi) GetFullJoinGroupIDs(c *gin.Context) { | ||||||
| 	a2r.Call(c, group.GroupClient.GetFullJoinGroupIDs, o.Client) | 	a2r.Call(c, group.GroupClient.GetFullJoinGroupIDs, o.Client) | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | func (o *GroupApi) GetGroupApplicationUnhandledCount(c *gin.Context) { | ||||||
|  | 	a2r.Call(c, group.GroupClient.GetGroupApplicationUnhandledCount, o.Client) | ||||||
|  | } | ||||||
|  | |||||||
| @ -39,7 +39,7 @@ type Config struct { | |||||||
| 	Index      conf.Index | 	Index      conf.Index | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func Start(ctx context.Context, config *Config, client discovery.Conn, service grpc.ServiceRegistrar) error { | func Start(ctx context.Context, config *Config, client discovery.SvcDiscoveryRegistry, service grpc.ServiceRegistrar) error { | ||||||
| 	apiPort, err := datautil.GetElemByIndex(config.API.Api.Ports, int(config.Index)) | 	apiPort, err := datautil.GetElemByIndex(config.API.Api.Ports, int(config.Index)) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| @ -90,7 +90,7 @@ func Start(ctx context.Context, config *Config, client discovery.Conn, service g | |||||||
| 	//case <-ctx.Done(): | 	//case <-ctx.Done(): | ||||||
| 	//} | 	//} | ||||||
| 	<-apiCtx.Done() | 	<-apiCtx.Done() | ||||||
| 	exitCause := context.Cause(ctx) | 	exitCause := context.Cause(apiCtx) | ||||||
| 	log.ZWarn(ctx, "api server exit", exitCause) | 	log.ZWarn(ctx, "api server exit", exitCause) | ||||||
| 	timer := time.NewTimer(time.Second * 15) | 	timer := time.NewTimer(time.Second * 15) | ||||||
| 	defer timer.Stop() | 	defer timer.Stop() | ||||||
|  | |||||||
| @ -2,10 +2,14 @@ package jssdk | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" |  | ||||||
| 	"sort" | 	"sort" | ||||||
| 
 | 
 | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | ||||||
|  | 	"github.com/openimsdk/protocol/constant" | ||||||
|  | 	"github.com/openimsdk/tools/log" | ||||||
|  | 
 | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
|  | 
 | ||||||
| 	"github.com/openimsdk/protocol/conversation" | 	"github.com/openimsdk/protocol/conversation" | ||||||
| 	"github.com/openimsdk/protocol/jssdk" | 	"github.com/openimsdk/protocol/jssdk" | ||||||
| 	"github.com/openimsdk/protocol/msg" | 	"github.com/openimsdk/protocol/msg" | ||||||
| @ -109,10 +113,7 @@ func (x *JSSdk) getActiveConversations(ctx context.Context, req *jssdk.GetActive | |||||||
| 	if len(conversationIDs) == 0 { | 	if len(conversationIDs) == 0 { | ||||||
| 		return &jssdk.GetActiveConversationsResp{}, nil | 		return &jssdk.GetActiveConversationsResp{}, nil | ||||||
| 	} | 	} | ||||||
| 	readSeq, err := x.msgClient.GetHasReadSeqs(ctx, conversationIDs, req.OwnerUserID) | 
 | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	activeConversation, err := x.msgClient.GetActiveConversation(ctx, conversationIDs) | 	activeConversation, err := x.msgClient.GetActiveConversation(ctx, conversationIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -120,6 +121,10 @@ func (x *JSSdk) getActiveConversations(ctx context.Context, req *jssdk.GetActive | |||||||
| 	if len(activeConversation) == 0 { | 	if len(activeConversation) == 0 { | ||||||
| 		return &jssdk.GetActiveConversationsResp{}, nil | 		return &jssdk.GetActiveConversationsResp{}, nil | ||||||
| 	} | 	} | ||||||
|  | 	readSeq, err := x.msgClient.GetHasReadSeqs(ctx, conversationIDs, req.OwnerUserID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	sortConversations := sortActiveConversations{ | 	sortConversations := sortActiveConversations{ | ||||||
| 		Conversation: activeConversation, | 		Conversation: activeConversation, | ||||||
| 	} | 	} | ||||||
| @ -147,6 +152,7 @@ func (x *JSSdk) getActiveConversations(ctx context.Context, req *jssdk.GetActive | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 	x.checkMessagesAndGetLastMessage(ctx, req.OwnerUserID, msgs) | ||||||
| 	conversationMap := datautil.SliceToMap(conversations, func(c *conversation.Conversation) string { | 	conversationMap := datautil.SliceToMap(conversations, func(c *conversation.Conversation) string { | ||||||
| 		return c.ConversationID | 		return c.ConversationID | ||||||
| 	}) | 	}) | ||||||
| @ -156,16 +162,15 @@ func (x *JSSdk) getActiveConversations(ctx context.Context, req *jssdk.GetActive | |||||||
| 		if !ok { | 		if !ok { | ||||||
| 			continue | 			continue | ||||||
| 		} | 		} | ||||||
| 		var lastMsg *sdkws.MsgData |  | ||||||
| 		if msgList, ok := msgs[c.ConversationID]; ok && len(msgList.Msgs) > 0 { | 		if msgList, ok := msgs[c.ConversationID]; ok && len(msgList.Msgs) > 0 { | ||||||
| 			lastMsg = msgList.Msgs[0] | 			resp = append(resp, &jssdk.ConversationMsg{ | ||||||
|  | 				Conversation: conv, | ||||||
|  | 				LastMsg:      msgList.Msgs[0], | ||||||
|  | 				MaxSeq:       c.MaxSeq, | ||||||
|  | 				ReadSeq:      readSeq[c.ConversationID], | ||||||
|  | 			}) | ||||||
| 		} | 		} | ||||||
| 		resp = append(resp, &jssdk.ConversationMsg{ | 
 | ||||||
| 			Conversation: conv, |  | ||||||
| 			LastMsg:      lastMsg, |  | ||||||
| 			MaxSeq:       c.MaxSeq, |  | ||||||
| 			ReadSeq:      readSeq[c.ConversationID], |  | ||||||
| 		}) |  | ||||||
| 	} | 	} | ||||||
| 	if err := x.fillConversations(ctx, resp); err != nil { | 	if err := x.fillConversations(ctx, resp); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -219,18 +224,18 @@ func (x *JSSdk) getConversations(ctx context.Context, req *jssdk.GetConversation | |||||||
| 			return nil, err | 			return nil, err | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | 	x.checkMessagesAndGetLastMessage(ctx, req.OwnerUserID, msgs) | ||||||
| 	resp := make([]*jssdk.ConversationMsg, 0, len(conversations)) | 	resp := make([]*jssdk.ConversationMsg, 0, len(conversations)) | ||||||
| 	for _, c := range conversations { | 	for _, c := range conversations { | ||||||
| 		var lastMsg *sdkws.MsgData |  | ||||||
| 		if msgList, ok := msgs[c.ConversationID]; ok && len(msgList.Msgs) > 0 { | 		if msgList, ok := msgs[c.ConversationID]; ok && len(msgList.Msgs) > 0 { | ||||||
| 			lastMsg = msgList.Msgs[0] | 			resp = append(resp, &jssdk.ConversationMsg{ | ||||||
|  | 				Conversation: c, | ||||||
|  | 				LastMsg:      msgList.Msgs[0], | ||||||
|  | 				MaxSeq:       maxSeqs[c.ConversationID], | ||||||
|  | 				ReadSeq:      readSeqs[c.ConversationID], | ||||||
|  | 			}) | ||||||
| 		} | 		} | ||||||
| 		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 { | 	if err := x.fillConversations(ctx, resp); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -247,3 +252,36 @@ func (x *JSSdk) getConversations(ctx context.Context, req *jssdk.GetConversation | |||||||
| 		UnreadCount:   unreadCount, | 		UnreadCount:   unreadCount, | ||||||
| 	}, nil | 	}, nil | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | // This function checks whether the latest MaxSeq message is valid. | ||||||
|  | // If not, it needs to fetch a valid message again. | ||||||
|  | func (x *JSSdk) checkMessagesAndGetLastMessage(ctx context.Context, userID string, messages map[string]*sdkws.PullMsgs) { | ||||||
|  | 	var conversationIDs []string | ||||||
|  | 
 | ||||||
|  | 	for conversationID, message := range messages { | ||||||
|  | 		allInValid := true | ||||||
|  | 		for _, data := range message.Msgs { | ||||||
|  | 			if data.Status < constant.MsgStatusHasDeleted { | ||||||
|  | 				allInValid = false | ||||||
|  | 				break | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		if allInValid { | ||||||
|  | 			conversationIDs = append(conversationIDs, conversationID) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	if len(conversationIDs) > 0 { | ||||||
|  | 		resp, err := x.msgClient.GetLastMessage(ctx, &msg.GetLastMessageReq{ | ||||||
|  | 			UserID:          userID, | ||||||
|  | 			ConversationIDs: conversationIDs, | ||||||
|  | 		}) | ||||||
|  | 		if err != nil { | ||||||
|  | 			log.ZError(ctx, "fetchLatestValidMessages", err, "conversationIDs", conversationIDs) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 		for conversationID, message := range resp.Msgs { | ||||||
|  | 			messages[conversationID] = &sdkws.PullMsgs{Msgs: []*sdkws.MsgData{message}} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | |||||||
| @ -17,10 +17,12 @@ package api | |||||||
| import ( | import ( | ||||||
| 	"encoding/base64" | 	"encoding/base64" | ||||||
| 	"encoding/json" | 	"encoding/json" | ||||||
|  | 	"sync" | ||||||
| 
 | 
 | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
| 	"github.com/go-playground/validator/v10" | 	"github.com/go-playground/validator/v10" | ||||||
| 	"github.com/mitchellh/mapstructure" | 	"github.com/mitchellh/mapstructure" | ||||||
|  | 	"google.golang.org/protobuf/reflect/protoreflect" | ||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/apistruct" | 	"github.com/openimsdk/open-im-server/v3/pkg/apistruct" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||||
| @ -41,6 +43,39 @@ import ( | |||||||
| 	"github.com/openimsdk/tools/utils/timeutil" | 	"github.com/openimsdk/tools/utils/timeutil" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | var ( | ||||||
|  | 	msgDataDescriptor     []protoreflect.FieldDescriptor | ||||||
|  | 	msgDataDescriptorOnce sync.Once | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func getMsgDataDescriptor() []protoreflect.FieldDescriptor { | ||||||
|  | 	msgDataDescriptorOnce.Do(func() { | ||||||
|  | 		skip := make(map[string]struct{}) | ||||||
|  | 		respFields := new(msg.SendMsgResp).ProtoReflect().Descriptor().Fields() | ||||||
|  | 		for i := 0; i < respFields.Len(); i++ { | ||||||
|  | 			field := respFields.Get(i) | ||||||
|  | 			if !field.HasJSONName() { | ||||||
|  | 				continue | ||||||
|  | 			} | ||||||
|  | 			skip[field.JSONName()] = struct{}{} | ||||||
|  | 		} | ||||||
|  | 		fields := new(sdkws.MsgData).ProtoReflect().Descriptor().Fields() | ||||||
|  | 		num := fields.Len() | ||||||
|  | 		msgDataDescriptor = make([]protoreflect.FieldDescriptor, 0, num) | ||||||
|  | 		for i := 0; i < num; i++ { | ||||||
|  | 			field := fields.Get(i) | ||||||
|  | 			if !field.HasJSONName() { | ||||||
|  | 				continue | ||||||
|  | 			} | ||||||
|  | 			if _, ok := skip[field.JSONName()]; ok { | ||||||
|  | 				continue | ||||||
|  | 			} | ||||||
|  | 			msgDataDescriptor = append(msgDataDescriptor, fields.Get(i)) | ||||||
|  | 		} | ||||||
|  | 	}) | ||||||
|  | 	return msgDataDescriptor | ||||||
|  | } | ||||||
|  | 
 | ||||||
| type MessageApi struct { | type MessageApi struct { | ||||||
| 	Client        msg.MsgClient | 	Client        msg.MsgClient | ||||||
| 	userClient    *rpcli.UserClient | 	userClient    *rpcli.UserClient | ||||||
| @ -197,6 +232,42 @@ func (m *MessageApi) getSendMsgReq(c *gin.Context, req apistruct.SendMsg) (sendM | |||||||
| 	return m.newUserSendMsgReq(c, &req), nil | 	return m.newUserSendMsgReq(c, &req), nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func (m *MessageApi) getModifyFields(req, respModify *sdkws.MsgData) map[string]any { | ||||||
|  | 	if req == nil || respModify == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	fields := make(map[string]any) | ||||||
|  | 	reqProtoReflect := req.ProtoReflect() | ||||||
|  | 	respProtoReflect := respModify.ProtoReflect() | ||||||
|  | 	for _, descriptor := range getMsgDataDescriptor() { | ||||||
|  | 		reqValue := reqProtoReflect.Get(descriptor) | ||||||
|  | 		respValue := respProtoReflect.Get(descriptor) | ||||||
|  | 		if !reqValue.Equal(respValue) { | ||||||
|  | 			val := respValue.Interface() | ||||||
|  | 			name := descriptor.JSONName() | ||||||
|  | 			if name == "content" { | ||||||
|  | 				if bs, ok := val.([]byte); ok { | ||||||
|  | 					val = string(bs) | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			fields[name] = val | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	if len(fields) == 0 { | ||||||
|  | 		fields = nil | ||||||
|  | 	} | ||||||
|  | 	return fields | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (m *MessageApi) ginRespSendMsg(c *gin.Context, req *msg.SendMsgReq, resp *msg.SendMsgResp) { | ||||||
|  | 	res := m.getModifyFields(req.MsgData, resp.Modify) | ||||||
|  | 	resp.Modify = nil | ||||||
|  | 	apiresp.GinSuccess(c, &apistruct.SendMsgResp{ | ||||||
|  | 		SendMsgResp: resp, | ||||||
|  | 		Modify:      res, | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // SendMessage handles the sending of a message. It's an HTTP handler function to be used with Gin framework. | // SendMessage handles the sending of a message. It's an HTTP handler function to be used with Gin framework. | ||||||
| func (m *MessageApi) SendMessage(c *gin.Context) { | func (m *MessageApi) SendMessage(c *gin.Context) { | ||||||
| 	// Initialize a request struct for sending a message. | 	// Initialize a request struct for sending a message. | ||||||
| @ -210,7 +281,7 @@ func (m *MessageApi) SendMessage(c *gin.Context) { | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	// Check if the user has the app manager role. | 	// Check if the user has the app manager role. | ||||||
| 	if !authverify.IsAppManagerUid(c, m.imAdminUserID) { | 	if !authverify.IsAdmin(c) { | ||||||
| 		// Respond with a permission error if the user is not an app manager. | 		// Respond with a permission error if the user is not an app manager. | ||||||
| 		apiresp.GinError(c, errs.ErrNoPermission.WrapMsg("only app manager can send message")) | 		apiresp.GinError(c, errs.ErrNoPermission.WrapMsg("only app manager can send message")) | ||||||
| 		return | 		return | ||||||
| @ -250,7 +321,7 @@ func (m *MessageApi) SendMessage(c *gin.Context) { | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	// Respond with a success message and the response payload. | 	// Respond with a success message and the response payload. | ||||||
| 	apiresp.GinSuccess(c, respPb) | 	m.ginRespSendMsg(c, sendMsgReq, respPb) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (m *MessageApi) SendBusinessNotification(c *gin.Context) { | func (m *MessageApi) SendBusinessNotification(c *gin.Context) { | ||||||
| @ -284,7 +355,7 @@ func (m *MessageApi) SendBusinessNotification(c *gin.Context) { | |||||||
| 	if req.ReliabilityLevel == nil { | 	if req.ReliabilityLevel == nil { | ||||||
| 		req.ReliabilityLevel = datautil.ToPtr(1) | 		req.ReliabilityLevel = datautil.ToPtr(1) | ||||||
| 	} | 	} | ||||||
| 	if !authverify.IsAppManagerUid(c, m.imAdminUserID) { | 	if !authverify.IsAdmin(c) { | ||||||
| 		apiresp.GinError(c, errs.ErrNoPermission.WrapMsg("only app manager can send message")) | 		apiresp.GinError(c, errs.ErrNoPermission.WrapMsg("only app manager can send message")) | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| @ -316,7 +387,7 @@ func (m *MessageApi) SendBusinessNotification(c *gin.Context) { | |||||||
| 		apiresp.GinError(c, err) | 		apiresp.GinError(c, err) | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	apiresp.GinSuccess(c, respPb) | 	m.ginRespSendMsg(c, &sendMsgReq, respPb) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (m *MessageApi) BatchSendMsg(c *gin.Context) { | func (m *MessageApi) BatchSendMsg(c *gin.Context) { | ||||||
| @ -328,7 +399,7 @@ func (m *MessageApi) BatchSendMsg(c *gin.Context) { | |||||||
| 		apiresp.GinError(c, errs.ErrArgs.WithDetail(err.Error()).Wrap()) | 		apiresp.GinError(c, errs.ErrArgs.WithDetail(err.Error()).Wrap()) | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	if err := authverify.CheckAdmin(c, m.imAdminUserID); err != nil { | 	if err := authverify.CheckAdmin(c); err != nil { | ||||||
| 		apiresp.GinError(c, errs.ErrNoPermission.WrapMsg("only app manager can send message")) | 		apiresp.GinError(c, errs.ErrNoPermission.WrapMsg("only app manager can send message")) | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| @ -370,6 +441,7 @@ func (m *MessageApi) BatchSendMsg(c *gin.Context) { | |||||||
| 			ClientMsgID: rpcResp.ClientMsgID, | 			ClientMsgID: rpcResp.ClientMsgID, | ||||||
| 			SendTime:    rpcResp.SendTime, | 			SendTime:    rpcResp.SendTime, | ||||||
| 			RecvID:      recvID, | 			RecvID:      recvID, | ||||||
|  | 			Modify:      m.getModifyFields(sendMsgReq.MsgData, rpcResp.Modify), | ||||||
| 		}) | 		}) | ||||||
| 	} | 	} | ||||||
| 	apiresp.GinSuccess(c, resp) | 	apiresp.GinSuccess(c, resp) | ||||||
| @ -395,6 +467,10 @@ func (m *MessageApi) SendSimpleMessage(c *gin.Context) { | |||||||
| 		sessionType int32 | 		sessionType int32 | ||||||
| 		recvID      string | 		recvID      string | ||||||
| 	) | 	) | ||||||
|  | 	if err = c.BindJSON(&req); err != nil { | ||||||
|  | 		apiresp.GinError(c, errs.ErrArgs.WithDetail(err.Error()).Wrap()) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
| 	err = json.Unmarshal(decodedData, &keyMsgData) | 	err = json.Unmarshal(decodedData, &keyMsgData) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		apiresp.GinError(c, errs.ErrArgs.WithDetail(err.Error()).Wrap()) | 		apiresp.GinError(c, errs.ErrArgs.WithDetail(err.Error()).Wrap()) | ||||||
| @ -418,6 +494,11 @@ func (m *MessageApi) SendSimpleMessage(c *gin.Context) { | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	content, err := jsonutil.JsonMarshal(apistruct.MarkdownTextElem{Content: req.Content}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		apiresp.GinError(c, errs.Wrap(err)) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
| 	msgData := &sdkws.MsgData{ | 	msgData := &sdkws.MsgData{ | ||||||
| 		SendID:           sendID, | 		SendID:           sendID, | ||||||
| 		RecvID:           recvID, | 		RecvID:           recvID, | ||||||
| @ -426,13 +507,17 @@ func (m *MessageApi) SendSimpleMessage(c *gin.Context) { | |||||||
| 		SenderPlatformID: constant.AdminPlatformID, | 		SenderPlatformID: constant.AdminPlatformID, | ||||||
| 		SessionType:      sessionType, | 		SessionType:      sessionType, | ||||||
| 		MsgFrom:          constant.UserMsgType, | 		MsgFrom:          constant.UserMsgType, | ||||||
| 		ContentType:      constant.Text, | 		ContentType:      constant.MarkdownText, | ||||||
| 		Content:          []byte(req.Content), | 		Content:          content, | ||||||
| 		OfflinePushInfo:  req.OfflinePushInfo, | 		OfflinePushInfo:  req.OfflinePushInfo, | ||||||
| 		Ex:               req.Ex, | 		Ex:               req.Ex, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	respPb, err := m.Client.SendMsg(c, &msg.SendMsgReq{MsgData: msgData}) | 	sendReq := &msg.SendSimpleMsgReq{ | ||||||
|  | 		MsgData: msgData, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	respPb, err := m.Client.SendSimpleMsg(c, sendReq) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		apiresp.GinError(c, err) | 		apiresp.GinError(c, err) | ||||||
| 		return | 		return | ||||||
| @ -449,7 +534,12 @@ func (m *MessageApi) SendSimpleMessage(c *gin.Context) { | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	apiresp.GinSuccess(c, respPb) | 	m.ginRespSendMsg(c, &msg.SendMsgReq{MsgData: sendReq.MsgData}, &msg.SendMsgResp{ | ||||||
|  | 		ServerMsgID: respPb.ServerMsgID, | ||||||
|  | 		ClientMsgID: respPb.ClientMsgID, | ||||||
|  | 		SendTime:    respPb.SendTime, | ||||||
|  | 		Modify:      respPb.Modify, | ||||||
|  | 	}) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (m *MessageApi) CheckMsgIsSendSuccess(c *gin.Context) { | func (m *MessageApi) CheckMsgIsSendSuccess(c *gin.Context) { | ||||||
| @ -475,11 +565,3 @@ func (m *MessageApi) SearchMsg(c *gin.Context) { | |||||||
| func (m *MessageApi) GetServerTime(c *gin.Context) { | func (m *MessageApi) GetServerTime(c *gin.Context) { | ||||||
| 	a2r.Call(c, msg.MsgClient.GetServerTime, m.Client) | 	a2r.Call(c, msg.MsgClient.GetServerTime, m.Client) | ||||||
| } | } | ||||||
| 
 |  | ||||||
| func (m *MessageApi) GetStreamMsg(c *gin.Context) { |  | ||||||
| 	a2r.Call(c, msg.MsgClient.GetServerTime, m.Client) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (m *MessageApi) AppendStreamMsg(c *gin.Context) { |  | ||||||
| 	a2r.Call(c, msg.MsgClient.GetServerTime, m.Client) |  | ||||||
| } |  | ||||||
|  | |||||||
| @ -6,35 +6,29 @@ import ( | |||||||
| 	"net/http" | 	"net/http" | ||||||
| 
 | 
 | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
| 	conf "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" | ||||||
| 	"github.com/openimsdk/tools/apiresp" | 	"github.com/openimsdk/tools/apiresp" | ||||||
| 	"github.com/openimsdk/tools/discovery" | 	"github.com/openimsdk/tools/discovery" | ||||||
| 	"github.com/openimsdk/tools/discovery/etcd" |  | ||||||
| 	"github.com/openimsdk/tools/errs" | 	"github.com/openimsdk/tools/errs" | ||||||
| 	clientv3 "go.etcd.io/etcd/client/v3" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| type PrometheusDiscoveryApi struct { | type PrometheusDiscoveryApi struct { | ||||||
| 	config *Config | 	config *Config | ||||||
| 	client *clientv3.Client |  | ||||||
| 	kv     discovery.KeyValue | 	kv     discovery.KeyValue | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func NewPrometheusDiscoveryApi(config *Config, client discovery.Conn) *PrometheusDiscoveryApi { | func NewPrometheusDiscoveryApi(config *Config, client discovery.SvcDiscoveryRegistry) *PrometheusDiscoveryApi { | ||||||
| 	api := &PrometheusDiscoveryApi{ | 	api := &PrometheusDiscoveryApi{ | ||||||
| 		config: config, | 		config: config, | ||||||
| 	} | 		kv:     client, | ||||||
| 	if config.Discovery.Enable == conf.ETCD { |  | ||||||
| 		api.client = client.(*etcd.SvcDiscoveryRegistryImpl).GetClient() |  | ||||||
| 	} | 	} | ||||||
| 	return api | 	return api | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (p *PrometheusDiscoveryApi) discovery(c *gin.Context, key string) { | func (p *PrometheusDiscoveryApi) discovery(c *gin.Context, key string) { | ||||||
| 	value, err := p.kv.GetKey(c, prommetrics.BuildDiscoveryKey(key)) | 	value, err := p.kv.GetKeyWithPrefix(c, prommetrics.BuildDiscoveryKeyPrefix(key)) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		if errors.Is(err, discovery.ErrNotSupportedKeyValue) { | 		if errors.Is(err, discovery.ErrNotSupported) { | ||||||
| 			c.JSON(http.StatusOK, []struct{}{}) | 			c.JSON(http.StatusOK, []struct{}{}) | ||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
| @ -46,10 +40,17 @@ func (p *PrometheusDiscoveryApi) discovery(c *gin.Context, key string) { | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	var resp prommetrics.RespTarget | 	var resp prommetrics.RespTarget | ||||||
| 	if err := json.Unmarshal(value, &resp); err != nil { | 	for i := range value { | ||||||
| 		apiresp.GinError(c, errs.WrapMsg(err, "json unmarshal err")) | 		var tmp prommetrics.Target | ||||||
| 		return | 		if err = json.Unmarshal(value[i], &tmp); err != nil { | ||||||
|  | 			apiresp.GinError(c, errs.WrapMsg(err, "json unmarshal err")) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		resp.Targets = append(resp.Targets, tmp.Target) | ||||||
|  | 		resp.Labels = tmp.Labels // default label is fixed. See prommetrics.BuildDefaultTarget | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	c.JSON(http.StatusOK, []*prommetrics.RespTarget{&resp}) | 	c.JSON(http.StatusOK, []*prommetrics.RespTarget{&resp}) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -10,6 +10,7 @@ import ( | |||||||
| 	"github.com/gin-gonic/gin/binding" | 	"github.com/gin-gonic/gin/binding" | ||||||
| 	"github.com/go-playground/validator/v10" | 	"github.com/go-playground/validator/v10" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/internal/api/jssdk" | 	"github.com/openimsdk/open-im-server/v3/internal/api/jssdk" | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||||
| 	"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" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/servererrs" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/servererrs" | ||||||
| @ -27,6 +28,7 @@ import ( | |||||||
| 	"github.com/openimsdk/tools/discovery/etcd" | 	"github.com/openimsdk/tools/discovery/etcd" | ||||||
| 	"github.com/openimsdk/tools/log" | 	"github.com/openimsdk/tools/log" | ||||||
| 	"github.com/openimsdk/tools/mw" | 	"github.com/openimsdk/tools/mw" | ||||||
|  | 	"github.com/openimsdk/tools/mw/api" | ||||||
| 	clientv3 "go.etcd.io/etcd/client/v3" | 	clientv3 "go.etcd.io/etcd/client/v3" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| @ -52,7 +54,7 @@ func prommetricsGin() gin.HandlerFunc { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func newGinRouter(ctx context.Context, client discovery.Conn, cfg *Config) (*gin.Engine, error) { | func newGinRouter(ctx context.Context, client discovery.SvcDiscoveryRegistry, cfg *Config) (*gin.Engine, error) { | ||||||
| 	authConn, err := client.GetConn(ctx, cfg.Discovery.RpcService.Auth) | 	authConn, err := client.GetConn(ctx, cfg.Discovery.RpcService.Auth) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -95,8 +97,8 @@ func newGinRouter(ctx context.Context, client discovery.Conn, cfg *Config) (*gin | |||||||
| 	case BestSpeed: | 	case BestSpeed: | ||||||
| 		r.Use(gzip.Gzip(gzip.BestSpeed)) | 		r.Use(gzip.Gzip(gzip.BestSpeed)) | ||||||
| 	} | 	} | ||||||
| 	r.Use(prommetricsGin(), gin.RecoveryWithWriter(gin.DefaultErrorWriter, mw.GinPanicErr), mw.CorsHandler(), | 	r.Use(api.GinLogger(), prommetricsGin(), gin.RecoveryWithWriter(gin.DefaultErrorWriter, mw.GinPanicErr), mw.CorsHandler(), | ||||||
| 		mw.GinParseOperationID(), GinParseToken(rpcli.NewAuthClient(authConn))) | 		mw.GinParseOperationID(), GinParseToken(rpcli.NewAuthClient(authConn)), setGinIsAdmin(cfg.Share.IMAdminUser.UserIDs)) | ||||||
| 
 | 
 | ||||||
| 	u := NewUserApi(user.NewUserClient(userConn), client, cfg.Discovery.RpcService) | 	u := NewUserApi(user.NewUserClient(userConn), client, cfg.Discovery.RpcService) | ||||||
| 	{ | 	{ | ||||||
| @ -124,6 +126,11 @@ func newGinRouter(ctx context.Context, client discovery.Conn, cfg *Config) (*gin | |||||||
| 		userRouterGroup.POST("/add_notification_account", u.AddNotificationAccount) | 		userRouterGroup.POST("/add_notification_account", u.AddNotificationAccount) | ||||||
| 		userRouterGroup.POST("/update_notification_account", u.UpdateNotificationAccountInfo) | 		userRouterGroup.POST("/update_notification_account", u.UpdateNotificationAccountInfo) | ||||||
| 		userRouterGroup.POST("/search_notification_account", u.SearchNotificationAccount) | 		userRouterGroup.POST("/search_notification_account", u.SearchNotificationAccount) | ||||||
|  | 
 | ||||||
|  | 		userRouterGroup.POST("/get_user_client_config", u.GetUserClientConfig) | ||||||
|  | 		userRouterGroup.POST("/set_user_client_config", u.SetUserClientConfig) | ||||||
|  | 		userRouterGroup.POST("/del_user_client_config", u.DelUserClientConfig) | ||||||
|  | 		userRouterGroup.POST("/page_user_client_config", u.PageUserClientConfig) | ||||||
| 	} | 	} | ||||||
| 	// friend routing group | 	// friend routing group | ||||||
| 	{ | 	{ | ||||||
| @ -150,6 +157,7 @@ func newGinRouter(ctx context.Context, client discovery.Conn, cfg *Config) (*gin | |||||||
| 		friendRouterGroup.POST("/update_friends", f.UpdateFriends) | 		friendRouterGroup.POST("/update_friends", f.UpdateFriends) | ||||||
| 		friendRouterGroup.POST("/get_incremental_friends", f.GetIncrementalFriends) | 		friendRouterGroup.POST("/get_incremental_friends", f.GetIncrementalFriends) | ||||||
| 		friendRouterGroup.POST("/get_full_friend_user_ids", f.GetFullFriendUserIDs) | 		friendRouterGroup.POST("/get_full_friend_user_ids", f.GetFullFriendUserIDs) | ||||||
|  | 		friendRouterGroup.POST("/get_self_unhandled_apply_count", f.GetSelfUnhandledApplyCount) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	g := NewGroupApi(group.NewGroupClient(groupConn)) | 	g := NewGroupApi(group.NewGroupClient(groupConn)) | ||||||
| @ -186,6 +194,7 @@ func newGinRouter(ctx context.Context, client discovery.Conn, cfg *Config) (*gin | |||||||
| 		groupRouterGroup.POST("/get_incremental_group_members_batch", g.GetIncrementalGroupMemberBatch) | 		groupRouterGroup.POST("/get_incremental_group_members_batch", g.GetIncrementalGroupMemberBatch) | ||||||
| 		groupRouterGroup.POST("/get_full_group_member_user_ids", g.GetFullGroupMemberUserIDs) | 		groupRouterGroup.POST("/get_full_group_member_user_ids", g.GetFullGroupMemberUserIDs) | ||||||
| 		groupRouterGroup.POST("/get_full_join_group_ids", g.GetFullJoinGroupIDs) | 		groupRouterGroup.POST("/get_full_join_group_ids", g.GetFullJoinGroupIDs) | ||||||
|  | 		groupRouterGroup.POST("/get_group_application_unhandled_count", g.GetGroupApplicationUnhandledCount) | ||||||
| 	} | 	} | ||||||
| 	// certificate | 	// certificate | ||||||
| 	{ | 	{ | ||||||
| @ -223,7 +232,7 @@ func newGinRouter(ctx context.Context, client discovery.Conn, cfg *Config) (*gin | |||||||
| 		objectGroup.GET("/*name", t.ObjectRedirect) | 		objectGroup.GET("/*name", t.ObjectRedirect) | ||||||
| 	} | 	} | ||||||
| 	// Message | 	// Message | ||||||
| 	m := NewMessageApi(msg.NewMsgClient(msgConn), rpcli.NewUserClient(userConn), cfg.Share.IMAdminUserID) | 	m := NewMessageApi(msg.NewMsgClient(msgConn), rpcli.NewUserClient(userConn), cfg.Share.IMAdminUser.UserIDs) | ||||||
| 	{ | 	{ | ||||||
| 		msgGroup := r.Group("/msg") | 		msgGroup := r.Group("/msg") | ||||||
| 		msgGroup.POST("/newest_seq", m.GetSeq) | 		msgGroup.POST("/newest_seq", m.GetSeq) | ||||||
| @ -244,10 +253,9 @@ func newGinRouter(ctx context.Context, client discovery.Conn, cfg *Config) (*gin | |||||||
| 		msgGroup.POST("/delete_msg_physical", m.DeleteMsgPhysical) | 		msgGroup.POST("/delete_msg_physical", m.DeleteMsgPhysical) | ||||||
| 
 | 
 | ||||||
| 		msgGroup.POST("/batch_send_msg", m.BatchSendMsg) | 		msgGroup.POST("/batch_send_msg", m.BatchSendMsg) | ||||||
|  | 		msgGroup.POST("/send_simple_msg", m.SendSimpleMessage) | ||||||
| 		msgGroup.POST("/check_msg_is_send_success", m.CheckMsgIsSendSuccess) | 		msgGroup.POST("/check_msg_is_send_success", m.CheckMsgIsSendSuccess) | ||||||
| 		msgGroup.POST("/get_server_time", m.GetServerTime) | 		msgGroup.POST("/get_server_time", m.GetServerTime) | ||||||
| 		msgGroup.POST("/get_stream_msg", m.GetStreamMsg) |  | ||||||
| 		msgGroup.POST("/append_stream_msg", m.AppendStreamMsg) |  | ||||||
| 	} | 	} | ||||||
| 	// Conversation | 	// Conversation | ||||||
| 	{ | 	{ | ||||||
| @ -258,7 +266,7 @@ func newGinRouter(ctx context.Context, client discovery.Conn, cfg *Config) (*gin | |||||||
| 		conversationGroup.POST("/get_conversation", c.GetConversation) | 		conversationGroup.POST("/get_conversation", c.GetConversation) | ||||||
| 		conversationGroup.POST("/get_conversations", c.GetConversations) | 		conversationGroup.POST("/get_conversations", c.GetConversations) | ||||||
| 		conversationGroup.POST("/set_conversations", c.SetConversations) | 		conversationGroup.POST("/set_conversations", c.SetConversations) | ||||||
| 		conversationGroup.POST("/get_conversation_offline_push_user_ids", c.GetConversationOfflinePushUserIDs) | 		//conversationGroup.POST("/get_conversation_offline_push_user_ids", c.GetConversationOfflinePushUserIDs) | ||||||
| 		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) | ||||||
| @ -301,7 +309,7 @@ func newGinRouter(ctx context.Context, client discovery.Conn, cfg *Config) (*gin | |||||||
| 	if cfg.Discovery.Enable == config.ETCD { | 	if cfg.Discovery.Enable == config.ETCD { | ||||||
| 		etcdClient = client.(*etcd.SvcDiscoveryRegistryImpl).GetClient() | 		etcdClient = client.(*etcd.SvcDiscoveryRegistryImpl).GetClient() | ||||||
| 	} | 	} | ||||||
| 	cm := NewConfigManager(cfg.Share.IMAdminUserID, &cfg.AllConfig, etcdClient, string(cfg.ConfigPath)) | 	cm := NewConfigManager(cfg.Share.IMAdminUser.UserIDs, &cfg.AllConfig, etcdClient, string(cfg.ConfigPath)) | ||||||
| 	{ | 	{ | ||||||
| 		configGroup := r.Group("/config", cm.CheckAdmin) | 		configGroup := r.Group("/config", cm.CheckAdmin) | ||||||
| 		configGroup.POST("/get_config_list", cm.GetConfigList) | 		configGroup.POST("/get_config_list", cm.GetConfigList) | ||||||
| @ -348,6 +356,12 @@ func GinParseToken(authClient *rpcli.AuthClient) gin.HandlerFunc { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func setGinIsAdmin(imAdminUserID []string) gin.HandlerFunc { | ||||||
|  | 	return func(c *gin.Context) { | ||||||
|  | 		c.Set(authverify.CtxAdminUserIDsKey, imAdminUserID) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // Whitelist api not parse token | // Whitelist api not parse token | ||||||
| var Whitelist = []string{ | var Whitelist = []string{ | ||||||
| 	"/auth/get_admin_token", | 	"/auth/get_admin_token", | ||||||
|  | |||||||
| @ -242,3 +242,19 @@ func (u *UserApi) UpdateNotificationAccountInfo(c *gin.Context) { | |||||||
| func (u *UserApi) SearchNotificationAccount(c *gin.Context) { | func (u *UserApi) SearchNotificationAccount(c *gin.Context) { | ||||||
| 	a2r.Call(c, user.UserClient.SearchNotificationAccount, u.Client) | 	a2r.Call(c, user.UserClient.SearchNotificationAccount, u.Client) | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | func (u *UserApi) GetUserClientConfig(c *gin.Context) { | ||||||
|  | 	a2r.Call(c, user.UserClient.GetUserClientConfig, u.Client) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (u *UserApi) SetUserClientConfig(c *gin.Context) { | ||||||
|  | 	a2r.Call(c, user.UserClient.SetUserClientConfig, u.Client) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (u *UserApi) DelUserClientConfig(c *gin.Context) { | ||||||
|  | 	a2r.Call(c, user.UserClient.DelUserClientConfig, u.Client) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (u *UserApi) PageUserClientConfig(c *gin.Context) { | ||||||
|  | 	a2r.Call(c, user.UserClient.PageUserClientConfig, u.Client) | ||||||
|  | } | ||||||
|  | |||||||
| @ -100,7 +100,7 @@ func NewServer(longConnServer LongConnServer, conf *Config, ready func(srv *Serv | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *Server) GetUsersOnlineStatus(ctx context.Context, req *msggateway.GetUsersOnlineStatusReq) (*msggateway.GetUsersOnlineStatusResp, error) { | func (s *Server) GetUsersOnlineStatus(ctx context.Context, req *msggateway.GetUsersOnlineStatusReq) (*msggateway.GetUsersOnlineStatusResp, error) { | ||||||
| 	if !authverify.IsAppManagerUid(ctx, s.config.Share.IMAdminUserID) { | 	if !authverify.IsAdmin(ctx) { | ||||||
| 		return nil, errs.ErrNoPermission.WrapMsg("only app manager") | 		return nil, errs.ErrNoPermission.WrapMsg("only app manager") | ||||||
| 	} | 	} | ||||||
| 	var resp msggateway.GetUsersOnlineStatusResp | 	var resp msggateway.GetUsersOnlineStatusResp | ||||||
| @ -249,6 +249,7 @@ func (s *Server) MultiTerminalLoginCheck(ctx context.Context, req *msggateway.Mu | |||||||
| 		tempUserCtx.SetOperationID(mcontext.GetOperationID(ctx)) | 		tempUserCtx.SetOperationID(mcontext.GetOperationID(ctx)) | ||||||
| 		client := &Client{} | 		client := &Client{} | ||||||
| 		client.ctx = tempUserCtx | 		client.ctx = tempUserCtx | ||||||
|  | 		client.token = req.Token | ||||||
| 		client.UserID = req.UserID | 		client.UserID = req.UserID | ||||||
| 		client.PlatformID = int(req.PlatformID) | 		client.PlatformID = int(req.PlatformID) | ||||||
| 		i := &kickHandler{ | 		i := &kickHandler{ | ||||||
|  | |||||||
| @ -39,7 +39,7 @@ type Config struct { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Start run ws server. | // Start run ws server. | ||||||
| func Start(ctx context.Context, conf *Config, client discovery.Conn, server grpc.ServiceRegistrar) error { | func Start(ctx context.Context, conf *Config, client discovery.SvcDiscoveryRegistry, server grpc.ServiceRegistrar) error { | ||||||
| 	log.CInfo(ctx, "MSG-GATEWAY server is initializing", "runtimeEnv", runtimeenv.RuntimeEnvironment(), | 	log.CInfo(ctx, "MSG-GATEWAY server is initializing", "runtimeEnv", runtimeenv.RuntimeEnvironment(), | ||||||
| 		"rpcPorts", conf.MsgGateway.RPC.Ports, | 		"rpcPorts", conf.MsgGateway.RPC.Ports, | ||||||
| 		"wsPort", conf.MsgGateway.LongConnSvr.Ports, "prometheusPorts", conf.MsgGateway.Prometheus.Ports) | 		"wsPort", conf.MsgGateway.LongConnSvr.Ports, "prometheusPorts", conf.MsgGateway.Prometheus.Ports) | ||||||
|  | |||||||
| @ -49,7 +49,7 @@ type WsServer struct { | |||||||
| 	unregisterChan    chan *Client | 	unregisterChan    chan *Client | ||||||
| 	kickHandlerChan   chan *kickHandler | 	kickHandlerChan   chan *kickHandler | ||||||
| 	clients           UserMap | 	clients           UserMap | ||||||
| 	online            *rpccache.OnlineCache | 	online            rpccache.OnlineCache | ||||||
| 	subscription      *Subscription | 	subscription      *Subscription | ||||||
| 	clientPool        sync.Pool | 	clientPool        sync.Pool | ||||||
| 	onlineUserNum     atomic.Int64 | 	onlineUserNum     atomic.Int64 | ||||||
| @ -130,7 +130,7 @@ func NewWsServer(msgGatewayConfig *Config, opts ...Option) *WsServer { | |||||||
| 	for _, o := range opts { | 	for _, o := range opts { | ||||||
| 		o(&config) | 		o(&config) | ||||||
| 	} | 	} | ||||||
| 	//userRpcClient := rpcclient.NewUserRpcClient(client, config.Discovery.RpcService.User, config.Share.IMAdminUserID) | 	//userRpcClient := rpcclient.NewUserRpcClient(client, config.Discovery.RpcService.User, config.Share.IMAdminUser) | ||||||
| 
 | 
 | ||||||
| 	v := validator.New() | 	v := validator.New() | ||||||
| 	return &WsServer{ | 	return &WsServer{ | ||||||
| @ -334,6 +334,27 @@ func (ws *WsServer) multiTerminalLoginChecker(clientOK bool, oldClients []*Clien | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	// If reconnect: When multiple msgGateway instances are deployed, a client may disconnect from instance A and reconnect to instance B. | ||||||
|  | 	// During this process, instance A might still be executing, resulting in two clients with the same token existing simultaneously. | ||||||
|  | 	// This situation needs to be filtered to prevent duplicate clients. | ||||||
|  | 	checkSameTokenFunc := func(oldClients []*Client) []*Client { | ||||||
|  | 		var clientsNeedToKick []*Client | ||||||
|  | 
 | ||||||
|  | 		for _, c := range oldClients { | ||||||
|  | 			if c.token == newClient.token { | ||||||
|  | 				log.ZDebug(newClient.ctx, "token is same, not kick", | ||||||
|  | 					"userID", newClient.UserID, | ||||||
|  | 					"platformID", newClient.PlatformID, | ||||||
|  | 					"token", newClient.token) | ||||||
|  | 				continue | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			clientsNeedToKick = append(clientsNeedToKick, c) | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		return clientsNeedToKick | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	switch ws.msgGatewayConfig.Share.MultiLogin.Policy { | 	switch ws.msgGatewayConfig.Share.MultiLogin.Policy { | ||||||
| 	case constant.DefalutNotKick: | 	case constant.DefalutNotKick: | ||||||
| 	case constant.PCAndOther: | 	case constant.PCAndOther: | ||||||
| @ -349,11 +370,15 @@ func (ws *WsServer) multiTerminalLoginChecker(clientOK bool, oldClients []*Clien | |||||||
| 			} | 			} | ||||||
| 			oldClients = append(oldClients, c) | 			oldClients = append(oldClients, c) | ||||||
| 		} | 		} | ||||||
|  | 
 | ||||||
| 		fallthrough | 		fallthrough | ||||||
| 	case constant.AllLoginButSameTermKick: | 	case constant.AllLoginButSameTermKick: | ||||||
| 		if !clientOK { | 		if !clientOK { | ||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
|  | 
 | ||||||
|  | 		oldClients = checkSameTokenFunc(oldClients) | ||||||
|  | 
 | ||||||
| 		ws.clients.DeleteClients(newClient.UserID, oldClients) | 		ws.clients.DeleteClients(newClient.UserID, oldClients) | ||||||
| 		for _, c := range oldClients { | 		for _, c := range oldClients { | ||||||
| 			err := c.KickOnlineMessage() | 			err := c.KickOnlineMessage() | ||||||
| @ -361,6 +386,7 @@ func (ws *WsServer) multiTerminalLoginChecker(clientOK bool, oldClients []*Clien | |||||||
| 				log.ZWarn(c.ctx, "KickOnlineMessage", err) | 				log.ZWarn(c.ctx, "KickOnlineMessage", err) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  | 
 | ||||||
| 		ctx := mcontext.WithMustInfoCtx( | 		ctx := mcontext.WithMustInfoCtx( | ||||||
| 			[]string{newClient.ctx.GetOperationID(), newClient.ctx.GetUserID(), | 			[]string{newClient.ctx.GetOperationID(), newClient.ctx.GetUserID(), | ||||||
| 				constant.PlatformIDToName(newClient.PlatformID), newClient.ctx.GetConnID()}, | 				constant.PlatformIDToName(newClient.PlatformID), newClient.ctx.GetConnID()}, | ||||||
| @ -379,14 +405,17 @@ func (ws *WsServer) multiTerminalLoginChecker(clientOK bool, oldClients []*Clien | |||||||
| 		if !ok { | 		if !ok { | ||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
| 		var ( | 
 | ||||||
| 			kickClients []*Client | 		var kickClients []*Client | ||||||
| 		) |  | ||||||
| 		for _, client := range clients { | 		for _, client := range clients { | ||||||
| 			if constant.PlatformIDToClass(client.PlatformID) == constant.PlatformIDToClass(newClient.PlatformID) { | 			if constant.PlatformIDToClass(client.PlatformID) == constant.PlatformIDToClass(newClient.PlatformID) { | ||||||
| 				kickClients = append(kickClients, client) | 				{ | ||||||
|  | 					kickClients = append(kickClients, client) | ||||||
|  | 				} | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  | 		kickClients = checkSameTokenFunc(kickClients) | ||||||
|  | 
 | ||||||
| 		kickTokenFunc(kickClients) | 		kickTokenFunc(kickClients) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										132
									
								
								internal/msgtransfer/callback.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										132
									
								
								internal/msgtransfer/callback.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,132 @@ | |||||||
|  | package msgtransfer | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"context" | ||||||
|  | 	"encoding/base64" | ||||||
|  | 
 | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/apistruct" | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/common/webhook" | ||||||
|  | 	"github.com/openimsdk/protocol/constant" | ||||||
|  | 	"github.com/openimsdk/protocol/sdkws" | ||||||
|  | 	"github.com/openimsdk/tools/mcontext" | ||||||
|  | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
|  | 	"github.com/openimsdk/tools/utils/stringutil" | ||||||
|  | 	"google.golang.org/protobuf/proto" | ||||||
|  | 
 | ||||||
|  | 	cbapi "github.com/openimsdk/open-im-server/v3/pkg/callbackstruct" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func toCommonCallback(ctx context.Context, msg *sdkws.MsgData, command string) cbapi.CommonCallbackReq { | ||||||
|  | 	return cbapi.CommonCallbackReq{ | ||||||
|  | 		SendID:           msg.SendID, | ||||||
|  | 		ServerMsgID:      msg.ServerMsgID, | ||||||
|  | 		CallbackCommand:  command, | ||||||
|  | 		ClientMsgID:      msg.ClientMsgID, | ||||||
|  | 		OperationID:      mcontext.GetOperationID(ctx), | ||||||
|  | 		SenderPlatformID: msg.SenderPlatformID, | ||||||
|  | 		SenderNickname:   msg.SenderNickname, | ||||||
|  | 		SessionType:      msg.SessionType, | ||||||
|  | 		MsgFrom:          msg.MsgFrom, | ||||||
|  | 		ContentType:      msg.ContentType, | ||||||
|  | 		Status:           msg.Status, | ||||||
|  | 		SendTime:         msg.SendTime, | ||||||
|  | 		CreateTime:       msg.CreateTime, | ||||||
|  | 		AtUserIDList:     msg.AtUserIDList, | ||||||
|  | 		SenderFaceURL:    msg.SenderFaceURL, | ||||||
|  | 		Content:          GetContent(msg), | ||||||
|  | 		Seq:              uint32(msg.Seq), | ||||||
|  | 		Ex:               msg.Ex, | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func GetContent(msg *sdkws.MsgData) string { | ||||||
|  | 	if msg.ContentType >= constant.NotificationBegin && msg.ContentType <= constant.NotificationEnd { | ||||||
|  | 		var tips sdkws.TipsComm | ||||||
|  | 		_ = proto.Unmarshal(msg.Content, &tips) | ||||||
|  | 		content := tips.JsonDetail | ||||||
|  | 		return content | ||||||
|  | 	} else { | ||||||
|  | 		return string(msg.Content) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (mc *OnlineHistoryMongoConsumerHandler) webhookAfterSendSingleMsg(ctx context.Context, after *config.AfterConfig, msg *sdkws.MsgData) { | ||||||
|  | 	if msg.ContentType == constant.Typing { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if !filterAfterMsg(msg, after) { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	cbReq := &cbapi.CallbackAfterSendSingleMsgReq{ | ||||||
|  | 		CommonCallbackReq: toCommonCallback(ctx, msg, cbapi.CallbackAfterSendSingleMsgCommand), | ||||||
|  | 		RecvID:            msg.RecvID, | ||||||
|  | 	} | ||||||
|  | 	mc.webhookClient.AsyncPostWithQuery(ctx, cbReq.GetCallbackCommand(), cbReq, &cbapi.CallbackAfterSendSingleMsgResp{}, after, buildKeyMsgDataQuery(msg)) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (mc *OnlineHistoryMongoConsumerHandler) webhookAfterSendGroupMsg(ctx context.Context, after *config.AfterConfig, msg *sdkws.MsgData) { | ||||||
|  | 	if msg.ContentType == constant.Typing { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if !filterAfterMsg(msg, after) { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	cbReq := &cbapi.CallbackAfterSendGroupMsgReq{ | ||||||
|  | 		CommonCallbackReq: toCommonCallback(ctx, msg, cbapi.CallbackAfterSendGroupMsgCommand), | ||||||
|  | 		GroupID:           msg.GroupID, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	mc.webhookClient.AsyncPostWithQuery(ctx, cbReq.GetCallbackCommand(), cbReq, &cbapi.CallbackAfterSendGroupMsgResp{}, after, buildKeyMsgDataQuery(msg)) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func buildKeyMsgDataQuery(msg *sdkws.MsgData) map[string]string { | ||||||
|  | 	keyMsgData := apistruct.KeyMsgData{ | ||||||
|  | 		SendID:  msg.SendID, | ||||||
|  | 		RecvID:  msg.RecvID, | ||||||
|  | 		GroupID: msg.GroupID, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return map[string]string{ | ||||||
|  | 		webhook.Key: base64.StdEncoding.EncodeToString(stringutil.StructToJsonBytes(keyMsgData)), | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func filterAfterMsg(msg *sdkws.MsgData, after *config.AfterConfig) bool { | ||||||
|  | 	return filterMsg(msg, after.AttentionIds, after.DeniedTypes) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func filterMsg(msg *sdkws.MsgData, attentionIds []string, deniedTypes []int32) bool { | ||||||
|  | 	// According to the attentionIds configuration, only some users are sent | ||||||
|  | 	if len(attentionIds) != 0 && msg.ContentType == constant.SingleChatType && !datautil.Contain(msg.RecvID, attentionIds...) { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if len(attentionIds) != 0 && msg.ContentType == constant.ReadGroupChatType && !datautil.Contain(msg.GroupID, attentionIds...) { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if defaultDeniedTypes(msg.ContentType) { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if len(deniedTypes) != 0 && datautil.Contain(msg.ContentType, deniedTypes...) { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return true | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func defaultDeniedTypes(contentType int32) bool { | ||||||
|  | 	if contentType >= constant.NotificationBegin && contentType <= constant.NotificationEnd { | ||||||
|  | 		return true | ||||||
|  | 	} | ||||||
|  | 	if contentType == constant.Typing { | ||||||
|  | 		return true | ||||||
|  | 	} | ||||||
|  | 	return false | ||||||
|  | } | ||||||
| @ -58,7 +58,7 @@ type Config struct { | |||||||
| 	Index          conf.Index | 	Index          conf.Index | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func Start(ctx context.Context, config *Config, client discovery.Conn, server grpc.ServiceRegistrar) error { | func Start(ctx context.Context, config *Config, client discovery.SvcDiscoveryRegistry, server grpc.ServiceRegistrar) error { | ||||||
| 	builder := mqbuild.NewBuilder(&config.KafkaConfig) | 	builder := mqbuild.NewBuilder(&config.KafkaConfig) | ||||||
| 
 | 
 | ||||||
| 	log.CInfo(ctx, "MSG-TRANSFER server is initializing", "runTimeEnv", runtimeenv.RuntimeEnvironment(), "prometheusPorts", | 	log.CInfo(ctx, "MSG-TRANSFER server is initializing", "runTimeEnv", runtimeenv.RuntimeEnvironment(), "prometheusPorts", | ||||||
| @ -134,7 +134,7 @@ func Start(ctx context.Context, config *Config, client discovery.Conn, server gr | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	historyMongoHandler := NewOnlineHistoryMongoConsumerHandler(msgTransferDatabase) | 	historyMongoHandler := NewOnlineHistoryMongoConsumerHandler(msgTransferDatabase,config) | ||||||
| 
 | 
 | ||||||
| 	msgTransfer := &MsgTransfer{ | 	msgTransfer := &MsgTransfer{ | ||||||
| 		historyConsumer:      historyConsumer, | 		historyConsumer:      historyConsumer, | ||||||
|  | |||||||
| @ -19,6 +19,8 @@ import ( | |||||||
| 
 | 
 | ||||||
| 	"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/storage/controller" | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/common/webhook" | ||||||
|  | 	"github.com/openimsdk/protocol/constant" | ||||||
| 	pbmsg "github.com/openimsdk/protocol/msg" | 	pbmsg "github.com/openimsdk/protocol/msg" | ||||||
| 	"github.com/openimsdk/tools/log" | 	"github.com/openimsdk/tools/log" | ||||||
| 	"google.golang.org/protobuf/proto" | 	"google.golang.org/protobuf/proto" | ||||||
| @ -26,11 +28,15 @@ import ( | |||||||
| 
 | 
 | ||||||
| type OnlineHistoryMongoConsumerHandler struct { | type OnlineHistoryMongoConsumerHandler struct { | ||||||
| 	msgTransferDatabase controller.MsgTransferDatabase | 	msgTransferDatabase controller.MsgTransferDatabase | ||||||
|  | 	config              *Config | ||||||
|  | 	webhookClient       *webhook.Client | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func NewOnlineHistoryMongoConsumerHandler(database controller.MsgTransferDatabase) *OnlineHistoryMongoConsumerHandler { | func NewOnlineHistoryMongoConsumerHandler(database controller.MsgTransferDatabase, config *Config) *OnlineHistoryMongoConsumerHandler { | ||||||
| 	return &OnlineHistoryMongoConsumerHandler{ | 	return &OnlineHistoryMongoConsumerHandler{ | ||||||
| 		msgTransferDatabase: database, | 		msgTransferDatabase: database, | ||||||
|  | 		config:              config, | ||||||
|  | 		webhookClient:       webhook.NewWebhookClient(config.WebhooksConfig.URL), | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -48,19 +54,21 @@ func (mc *OnlineHistoryMongoConsumerHandler) HandleChatWs2Mongo(ctx context.Cont | |||||||
| 	log.ZDebug(ctx, "mongo consumer recv msg", "msgs", msgFromMQ.String()) | 	log.ZDebug(ctx, "mongo consumer recv msg", "msgs", msgFromMQ.String()) | ||||||
| 	err = mc.msgTransferDatabase.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, "single data insert to mongo err", err, "msg", msgFromMQ.MsgData, "conversationID", msgFromMQ.ConversationID) | ||||||
| 			ctx, |  | ||||||
| 			"single data insert to mongo err", |  | ||||||
| 			err, |  | ||||||
| 			"msg", |  | ||||||
| 			msgFromMQ.MsgData, |  | ||||||
| 			"conversationID", |  | ||||||
| 			msgFromMQ.ConversationID, |  | ||||||
| 		) |  | ||||||
| 		prommetrics.MsgInsertMongoFailedCounter.Inc() | 		prommetrics.MsgInsertMongoFailedCounter.Inc() | ||||||
| 	} else { | 	} else { | ||||||
| 		prommetrics.MsgInsertMongoSuccessCounter.Inc() | 		prommetrics.MsgInsertMongoSuccessCounter.Inc() | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
|  | 	for _, msgData := range msgFromMQ.MsgData { | ||||||
|  | 		switch msgData.SessionType { | ||||||
|  | 		case constant.SingleChatType: | ||||||
|  | 			mc.webhookAfterSendSingleMsg(ctx, &mc.config.WebhooksConfig.AfterSendSingleMsg, msgData) | ||||||
|  | 		case constant.ReadGroupChatType: | ||||||
|  | 			mc.webhookAfterSendGroupMsg(ctx, &mc.config.WebhooksConfig.AfterSendGroupMsg, msgData) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	//var seqs []int64 | 	//var seqs []int64 | ||||||
| 	//for _, msg := range msgFromMQ.MsgData { | 	//for _, msg := range msgFromMQ.MsgData { | ||||||
| 	//	seqs = append(seqs, msg.Seq) | 	//	seqs = append(seqs, msg.Seq) | ||||||
|  | |||||||
| @ -6,6 +6,7 @@ import ( | |||||||
| 	"strconv" | 	"strconv" | ||||||
| 
 | 
 | ||||||
| 	"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/authverify" | ||||||
| 	"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/open-im-server/v3/pkg/common/storage/cache/mcache" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache/mcache" | ||||||
| @ -49,7 +50,7 @@ func (p pushServer) DelUserPushToken(ctx context.Context, | |||||||
| 	return &pbpush.DelUserPushTokenResp{}, nil | 	return &pbpush.DelUserPushTokenResp{}, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func Start(ctx context.Context, config *Config, client discovery.Conn, server grpc.ServiceRegistrar) error { | func Start(ctx context.Context, config *Config, client discovery.SvcDiscoveryRegistry, server grpc.ServiceRegistrar) error { | ||||||
| 	dbb := dbbuild.NewBuilder(&config.MongoConfig, &config.RedisConfig) | 	dbb := dbbuild.NewBuilder(&config.MongoConfig, &config.RedisConfig) | ||||||
| 	rdb, err := dbb.Redis(ctx) | 	rdb, err := dbb.Redis(ctx) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -106,7 +107,7 @@ func Start(ctx context.Context, config *Config, client discovery.Conn, server gr | |||||||
| 	go func() { | 	go func() { | ||||||
| 		pushHandler.WaitCache() | 		pushHandler.WaitCache() | ||||||
| 		fn := func(ctx context.Context, key string, value []byte) error { | 		fn := func(ctx context.Context, key string, value []byte) error { | ||||||
| 			pushHandler.HandleMs2PsChat(ctx, value) | 			pushHandler.HandleMs2PsChat(authverify.WithTempAdmin(ctx), value) | ||||||
| 			return nil | 			return nil | ||||||
| 		} | 		} | ||||||
| 		consumerCtx := mcontext.SetOperationID(context.Background(), "push_"+strconv.Itoa(int(rand.Uint32()))) | 		consumerCtx := mcontext.SetOperationID(context.Background(), "push_"+strconv.Itoa(int(rand.Uint32()))) | ||||||
|  | |||||||
| @ -33,7 +33,7 @@ type ConsumerHandler struct { | |||||||
| 	offlinePusher          offlinepush.OfflinePusher | 	offlinePusher          offlinepush.OfflinePusher | ||||||
| 	onlinePusher           OnlinePusher | 	onlinePusher           OnlinePusher | ||||||
| 	pushDatabase           controller.PushDatabase | 	pushDatabase           controller.PushDatabase | ||||||
| 	onlineCache            *rpccache.OnlineCache | 	onlineCache            rpccache.OnlineCache | ||||||
| 	groupLocalCache        *rpccache.GroupLocalCache | 	groupLocalCache        *rpccache.GroupLocalCache | ||||||
| 	conversationLocalCache *rpccache.ConversationLocalCache | 	conversationLocalCache *rpccache.ConversationLocalCache | ||||||
| 	webhookClient          *webhook.Client | 	webhookClient          *webhook.Client | ||||||
| @ -120,11 +120,7 @@ func (c *ConsumerHandler) HandleMs2PsChat(ctx context.Context, msg []byte) { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *ConsumerHandler) WaitCache() { | func (c *ConsumerHandler) WaitCache() { | ||||||
| 	c.onlineCache.Lock.Lock() | 	c.onlineCache.WaitCache() | ||||||
| 	for c.onlineCache.CurrentPhase.Load() < rpccache.DoSubscribeOver { |  | ||||||
| 		c.onlineCache.Cond.Wait() |  | ||||||
| 	} |  | ||||||
| 	c.onlineCache.Lock.Unlock() |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // 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. | ||||||
| @ -321,8 +317,8 @@ func (c *ConsumerHandler) groupMessagesHandler(ctx context.Context, groupID stri | |||||||
| 					return err | 					return err | ||||||
| 				} | 				} | ||||||
| 				log.ZDebug(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.IMAdminUser.UserIDs) > 0 { | ||||||
| 					ctx = mcontext.WithOpUserIDContext(ctx, c.config.Share.IMAdminUserID[0]) | 					ctx = mcontext.WithOpUserIDContext(ctx, c.config.Share.IMAdminUser.UserIDs[0]) | ||||||
| 				} | 				} | ||||||
| 				defer func(groupID string) { | 				defer func(groupID string) { | ||||||
| 					if err := c.groupClient.DismissGroup(ctx, groupID, true); err != nil { | 					if err := c.groupClient.DismissGroup(ctx, groupID, true); err != nil { | ||||||
|  | |||||||
| @ -49,6 +49,7 @@ type authServer struct { | |||||||
| 	RegisterCenter discovery.Conn | 	RegisterCenter discovery.Conn | ||||||
| 	config         *Config | 	config         *Config | ||||||
| 	userClient     *rpcli.UserClient | 	userClient     *rpcli.UserClient | ||||||
|  | 	adminUserIDs   []string | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| type Config struct { | type Config struct { | ||||||
| @ -59,7 +60,7 @@ type Config struct { | |||||||
| 	Discovery   config.Discovery | 	Discovery   config.Discovery | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func Start(ctx context.Context, config *Config, client discovery.Conn, server grpc.ServiceRegistrar) error { | func Start(ctx context.Context, config *Config, client discovery.SvcDiscoveryRegistry, server grpc.ServiceRegistrar) error { | ||||||
| 	dbb := dbbuild.NewBuilder(&config.MongoConfig, &config.RedisConfig) | 	dbb := dbbuild.NewBuilder(&config.MongoConfig, &config.RedisConfig) | ||||||
| 	rdb, err := dbb.Redis(ctx) | 	rdb, err := dbb.Redis(ctx) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -90,10 +91,11 @@ func Start(ctx context.Context, config *Config, client discovery.Conn, server gr | |||||||
| 			config.Share.Secret, | 			config.Share.Secret, | ||||||
| 			config.RpcConfig.TokenPolicy.Expire, | 			config.RpcConfig.TokenPolicy.Expire, | ||||||
| 			config.Share.MultiLogin, | 			config.Share.MultiLogin, | ||||||
| 			config.Share.IMAdminUserID, | 			config.Share.IMAdminUser.UserIDs, | ||||||
| 		), | 		), | ||||||
| 		config:     config, | 		config:       config, | ||||||
| 		userClient: rpcli.NewUserClient(userConn), | 		userClient:   rpcli.NewUserClient(userConn), | ||||||
|  | 		adminUserIDs: config.Share.IMAdminUser.UserIDs, | ||||||
| 	}) | 	}) | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| @ -104,8 +106,8 @@ func (s *authServer) GetAdminToken(ctx context.Context, req *pbauth.GetAdminToke | |||||||
| 		return nil, errs.ErrNoPermission.WrapMsg("secret invalid") | 		return nil, errs.ErrNoPermission.WrapMsg("secret invalid") | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if !datautil.Contain(req.UserID, s.config.Share.IMAdminUserID...) { | 	if !datautil.Contain(req.UserID, s.adminUserIDs...) { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("userID is error.", "userID", req.UserID, "adminUserID", s.config.Share.IMAdminUserID) | 		return nil, errs.ErrArgs.WrapMsg("userID is error.", "userID", req.UserID, "adminUserID", s.adminUserIDs) | ||||||
| 
 | 
 | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -125,7 +127,7 @@ func (s *authServer) GetAdminToken(ctx context.Context, req *pbauth.GetAdminToke | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *authServer) GetUserToken(ctx context.Context, req *pbauth.GetUserTokenReq) (*pbauth.GetUserTokenResp, error) { | func (s *authServer) GetUserToken(ctx context.Context, req *pbauth.GetUserTokenReq) (*pbauth.GetUserTokenResp, error) { | ||||||
| 	if err := authverify.CheckAdmin(ctx, s.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAdmin(ctx); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -135,7 +137,7 @@ func (s *authServer) GetUserToken(ctx context.Context, req *pbauth.GetUserTokenR | |||||||
| 
 | 
 | ||||||
| 	resp := pbauth.GetUserTokenResp{} | 	resp := pbauth.GetUserTokenResp{} | ||||||
| 
 | 
 | ||||||
| 	if authverify.IsManagerUserID(req.UserID, s.config.Share.IMAdminUserID) { | 	if authverify.CheckUserIsAdmin(ctx, req.UserID) { | ||||||
| 		return nil, errs.ErrNoPermission.WrapMsg("don't get Admin token") | 		return nil, errs.ErrNoPermission.WrapMsg("don't get Admin token") | ||||||
| 	} | 	} | ||||||
| 	user, err := s.userClient.GetUserInfo(ctx, req.UserID) | 	user, err := s.userClient.GetUserInfo(ctx, req.UserID) | ||||||
| @ -159,15 +161,17 @@ func (s *authServer) parseToken(ctx context.Context, tokensString string) (claim | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	isAdmin := authverify.IsManagerUserID(claims.UserID, s.config.Share.IMAdminUserID) |  | ||||||
| 	if isAdmin { |  | ||||||
| 		return claims, nil |  | ||||||
| 	} |  | ||||||
| 	m, err := s.authDatabase.GetTokensWithoutError(ctx, claims.UserID, claims.PlatformID) | 	m, err := s.authDatabase.GetTokensWithoutError(ctx, claims.UserID, claims.PlatformID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	if len(m) == 0 { | 	if len(m) == 0 { | ||||||
|  | 		isAdmin := authverify.CheckUserIsAdmin(ctx, claims.UserID) | ||||||
|  | 		if isAdmin { | ||||||
|  | 			if err = s.authDatabase.GetTemporaryTokensWithoutError(ctx, claims.UserID, claims.PlatformID, tokensString); err == nil { | ||||||
|  | 				return claims, nil | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
| 		return nil, servererrs.ErrTokenNotExist.Wrap() | 		return nil, servererrs.ErrTokenNotExist.Wrap() | ||||||
| 	} | 	} | ||||||
| 	if v, ok := m[tokensString]; ok { | 	if v, ok := m[tokensString]; ok { | ||||||
| @ -179,6 +183,13 @@ func (s *authServer) parseToken(ctx context.Context, tokensString string) (claim | |||||||
| 		default: | 		default: | ||||||
| 			return nil, errs.Wrap(errs.ErrTokenUnknown) | 			return nil, errs.Wrap(errs.ErrTokenUnknown) | ||||||
| 		} | 		} | ||||||
|  | 	} else { | ||||||
|  | 		isAdmin := authverify.CheckUserIsAdmin(ctx, claims.UserID) | ||||||
|  | 		if isAdmin { | ||||||
|  | 			if err = s.authDatabase.GetTemporaryTokensWithoutError(ctx, claims.UserID, claims.PlatformID, tokensString); err == nil { | ||||||
|  | 				return claims, nil | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 	return nil, servererrs.ErrTokenNotExist.Wrap() | 	return nil, servererrs.ErrTokenNotExist.Wrap() | ||||||
| } | } | ||||||
| @ -196,7 +207,7 @@ func (s *authServer) ParseToken(ctx context.Context, req *pbauth.ParseTokenReq) | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *authServer) ForceLogout(ctx context.Context, req *pbauth.ForceLogoutReq) (*pbauth.ForceLogoutResp, error) { | func (s *authServer) ForceLogout(ctx context.Context, req *pbauth.ForceLogoutReq) (*pbauth.ForceLogoutResp, error) { | ||||||
| 	if err := authverify.CheckAdmin(ctx, s.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAdmin(ctx); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	if err := s.forceKickOff(ctx, req.UserID, req.PlatformID); err != nil { | 	if err := s.forceKickOff(ctx, req.UserID, req.PlatformID); err != nil { | ||||||
|  | |||||||
							
								
								
									
										117
									
								
								internal/rpc/conversation/callback.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										117
									
								
								internal/rpc/conversation/callback.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,117 @@ | |||||||
|  | package conversation | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"context" | ||||||
|  | 
 | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/callbackstruct" | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||||
|  | 	dbModel "github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/common/webhook" | ||||||
|  | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func (c *conversationServer) webhookBeforeCreateSingleChatConversations(ctx context.Context, before *config.BeforeConfig, req *dbModel.Conversation) error { | ||||||
|  | 	return webhook.WithCondition(ctx, before, func(ctx context.Context) error { | ||||||
|  | 		cbReq := &callbackstruct.CallbackBeforeCreateSingleChatConversationsReq{ | ||||||
|  | 			CallbackCommand:  callbackstruct.CallbackBeforeCreateSingleChatConversationsCommand, | ||||||
|  | 			OwnerUserID:      req.OwnerUserID, | ||||||
|  | 			ConversationID:   req.ConversationID, | ||||||
|  | 			ConversationType: req.ConversationType, | ||||||
|  | 			UserID:           req.UserID, | ||||||
|  | 			RecvMsgOpt:       req.RecvMsgOpt, | ||||||
|  | 			IsPinned:         req.IsPinned, | ||||||
|  | 			IsPrivateChat:    req.IsPrivateChat, | ||||||
|  | 			BurnDuration:     req.BurnDuration, | ||||||
|  | 			GroupAtType:      req.GroupAtType, | ||||||
|  | 			AttachedInfo:     req.AttachedInfo, | ||||||
|  | 			Ex:               req.Ex, | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		resp := &callbackstruct.CallbackBeforeCreateSingleChatConversationsResp{} | ||||||
|  | 
 | ||||||
|  | 		if err := c.webhookClient.SyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, resp, before); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		datautil.NotNilReplace(&req.RecvMsgOpt, resp.RecvMsgOpt) | ||||||
|  | 		datautil.NotNilReplace(&req.IsPinned, resp.IsPinned) | ||||||
|  | 		datautil.NotNilReplace(&req.IsPrivateChat, resp.IsPrivateChat) | ||||||
|  | 		datautil.NotNilReplace(&req.BurnDuration, resp.BurnDuration) | ||||||
|  | 		datautil.NotNilReplace(&req.GroupAtType, resp.GroupAtType) | ||||||
|  | 		datautil.NotNilReplace(&req.AttachedInfo, resp.AttachedInfo) | ||||||
|  | 		datautil.NotNilReplace(&req.Ex, resp.Ex) | ||||||
|  | 		return nil | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (c *conversationServer) webhookAfterCreateSingleChatConversations(ctx context.Context, after *config.AfterConfig, req *dbModel.Conversation) error { | ||||||
|  | 	cbReq := &callbackstruct.CallbackAfterCreateSingleChatConversationsReq{ | ||||||
|  | 		CallbackCommand:  callbackstruct.CallbackAfterCreateSingleChatConversationsCommand, | ||||||
|  | 		OwnerUserID:      req.OwnerUserID, | ||||||
|  | 		ConversationID:   req.ConversationID, | ||||||
|  | 		ConversationType: req.ConversationType, | ||||||
|  | 		UserID:           req.UserID, | ||||||
|  | 		RecvMsgOpt:       req.RecvMsgOpt, | ||||||
|  | 		IsPinned:         req.IsPinned, | ||||||
|  | 		IsPrivateChat:    req.IsPrivateChat, | ||||||
|  | 		BurnDuration:     req.BurnDuration, | ||||||
|  | 		GroupAtType:      req.GroupAtType, | ||||||
|  | 		AttachedInfo:     req.AttachedInfo, | ||||||
|  | 		Ex:               req.Ex, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	c.webhookClient.AsyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, &callbackstruct.CallbackAfterCreateSingleChatConversationsResp{}, after) | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (c *conversationServer) webhookBeforeCreateGroupChatConversations(ctx context.Context, before *config.BeforeConfig, req *dbModel.Conversation) error { | ||||||
|  | 	return webhook.WithCondition(ctx, before, func(ctx context.Context) error { | ||||||
|  | 		cbReq := &callbackstruct.CallbackBeforeCreateGroupChatConversationsReq{ | ||||||
|  | 			CallbackCommand:  callbackstruct.CallbackBeforeCreateGroupChatConversationsCommand, | ||||||
|  | 			ConversationID:   req.ConversationID, | ||||||
|  | 			ConversationType: req.ConversationType, | ||||||
|  | 			GroupID:          req.GroupID, | ||||||
|  | 			RecvMsgOpt:       req.RecvMsgOpt, | ||||||
|  | 			IsPinned:         req.IsPinned, | ||||||
|  | 			IsPrivateChat:    req.IsPrivateChat, | ||||||
|  | 			BurnDuration:     req.BurnDuration, | ||||||
|  | 			GroupAtType:      req.GroupAtType, | ||||||
|  | 			AttachedInfo:     req.AttachedInfo, | ||||||
|  | 			Ex:               req.Ex, | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		resp := &callbackstruct.CallbackBeforeCreateGroupChatConversationsResp{} | ||||||
|  | 
 | ||||||
|  | 		if err := c.webhookClient.SyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, resp, before); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		datautil.NotNilReplace(&req.RecvMsgOpt, resp.RecvMsgOpt) | ||||||
|  | 		datautil.NotNilReplace(&req.IsPinned, resp.IsPinned) | ||||||
|  | 		datautil.NotNilReplace(&req.IsPrivateChat, resp.IsPrivateChat) | ||||||
|  | 		datautil.NotNilReplace(&req.BurnDuration, resp.BurnDuration) | ||||||
|  | 		datautil.NotNilReplace(&req.GroupAtType, resp.GroupAtType) | ||||||
|  | 		datautil.NotNilReplace(&req.AttachedInfo, resp.AttachedInfo) | ||||||
|  | 		datautil.NotNilReplace(&req.Ex, resp.Ex) | ||||||
|  | 		return nil | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (c *conversationServer) webhookAfterCreateGroupChatConversations(ctx context.Context, after *config.AfterConfig, req *dbModel.Conversation) error { | ||||||
|  | 	cbReq := &callbackstruct.CallbackAfterCreateGroupChatConversationsReq{ | ||||||
|  | 		CallbackCommand:  callbackstruct.CallbackAfterCreateGroupChatConversationsCommand, | ||||||
|  | 		ConversationID:   req.ConversationID, | ||||||
|  | 		ConversationType: req.ConversationType, | ||||||
|  | 		GroupID:          req.GroupID, | ||||||
|  | 		RecvMsgOpt:       req.RecvMsgOpt, | ||||||
|  | 		IsPinned:         req.IsPinned, | ||||||
|  | 		IsPrivateChat:    req.IsPrivateChat, | ||||||
|  | 		BurnDuration:     req.BurnDuration, | ||||||
|  | 		GroupAtType:      req.GroupAtType, | ||||||
|  | 		AttachedInfo:     req.AttachedInfo, | ||||||
|  | 		Ex:               req.Ex, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	c.webhookClient.AsyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, &callbackstruct.CallbackAfterCreateGroupChatConversationsResp{}, after) | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
| @ -19,17 +19,20 @@ import ( | |||||||
| 	"sort" | 	"sort" | ||||||
| 	"time" | 	"time" | ||||||
| 
 | 
 | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/dbbuild" | 	"github.com/openimsdk/open-im-server/v3/pkg/dbbuild" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | ||||||
| 
 | 
 | ||||||
|  | 	"google.golang.org/grpc" | ||||||
|  | 
 | ||||||
| 	"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/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/servererrs" | ||||||
| 	"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/controller" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/controller" | ||||||
| 	"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/open-im-server/v3/pkg/common/storage/model" |  | ||||||
| 	dbModel "github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | 	dbModel "github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/common/webhook" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/localcache" | 	"github.com/openimsdk/open-im-server/v3/pkg/localcache" | ||||||
| 	"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" | ||||||
| @ -39,7 +42,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/utils/datautil" | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
| 	"google.golang.org/grpc" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| type conversationServer struct { | type conversationServer struct { | ||||||
| @ -49,9 +51,10 @@ type conversationServer struct { | |||||||
| 	conversationNotificationSender *ConversationNotificationSender | 	conversationNotificationSender *ConversationNotificationSender | ||||||
| 	config                         *Config | 	config                         *Config | ||||||
| 
 | 
 | ||||||
| 	userClient  *rpcli.UserClient | 	webhookClient *webhook.Client | ||||||
| 	msgClient   *rpcli.MsgClient | 	userClient    *rpcli.UserClient | ||||||
| 	groupClient *rpcli.GroupClient | 	msgClient     *rpcli.MsgClient | ||||||
|  | 	groupClient   *rpcli.GroupClient | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| type Config struct { | type Config struct { | ||||||
| @ -60,11 +63,12 @@ type Config struct { | |||||||
| 	MongodbConfig      config.Mongo | 	MongodbConfig      config.Mongo | ||||||
| 	NotificationConfig config.Notification | 	NotificationConfig config.Notification | ||||||
| 	Share              config.Share | 	Share              config.Share | ||||||
|  | 	WebhooksConfig     config.Webhooks | ||||||
| 	LocalCacheConfig   config.LocalCache | 	LocalCacheConfig   config.LocalCache | ||||||
| 	Discovery          config.Discovery | 	Discovery          config.Discovery | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func Start(ctx context.Context, config *Config, client discovery.Conn, server grpc.ServiceRegistrar) error { | func Start(ctx context.Context, config *Config, client discovery.SvcDiscoveryRegistry, server grpc.ServiceRegistrar) error { | ||||||
| 	dbb := dbbuild.NewBuilder(&config.MongodbConfig, &config.RedisConfig) | 	dbb := dbbuild.NewBuilder(&config.MongodbConfig, &config.RedisConfig) | ||||||
| 	mgocli, err := dbb.Mongo(ctx) | 	mgocli, err := dbb.Mongo(ctx) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -90,20 +94,32 @@ func Start(ctx context.Context, config *Config, client discovery.Conn, server gr | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	msgClient := rpcli.NewMsgClient(msgConn) | 	msgClient := rpcli.NewMsgClient(msgConn) | ||||||
|  | 
 | ||||||
|  | 	cs := conversationServer{ | ||||||
|  | 		config:        config, | ||||||
|  | 		webhookClient: webhook.NewWebhookClient(config.WebhooksConfig.URL), | ||||||
|  | 		userClient:    rpcli.NewUserClient(userConn), | ||||||
|  | 		groupClient:   rpcli.NewGroupClient(groupConn), | ||||||
|  | 		msgClient:     msgClient, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	cs.conversationNotificationSender = NewConversationNotificationSender(&config.NotificationConfig, msgClient) | ||||||
|  | 	cs.conversationDatabase = controller.NewConversationDatabase( | ||||||
|  | 		conversationDB, | ||||||
|  | 		redis.NewConversationRedis(rdb, &config.LocalCacheConfig, conversationDB), | ||||||
|  | 		mgocli.GetTx()) | ||||||
|  | 
 | ||||||
| 	localcache.InitLocalCache(&config.LocalCacheConfig) | 	localcache.InitLocalCache(&config.LocalCacheConfig) | ||||||
| 	pbconversation.RegisterConversationServer(server, &conversationServer{ | 	pbconversation.RegisterConversationServer(server, &cs) | ||||||
| 		conversationNotificationSender: NewConversationNotificationSender(&config.NotificationConfig, msgClient), |  | ||||||
| 		conversationDatabase: controller.NewConversationDatabase(conversationDB, |  | ||||||
| 			redis.NewConversationRedis(rdb, &config.LocalCacheConfig, conversationDB), mgocli.GetTx()), |  | ||||||
| 		userClient:  rpcli.NewUserClient(userConn), |  | ||||||
| 		groupClient: rpcli.NewGroupClient(groupConn), |  | ||||||
| 		msgClient:   msgClient, |  | ||||||
| 	}) |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *conversationServer) GetConversation(ctx context.Context, req *pbconversation.GetConversationReq) (*pbconversation.GetConversationResp, error) { | func (c *conversationServer) GetConversation(ctx context.Context, req *pbconversation.GetConversationReq) (*pbconversation.GetConversationResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.OwnerUserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	conversations, err := c.conversationDatabase.FindConversations(ctx, req.OwnerUserID, []string{req.ConversationID}) | 	conversations, err := c.conversationDatabase.FindConversations(ctx, req.OwnerUserID, []string{req.ConversationID}) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -117,7 +133,9 @@ func (c *conversationServer) GetConversation(ctx context.Context, req *pbconvers | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *conversationServer) GetSortedConversationList(ctx context.Context, req *pbconversation.GetSortedConversationListReq) (resp *pbconversation.GetSortedConversationListResp, err error) { | func (c *conversationServer) GetSortedConversationList(ctx context.Context, req *pbconversation.GetSortedConversationListReq) (resp *pbconversation.GetSortedConversationListResp, err error) { | ||||||
| 	log.ZDebug(ctx, "GetSortedConversationList", "seqs", req, "userID", req.UserID) | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	var conversationIDs []string | 	var conversationIDs []string | ||||||
| 	if len(req.ConversationIDs) == 0 { | 	if len(req.ConversationIDs) == 0 { | ||||||
| 		conversationIDs, err = c.conversationDatabase.GetConversationIDs(ctx, req.UserID) | 		conversationIDs, err = c.conversationDatabase.GetConversationIDs(ctx, req.UserID) | ||||||
| @ -190,6 +208,9 @@ func (c *conversationServer) GetSortedConversationList(ctx context.Context, req | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *conversationServer) GetAllConversations(ctx context.Context, req *pbconversation.GetAllConversationsReq) (*pbconversation.GetAllConversationsResp, error) { | func (c *conversationServer) GetAllConversations(ctx context.Context, req *pbconversation.GetAllConversationsReq) (*pbconversation.GetAllConversationsResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.OwnerUserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	conversations, err := c.conversationDatabase.GetUserAllConversation(ctx, req.OwnerUserID) | 	conversations, err := c.conversationDatabase.GetUserAllConversation(ctx, req.OwnerUserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -200,6 +221,9 @@ func (c *conversationServer) GetAllConversations(ctx context.Context, req *pbcon | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *conversationServer) GetConversations(ctx context.Context, req *pbconversation.GetConversationsReq) (*pbconversation.GetConversationsResp, error) { | func (c *conversationServer) GetConversations(ctx context.Context, req *pbconversation.GetConversationsReq) (*pbconversation.GetConversationsResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.OwnerUserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	conversations, err := c.getConversations(ctx, req.OwnerUserID, req.ConversationIDs) | 	conversations, err := c.getConversations(ctx, req.OwnerUserID, req.ConversationIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -219,8 +243,14 @@ func (c *conversationServer) getConversations(ctx context.Context, ownerUserID s | |||||||
| 	return convert.ConversationsDB2Pb(conversations), nil | 	return convert.ConversationsDB2Pb(conversations), nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // Deprecated | ||||||
| func (c *conversationServer) SetConversation(ctx context.Context, req *pbconversation.SetConversationReq) (*pbconversation.SetConversationResp, error) { | func (c *conversationServer) SetConversation(ctx context.Context, req *pbconversation.SetConversationReq) (*pbconversation.SetConversationResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.GetConversation().GetUserID()); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	var conversation dbModel.Conversation | 	var conversation dbModel.Conversation | ||||||
|  | 	conversation.CreateTime = time.Now() | ||||||
|  | 
 | ||||||
| 	if err := datautil.CopyStructFields(&conversation, req.Conversation); err != nil { | 	if err := datautil.CopyStructFields(&conversation, req.Conversation); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -234,8 +264,10 @@ func (c *conversationServer) SetConversation(ctx context.Context, req *pbconvers | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *conversationServer) SetConversations(ctx context.Context, req *pbconversation.SetConversationsReq) (*pbconversation.SetConversationsResp, error) { | func (c *conversationServer) SetConversations(ctx context.Context, req *pbconversation.SetConversationsReq) (*pbconversation.SetConversationsResp, error) { | ||||||
| 	if req.Conversation == nil { | 	for _, userID := range req.UserIDs { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("conversation must not be nil") | 		if err := authverify.CheckAccess(ctx, userID); err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 	if req.Conversation.ConversationType == constant.WriteGroupChatType { | 	if req.Conversation.ConversationType == constant.WriteGroupChatType { | ||||||
| 		groupInfo, err := c.groupClient.GetGroupInfo(ctx, req.Conversation.GroupID) | 		groupInfo, err := c.groupClient.GetGroupInfo(ctx, req.Conversation.GroupID) | ||||||
| @ -270,110 +302,31 @@ func (c *conversationServer) SetConversations(ctx context.Context, req *pbconver | |||||||
| 	conversation.ConversationType = req.Conversation.ConversationType | 	conversation.ConversationType = req.Conversation.ConversationType | ||||||
| 	conversation.UserID = req.Conversation.UserID | 	conversation.UserID = req.Conversation.UserID | ||||||
| 	conversation.GroupID = req.Conversation.GroupID | 	conversation.GroupID = req.Conversation.GroupID | ||||||
|  | 	conversation.CreateTime = time.Now() | ||||||
| 
 | 
 | ||||||
| 	m := make(map[string]any) | 	m, conversation, err := UpdateConversationsMap(ctx, req) | ||||||
| 
 | 	if err != nil { | ||||||
| 	setConversationFieldsFunc := func() { | 		return nil, err | ||||||
| 		if req.Conversation.RecvMsgOpt != nil { |  | ||||||
| 			conversation.RecvMsgOpt = req.Conversation.RecvMsgOpt.Value |  | ||||||
| 			m["recv_msg_opt"] = req.Conversation.RecvMsgOpt.Value |  | ||||||
| 		} |  | ||||||
| 		if req.Conversation.AttachedInfo != nil { |  | ||||||
| 			conversation.AttachedInfo = req.Conversation.AttachedInfo.Value |  | ||||||
| 			m["attached_info"] = req.Conversation.AttachedInfo.Value |  | ||||||
| 		} |  | ||||||
| 		if req.Conversation.Ex != nil { |  | ||||||
| 			conversation.Ex = req.Conversation.Ex.Value |  | ||||||
| 			m["ex"] = req.Conversation.Ex.Value |  | ||||||
| 		} |  | ||||||
| 		if req.Conversation.IsPinned != nil { |  | ||||||
| 			conversation.IsPinned = req.Conversation.IsPinned.Value |  | ||||||
| 			m["is_pinned"] = req.Conversation.IsPinned.Value |  | ||||||
| 		} |  | ||||||
| 		if req.Conversation.GroupAtType != nil { |  | ||||||
| 			conversation.GroupAtType = req.Conversation.GroupAtType.Value |  | ||||||
| 			m["group_at_type"] = req.Conversation.GroupAtType.Value |  | ||||||
| 		} |  | ||||||
| 		if req.Conversation.MsgDestructTime != nil { |  | ||||||
| 			conversation.MsgDestructTime = req.Conversation.MsgDestructTime.Value |  | ||||||
| 			m["msg_destruct_time"] = req.Conversation.MsgDestructTime.Value |  | ||||||
| 		} |  | ||||||
| 		if req.Conversation.IsMsgDestruct != nil { |  | ||||||
| 			conversation.IsMsgDestruct = req.Conversation.IsMsgDestruct.Value |  | ||||||
| 			m["is_msg_destruct"] = req.Conversation.IsMsgDestruct.Value |  | ||||||
| 		} |  | ||||||
| 		if req.Conversation.BurnDuration != nil { |  | ||||||
| 			conversation.BurnDuration = req.Conversation.BurnDuration.Value |  | ||||||
| 			m["burn_duration"] = req.Conversation.BurnDuration.Value |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	// set need set field in conversation |  | ||||||
| 	setConversationFieldsFunc() |  | ||||||
| 
 |  | ||||||
| 	for userID := range conversationMap { | 	for userID := range conversationMap { | ||||||
| 		unequal := len(m) | 		unequal := UserUpdateCheckMap(ctx, userID, req.Conversation, conversationMap[userID]) | ||||||
| 
 | 
 | ||||||
| 		if req.Conversation.RecvMsgOpt != nil { | 		if unequal { | ||||||
| 			if req.Conversation.RecvMsgOpt.Value == conversationMap[userID].RecvMsgOpt { |  | ||||||
| 				unequal-- |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		if req.Conversation.AttachedInfo != nil { |  | ||||||
| 			if req.Conversation.AttachedInfo.Value == conversationMap[userID].AttachedInfo { |  | ||||||
| 				unequal-- |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		if req.Conversation.Ex != nil { |  | ||||||
| 			if req.Conversation.Ex.Value == conversationMap[userID].Ex { |  | ||||||
| 				unequal-- |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		if req.Conversation.IsPinned != nil { |  | ||||||
| 			if req.Conversation.IsPinned.Value == conversationMap[userID].IsPinned { |  | ||||||
| 				unequal-- |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		if req.Conversation.GroupAtType != nil { |  | ||||||
| 			if req.Conversation.GroupAtType.Value == conversationMap[userID].GroupAtType { |  | ||||||
| 				unequal-- |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		if req.Conversation.MsgDestructTime != nil { |  | ||||||
| 			if req.Conversation.MsgDestructTime.Value == conversationMap[userID].MsgDestructTime { |  | ||||||
| 				unequal-- |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		if req.Conversation.IsMsgDestruct != nil { |  | ||||||
| 			if req.Conversation.IsMsgDestruct.Value == conversationMap[userID].IsMsgDestruct { |  | ||||||
| 				unequal-- |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		if req.Conversation.BurnDuration != nil { |  | ||||||
| 			if req.Conversation.BurnDuration.Value == conversationMap[userID].BurnDuration { |  | ||||||
| 				unequal-- |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		if unequal > 0 { |  | ||||||
| 			needUpdateUsersList = append(needUpdateUsersList, userID) | 			needUpdateUsersList = append(needUpdateUsersList, userID) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	if len(m) != 0 && len(needUpdateUsersList) != 0 { | 	if len(m) != 0 && len(needUpdateUsersList) != 0 { | ||||||
| 		if err := c.conversationDatabase.SetUsersConversationFieldTx(ctx, needUpdateUsersList, &conversation, m); err != nil { | 		if err := c.conversationDatabase.SetUsersConversationFieldTx(ctx, needUpdateUsersList, &conversation, m); err != nil { | ||||||
| 			return nil, err | 			return nil, err | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		for _, v := range needUpdateUsersList { | 		for _, userID := range needUpdateUsersList { | ||||||
| 			c.conversationNotificationSender.ConversationChangeNotification(ctx, v, []string{req.Conversation.ConversationID}) | 			c.conversationNotificationSender.ConversationChangeNotification(ctx, userID, []string{req.Conversation.ConversationID}) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	if req.Conversation.IsPrivateChat != nil && req.Conversation.ConversationType != constant.ReadGroupChatType { | 	if req.Conversation.IsPrivateChat != nil && req.Conversation.ConversationType != constant.ReadGroupChatType { | ||||||
| 		var conversations []*dbModel.Conversation | 		var conversations []*dbModel.Conversation | ||||||
| 		for _, ownerUserID := range req.UserIDs { | 		for _, ownerUserID := range req.UserIDs { | ||||||
| @ -396,58 +349,103 @@ func (c *conversationServer) SetConversations(ctx context.Context, req *pbconver | |||||||
| 	return &pbconversation.SetConversationsResp{}, nil | 	return &pbconversation.SetConversationsResp{}, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Get user IDs with "Do Not Disturb" enabled in super large groups. | func (c *conversationServer) UpdateConversationsByUser(ctx context.Context, req *pbconversation.UpdateConversationsByUserReq) (*pbconversation.UpdateConversationsByUserResp, error) { | ||||||
| func (c *conversationServer) GetRecvMsgNotNotifyUserIDs(ctx context.Context, req *pbconversation.GetRecvMsgNotNotifyUserIDsReq) (*pbconversation.GetRecvMsgNotNotifyUserIDsResp, error) { | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
| 	return nil, errs.New("deprecated") | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	m := make(map[string]any) | ||||||
|  | 	if req.Ex != nil { | ||||||
|  | 		m["ex"] = req.Ex.Value | ||||||
|  | 	} | ||||||
|  | 	if len(m) > 0 { | ||||||
|  | 		if err := c.conversationDatabase.UpdateUserConversations(ctx, req.UserID, m); err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return &pbconversation.UpdateConversationsByUserResp{}, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // create conversation without notification for msg redis transfer. | // create conversation without notification for msg redis transfer. | ||||||
| func (c *conversationServer) CreateSingleChatConversations(ctx context.Context, | func (c *conversationServer) CreateSingleChatConversations(ctx context.Context, req *pbconversation.CreateSingleChatConversationsReq) (*pbconversation.CreateSingleChatConversationsResp, error) { | ||||||
| 	req *pbconversation.CreateSingleChatConversationsReq, | 	var conversation dbModel.Conversation | ||||||
| ) (*pbconversation.CreateSingleChatConversationsResp, error) { | 	conversation.CreateTime = time.Now() | ||||||
|  | 
 | ||||||
| 	switch req.ConversationType { | 	switch req.ConversationType { | ||||||
| 	case constant.SingleChatType: | 	case constant.SingleChatType: | ||||||
| 		var conversation dbModel.Conversation | 		// sendUser create | ||||||
| 		conversation.ConversationID = req.ConversationID | 		conversation.ConversationID = req.ConversationID | ||||||
| 		conversation.ConversationType = req.ConversationType | 		conversation.ConversationType = req.ConversationType | ||||||
| 		conversation.OwnerUserID = req.SendID | 		conversation.OwnerUserID = req.SendID | ||||||
| 		conversation.UserID = req.RecvID | 		conversation.UserID = req.RecvID | ||||||
|  | 
 | ||||||
|  | 		if err := c.webhookBeforeCreateSingleChatConversations(ctx, &c.config.WebhooksConfig.BeforeCreateSingleChatConversations, &conversation); err != nil && err != servererrs.ErrCallbackContinue { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
| 		err := c.conversationDatabase.CreateConversation(ctx, []*dbModel.Conversation{&conversation}) | 		err := c.conversationDatabase.CreateConversation(ctx, []*dbModel.Conversation{&conversation}) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.ZWarn(ctx, "create conversation failed", err, "conversation", conversation) | 			log.ZWarn(ctx, "create conversation failed", err, "conversation", conversation) | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
|  | 		c.webhookAfterCreateSingleChatConversations(ctx, &c.config.WebhooksConfig.AfterCreateSingleChatConversations, &conversation) | ||||||
|  | 
 | ||||||
|  | 		// recvUser create | ||||||
| 		conversation2 := conversation | 		conversation2 := conversation | ||||||
| 		conversation2.OwnerUserID = req.RecvID | 		conversation2.OwnerUserID = req.RecvID | ||||||
| 		conversation2.UserID = req.SendID | 		conversation2.UserID = req.SendID | ||||||
|  | 
 | ||||||
|  | 		if err := c.webhookBeforeCreateSingleChatConversations(ctx, &c.config.WebhooksConfig.BeforeCreateSingleChatConversations, &conversation); err != nil && err != servererrs.ErrCallbackContinue { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
| 		err = c.conversationDatabase.CreateConversation(ctx, []*dbModel.Conversation{&conversation2}) | 		err = c.conversationDatabase.CreateConversation(ctx, []*dbModel.Conversation{&conversation2}) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.ZWarn(ctx, "create conversation failed", err, "conversation2", conversation) | 			log.ZWarn(ctx, "create conversation failed", err, "conversation2", conversation) | ||||||
| 		} | 		} | ||||||
|  | 
 | ||||||
|  | 		c.webhookAfterCreateSingleChatConversations(ctx, &c.config.WebhooksConfig.AfterCreateSingleChatConversations, &conversation2) | ||||||
| 	case constant.NotificationChatType: | 	case constant.NotificationChatType: | ||||||
| 		var conversation dbModel.Conversation |  | ||||||
| 		conversation.ConversationID = req.ConversationID | 		conversation.ConversationID = req.ConversationID | ||||||
| 		conversation.ConversationType = req.ConversationType | 		conversation.ConversationType = req.ConversationType | ||||||
| 		conversation.OwnerUserID = req.RecvID | 		conversation.OwnerUserID = req.RecvID | ||||||
| 		conversation.UserID = req.SendID | 		conversation.UserID = req.SendID | ||||||
|  | 
 | ||||||
|  | 		if err := c.webhookBeforeCreateSingleChatConversations(ctx, &c.config.WebhooksConfig.BeforeCreateSingleChatConversations, &conversation); err != nil && err != servererrs.ErrCallbackContinue { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
| 		err := c.conversationDatabase.CreateConversation(ctx, []*dbModel.Conversation{&conversation}) | 		err := c.conversationDatabase.CreateConversation(ctx, []*dbModel.Conversation{&conversation}) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.ZWarn(ctx, "create conversation failed", err, "conversation2", conversation) | 			log.ZWarn(ctx, "create conversation failed", err, "conversation2", conversation) | ||||||
| 		} | 		} | ||||||
|  | 
 | ||||||
|  | 		c.webhookAfterCreateSingleChatConversations(ctx, &c.config.WebhooksConfig.AfterCreateSingleChatConversations, &conversation) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	return &pbconversation.CreateSingleChatConversationsResp{}, nil | 	return &pbconversation.CreateSingleChatConversationsResp{}, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *conversationServer) CreateGroupChatConversations(ctx context.Context, req *pbconversation.CreateGroupChatConversationsReq) (*pbconversation.CreateGroupChatConversationsResp, error) { | func (c *conversationServer) CreateGroupChatConversations(ctx context.Context, req *pbconversation.CreateGroupChatConversationsReq) (*pbconversation.CreateGroupChatConversationsResp, error) { | ||||||
| 	err := c.conversationDatabase.CreateGroupChatConversation(ctx, req.GroupID, req.UserIDs) | 	var conversation dbModel.Conversation | ||||||
|  | 
 | ||||||
|  | 	conversation.ConversationID = msgprocessor.GetConversationIDBySessionType(constant.ReadGroupChatType, req.GroupID) | ||||||
|  | 	conversation.GroupID = req.GroupID | ||||||
|  | 	conversation.ConversationType = constant.ReadGroupChatType | ||||||
|  | 	conversation.CreateTime = time.Now() | ||||||
|  | 
 | ||||||
|  | 	if err := c.webhookBeforeCreateGroupChatConversations(ctx, &c.config.WebhooksConfig.BeforeCreateGroupChatConversations, &conversation); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err := c.conversationDatabase.CreateGroupChatConversation(ctx, req.GroupID, req.UserIDs, &conversation) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	conversationID := msgprocessor.GetConversationIDBySessionType(constant.ReadGroupChatType, req.GroupID) | 	if err := c.msgClient.SetUserConversationMaxSeq(ctx, conversation.ConversationID, req.UserIDs, 0); err != nil { | ||||||
| 	if err := c.msgClient.SetUserConversationMaxSeq(ctx, conversationID, req.UserIDs, 0); err != nil { |  | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
|  | 	c.webhookAfterCreateGroupChatConversations(ctx, &c.config.WebhooksConfig.AfterCreateGroupChatConversations, &conversation) | ||||||
| 	return &pbconversation.CreateGroupChatConversationsResp{}, nil | 	return &pbconversation.CreateGroupChatConversationsResp{}, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -480,6 +478,9 @@ func (c *conversationServer) SetConversationMinSeq(ctx context.Context, req *pbc | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *conversationServer) GetConversationIDs(ctx context.Context, req *pbconversation.GetConversationIDsReq) (*pbconversation.GetConversationIDsResp, error) { | func (c *conversationServer) GetConversationIDs(ctx context.Context, req *pbconversation.GetConversationIDsReq) (*pbconversation.GetConversationIDsResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	conversationIDs, err := c.conversationDatabase.GetConversationIDs(ctx, req.UserID) | 	conversationIDs, err := c.conversationDatabase.GetConversationIDs(ctx, req.UserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -488,6 +489,9 @@ func (c *conversationServer) GetConversationIDs(ctx context.Context, req *pbconv | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *conversationServer) GetUserConversationIDsHash(ctx context.Context, req *pbconversation.GetUserConversationIDsHashReq) (*pbconversation.GetUserConversationIDsHashResp, error) { | func (c *conversationServer) GetUserConversationIDsHash(ctx context.Context, req *pbconversation.GetUserConversationIDsHashReq) (*pbconversation.GetUserConversationIDsHashResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.OwnerUserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	hash, err := c.conversationDatabase.GetUserConversationIDsHash(ctx, req.OwnerUserID) | 	hash, err := c.conversationDatabase.GetUserConversationIDsHash(ctx, req.OwnerUserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -495,10 +499,7 @@ func (c *conversationServer) GetUserConversationIDsHash(ctx context.Context, req | |||||||
| 	return &pbconversation.GetUserConversationIDsHashResp{Hash: hash}, nil | 	return &pbconversation.GetUserConversationIDsHashResp{Hash: hash}, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *conversationServer) GetConversationsByConversationID( | func (c *conversationServer) GetConversationsByConversationID(ctx context.Context, req *pbconversation.GetConversationsByConversationIDReq) (*pbconversation.GetConversationsByConversationIDResp, error) { | ||||||
| 	ctx context.Context, |  | ||||||
| 	req *pbconversation.GetConversationsByConversationIDReq, |  | ||||||
| ) (*pbconversation.GetConversationsByConversationIDResp, error) { |  | ||||||
| 	conversations, err := c.conversationDatabase.GetConversationsByConversationID(ctx, req.ConversationIDs) | 	conversations, err := c.conversationDatabase.GetConversationsByConversationID(ctx, req.ConversationIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -552,10 +553,7 @@ func (c *conversationServer) conversationSort(conversations map[int64]string, re | |||||||
| 	resp.ConversationElems = append(resp.ConversationElems, cons...) | 	resp.ConversationElems = append(resp.ConversationElems, cons...) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *conversationServer) getConversationInfo( | func (c *conversationServer) getConversationInfo(ctx context.Context, chatLogs map[string]*sdkws.MsgData, userID string) (map[string]*pbconversation.ConversationElem, error) { | ||||||
| 	ctx context.Context, |  | ||||||
| 	chatLogs map[string]*sdkws.MsgData, |  | ||||||
| 	userID string) (map[string]*pbconversation.ConversationElem, error) { |  | ||||||
| 	var ( | 	var ( | ||||||
| 		sendIDs         []string | 		sendIDs         []string | ||||||
| 		groupIDs        []string | 		groupIDs        []string | ||||||
| @ -641,6 +639,11 @@ func (c *conversationServer) GetConversationNotReceiveMessageUserIDs(ctx context | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *conversationServer) UpdateConversation(ctx context.Context, req *pbconversation.UpdateConversationReq) (*pbconversation.UpdateConversationResp, error) { | func (c *conversationServer) UpdateConversation(ctx context.Context, req *pbconversation.UpdateConversationReq) (*pbconversation.UpdateConversationResp, error) { | ||||||
|  | 	for _, userID := range req.UserIDs { | ||||||
|  | 		if err := authverify.CheckAccess(ctx, userID); err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
| 	m := make(map[string]any) | 	m := make(map[string]any) | ||||||
| 	if req.RecvMsgOpt != nil { | 	if req.RecvMsgOpt != nil { | ||||||
| 		m["recv_msg_opt"] = req.RecvMsgOpt.Value | 		m["recv_msg_opt"] = req.RecvMsgOpt.Value | ||||||
| @ -687,6 +690,9 @@ func (c *conversationServer) UpdateConversation(ctx context.Context, req *pbconv | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *conversationServer) GetOwnerConversation(ctx context.Context, req *pbconversation.GetOwnerConversationReq) (*pbconversation.GetOwnerConversationResp, error) { | func (c *conversationServer) GetOwnerConversation(ctx context.Context, req *pbconversation.GetOwnerConversationReq) (*pbconversation.GetOwnerConversationResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	total, conversations, err := c.conversationDatabase.GetOwnerConversation(ctx, req.UserID, req.Pagination) | 	total, conversations, err := c.conversationDatabase.GetOwnerConversation(ctx, req.UserID, req.Pagination) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -711,7 +717,7 @@ func (c *conversationServer) GetConversationsNeedClearMsg(ctx context.Context, _ | |||||||
| 
 | 
 | ||||||
| 	maxPage := (num + batchNum - 1) / batchNum | 	maxPage := (num + batchNum - 1) / batchNum | ||||||
| 
 | 
 | ||||||
| 	temp := make([]*model.Conversation, 0, maxPage*batchNum) | 	temp := make([]*dbModel.Conversation, 0, maxPage*batchNum) | ||||||
| 
 | 
 | ||||||
| 	for pageNumber := 0; pageNumber < int(maxPage); pageNumber++ { | 	for pageNumber := 0; pageNumber < int(maxPage); pageNumber++ { | ||||||
| 		pagination := &sdkws.RequestPagination{ | 		pagination := &sdkws.RequestPagination{ | ||||||
| @ -748,6 +754,9 @@ func (c *conversationServer) GetConversationsNeedClearMsg(ctx context.Context, _ | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *conversationServer) GetNotNotifyConversationIDs(ctx context.Context, req *pbconversation.GetNotNotifyConversationIDsReq) (*pbconversation.GetNotNotifyConversationIDsResp, error) { | func (c *conversationServer) GetNotNotifyConversationIDs(ctx context.Context, req *pbconversation.GetNotNotifyConversationIDsReq) (*pbconversation.GetNotNotifyConversationIDsResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	conversationIDs, err := c.conversationDatabase.GetNotNotifyConversationIDs(ctx, req.UserID) | 	conversationIDs, err := c.conversationDatabase.GetNotNotifyConversationIDs(ctx, req.UserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -756,6 +765,9 @@ func (c *conversationServer) GetNotNotifyConversationIDs(ctx context.Context, re | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *conversationServer) GetPinnedConversationIDs(ctx context.Context, req *pbconversation.GetPinnedConversationIDsReq) (*pbconversation.GetPinnedConversationIDsResp, error) { | func (c *conversationServer) GetPinnedConversationIDs(ctx context.Context, req *pbconversation.GetPinnedConversationIDsReq) (*pbconversation.GetPinnedConversationIDsResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	conversationIDs, err := c.conversationDatabase.GetPinnedConversationIDs(ctx, req.UserID) | 	conversationIDs, err := c.conversationDatabase.GetPinnedConversationIDs(ctx, req.UserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
|  | |||||||
							
								
								
									
										85
									
								
								internal/rpc/conversation/db_map.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										85
									
								
								internal/rpc/conversation/db_map.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,85 @@ | |||||||
|  | package conversation | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"context" | ||||||
|  | 
 | ||||||
|  | 	dbModel "github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||||
|  | 	"github.com/openimsdk/protocol/conversation" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func UpdateConversationsMap(ctx context.Context, req *conversation.SetConversationsReq) (m map[string]any, conversation dbModel.Conversation, err error) { | ||||||
|  | 	m = make(map[string]any) | ||||||
|  | 
 | ||||||
|  | 	conversation.ConversationID = req.Conversation.ConversationID | ||||||
|  | 	conversation.ConversationType = req.Conversation.ConversationType | ||||||
|  | 	conversation.UserID = req.Conversation.UserID | ||||||
|  | 	conversation.GroupID = req.Conversation.GroupID | ||||||
|  | 
 | ||||||
|  | 	if req.Conversation.RecvMsgOpt != nil { | ||||||
|  | 		conversation.RecvMsgOpt = req.Conversation.RecvMsgOpt.Value | ||||||
|  | 		m["recv_msg_opt"] = req.Conversation.RecvMsgOpt.Value | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if req.Conversation.AttachedInfo != nil { | ||||||
|  | 		conversation.AttachedInfo = req.Conversation.AttachedInfo.Value | ||||||
|  | 		m["attached_info"] = req.Conversation.AttachedInfo.Value | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if req.Conversation.Ex != nil { | ||||||
|  | 		conversation.Ex = req.Conversation.Ex.Value | ||||||
|  | 		m["ex"] = req.Conversation.Ex.Value | ||||||
|  | 	} | ||||||
|  | 	if req.Conversation.IsPinned != nil { | ||||||
|  | 		conversation.IsPinned = req.Conversation.IsPinned.Value | ||||||
|  | 		m["is_pinned"] = req.Conversation.IsPinned.Value | ||||||
|  | 	} | ||||||
|  | 	if req.Conversation.GroupAtType != nil { | ||||||
|  | 		conversation.GroupAtType = req.Conversation.GroupAtType.Value | ||||||
|  | 		m["group_at_type"] = req.Conversation.GroupAtType.Value | ||||||
|  | 	} | ||||||
|  | 	if req.Conversation.MsgDestructTime != nil { | ||||||
|  | 		conversation.MsgDestructTime = req.Conversation.MsgDestructTime.Value | ||||||
|  | 		m["msg_destruct_time"] = req.Conversation.MsgDestructTime.Value | ||||||
|  | 	} | ||||||
|  | 	if req.Conversation.IsMsgDestruct != nil { | ||||||
|  | 		conversation.IsMsgDestruct = req.Conversation.IsMsgDestruct.Value | ||||||
|  | 		m["is_msg_destruct"] = req.Conversation.IsMsgDestruct.Value | ||||||
|  | 	} | ||||||
|  | 	if req.Conversation.BurnDuration != nil { | ||||||
|  | 		conversation.BurnDuration = req.Conversation.BurnDuration.Value | ||||||
|  | 		m["burn_duration"] = req.Conversation.BurnDuration.Value | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return m, conversation, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func UserUpdateCheckMap(ctx context.Context, userID string, req *conversation.ConversationReq, conversation *dbModel.Conversation) (unequal bool) { | ||||||
|  | 	unequal = false | ||||||
|  | 
 | ||||||
|  | 	if req.RecvMsgOpt != nil && conversation.RecvMsgOpt != req.RecvMsgOpt.Value { | ||||||
|  | 		unequal = true | ||||||
|  | 	} | ||||||
|  | 	if req.AttachedInfo != nil && conversation.AttachedInfo != req.AttachedInfo.Value { | ||||||
|  | 		unequal = true | ||||||
|  | 	} | ||||||
|  | 	if req.Ex != nil && conversation.Ex != req.Ex.Value { | ||||||
|  | 		unequal = true | ||||||
|  | 	} | ||||||
|  | 	if req.IsPinned != nil && conversation.IsPinned != req.IsPinned.Value { | ||||||
|  | 		unequal = true | ||||||
|  | 	} | ||||||
|  | 	if req.GroupAtType != nil && conversation.GroupAtType != req.GroupAtType.Value { | ||||||
|  | 		unequal = true | ||||||
|  | 	} | ||||||
|  | 	if req.MsgDestructTime != nil && conversation.MsgDestructTime != req.MsgDestructTime.Value { | ||||||
|  | 		unequal = true | ||||||
|  | 	} | ||||||
|  | 	if req.IsMsgDestruct != nil && conversation.IsMsgDestruct != req.IsMsgDestruct.Value { | ||||||
|  | 		unequal = true | ||||||
|  | 	} | ||||||
|  | 	if req.BurnDuration != nil && conversation.BurnDuration != req.BurnDuration.Value { | ||||||
|  | 		unequal = true | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return unequal | ||||||
|  | } | ||||||
| @ -4,12 +4,16 @@ import ( | |||||||
| 	"context" | 	"context" | ||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/internal/rpc/incrversion" | 	"github.com/openimsdk/open-im-server/v3/internal/rpc/incrversion" | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||||
| 	"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/util/hashutil" | 	"github.com/openimsdk/open-im-server/v3/pkg/util/hashutil" | ||||||
| 	"github.com/openimsdk/protocol/conversation" | 	"github.com/openimsdk/protocol/conversation" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func (c *conversationServer) GetFullOwnerConversationIDs(ctx context.Context, req *conversation.GetFullOwnerConversationIDsReq) (*conversation.GetFullOwnerConversationIDsResp, error) { | func (c *conversationServer) GetFullOwnerConversationIDs(ctx context.Context, req *conversation.GetFullOwnerConversationIDsReq) (*conversation.GetFullOwnerConversationIDsResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	vl, err := c.conversationDatabase.FindMaxConversationUserVersionCache(ctx, req.UserID) | 	vl, err := c.conversationDatabase.FindMaxConversationUserVersionCache(ctx, req.UserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -31,6 +35,9 @@ func (c *conversationServer) GetFullOwnerConversationIDs(ctx context.Context, re | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *conversationServer) GetIncrementalConversation(ctx context.Context, req *conversation.GetIncrementalConversationReq) (*conversation.GetIncrementalConversationResp, error) { | func (c *conversationServer) GetIncrementalConversation(ctx context.Context, req *conversation.GetIncrementalConversationReq) (*conversation.GetIncrementalConversationResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	opt := incrversion.Option[*conversation.Conversation, conversation.GetIncrementalConversationResp]{ | 	opt := incrversion.Option[*conversation.Conversation, conversation.GetIncrementalConversationResp]{ | ||||||
| 		Ctx:             ctx, | 		Ctx:             ctx, | ||||||
| 		VersionKey:      req.UserID, | 		VersionKey:      req.UserID, | ||||||
|  | |||||||
| @ -33,6 +33,9 @@ func (g *groupServer) GetGroupInfoCache(ctx context.Context, req *pbgroup.GetGro | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *groupServer) GetGroupMemberCache(ctx context.Context, req *pbgroup.GetGroupMemberCacheReq) (*pbgroup.GetGroupMemberCacheResp, error) { | func (g *groupServer) GetGroupMemberCache(ctx context.Context, req *pbgroup.GetGroupMemberCacheReq) (*pbgroup.GetGroupMemberCacheResp, error) { | ||||||
|  | 	if err := g.checkAdminOrInGroup(ctx, req.GroupID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	members, err := g.db.TakeGroupMember(ctx, req.GroupID, req.GroupMemberID) | 	members, err := g.db.TakeGroupMember(ctx, req.GroupID, req.GroupMemberID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
|  | |||||||
| @ -16,6 +16,7 @@ package group | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 	"strings" | ||||||
| 	"time" | 	"time" | ||||||
| 
 | 
 | ||||||
| 	pbgroup "github.com/openimsdk/protocol/group" | 	pbgroup "github.com/openimsdk/protocol/group" | ||||||
| @ -55,41 +56,52 @@ func UpdateGroupInfoMap(ctx context.Context, group *sdkws.GroupInfoForSet) map[s | |||||||
| 	return m | 	return m | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func UpdateGroupInfoExMap(ctx context.Context, group *pbgroup.SetGroupInfoExReq) (map[string]any, error) { | func UpdateGroupInfoExMap(ctx context.Context, group *pbgroup.SetGroupInfoExReq) (m map[string]any, normalFlag, groupNameFlag, notificationFlag bool, err error) { | ||||||
| 	m := make(map[string]any) | 	m = make(map[string]any) | ||||||
| 
 | 
 | ||||||
| 	if group.GroupName != nil { | 	if group.GroupName != nil { | ||||||
| 		if group.GroupName.Value != "" { | 		if strings.TrimSpace(group.GroupName.Value) != "" { | ||||||
| 			m["group_name"] = group.GroupName.Value | 			m["group_name"] = group.GroupName.Value | ||||||
|  | 			groupNameFlag = true | ||||||
| 		} else { | 		} else { | ||||||
| 			return nil, errs.ErrArgs.WrapMsg("group name is empty") | 			return nil, normalFlag, notificationFlag, groupNameFlag, errs.ErrArgs.WrapMsg("group name is empty") | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	if group.Notification != nil { | 	if group.Notification != nil { | ||||||
|  | 		notificationFlag = true | ||||||
|  | 		group.Notification.Value = strings.TrimSpace(group.Notification.Value) // if Notification only contains spaces, set it to empty string | ||||||
|  | 
 | ||||||
| 		m["notification"] = group.Notification.Value | 		m["notification"] = group.Notification.Value | ||||||
| 		m["notification_update_time"] = time.Now() |  | ||||||
| 		m["notification_user_id"] = mcontext.GetOpUserID(ctx) | 		m["notification_user_id"] = mcontext.GetOpUserID(ctx) | ||||||
|  | 		m["notification_update_time"] = time.Now() | ||||||
| 	} | 	} | ||||||
| 	if group.Introduction != nil { | 	if group.Introduction != nil { | ||||||
| 		m["introduction"] = group.Introduction.Value | 		m["introduction"] = group.Introduction.Value | ||||||
|  | 		normalFlag = true | ||||||
| 	} | 	} | ||||||
| 	if group.FaceURL != nil { | 	if group.FaceURL != nil { | ||||||
| 		m["face_url"] = group.FaceURL.Value | 		m["face_url"] = group.FaceURL.Value | ||||||
|  | 		normalFlag = true | ||||||
| 	} | 	} | ||||||
| 	if group.NeedVerification != nil { | 	if group.NeedVerification != nil { | ||||||
| 		m["need_verification"] = group.NeedVerification.Value | 		m["need_verification"] = group.NeedVerification.Value | ||||||
|  | 		normalFlag = true | ||||||
| 	} | 	} | ||||||
| 	if group.LookMemberInfo != nil { | 	if group.LookMemberInfo != nil { | ||||||
| 		m["look_member_info"] = group.LookMemberInfo.Value | 		m["look_member_info"] = group.LookMemberInfo.Value | ||||||
|  | 		normalFlag = true | ||||||
| 	} | 	} | ||||||
| 	if group.ApplyMemberFriend != nil { | 	if group.ApplyMemberFriend != nil { | ||||||
| 		m["apply_member_friend"] = group.ApplyMemberFriend.Value | 		m["apply_member_friend"] = group.ApplyMemberFriend.Value | ||||||
|  | 		normalFlag = true | ||||||
| 	} | 	} | ||||||
| 	if group.Ex != nil { | 	if group.Ex != nil { | ||||||
| 		m["ex"] = group.Ex.Value | 		m["ex"] = group.Ex.Value | ||||||
|  | 		normalFlag = true | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	return m, nil | 	return m, normalFlag, groupNameFlag, notificationFlag, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func UpdateGroupStatusMap(status int) map[string]any { | func UpdateGroupStatusMap(status int) map[string]any { | ||||||
|  | |||||||
| @ -25,7 +25,6 @@ import ( | |||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/dbbuild" | 	"github.com/openimsdk/open-im-server/v3/pkg/dbbuild" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | ||||||
| 
 |  | ||||||
| 	"google.golang.org/grpc" | 	"google.golang.org/grpc" | ||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||||
| @ -64,6 +63,7 @@ type groupServer struct { | |||||||
| 	userClient         *rpcli.UserClient | 	userClient         *rpcli.UserClient | ||||||
| 	msgClient          *rpcli.MsgClient | 	msgClient          *rpcli.MsgClient | ||||||
| 	conversationClient *rpcli.ConversationClient | 	conversationClient *rpcli.ConversationClient | ||||||
|  | 	adminUserIDs       []string | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| type Config struct { | type Config struct { | ||||||
| @ -77,7 +77,7 @@ type Config struct { | |||||||
| 	Discovery          config.Discovery | 	Discovery          config.Discovery | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func Start(ctx context.Context, config *Config, client discovery.Conn, server grpc.ServiceRegistrar) error { | func Start(ctx context.Context, config *Config, client discovery.SvcDiscoveryRegistry, server grpc.ServiceRegistrar) error { | ||||||
| 	dbb := dbbuild.NewBuilder(&config.MongodbConfig, &config.RedisConfig) | 	dbb := dbbuild.NewBuilder(&config.MongodbConfig, &config.RedisConfig) | ||||||
| 	mgocli, err := dbb.Mongo(ctx) | 	mgocli, err := dbb.Mongo(ctx) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -117,6 +117,7 @@ func Start(ctx context.Context, config *Config, client discovery.Conn, server gr | |||||||
| 		userClient:         rpcli.NewUserClient(userConn), | 		userClient:         rpcli.NewUserClient(userConn), | ||||||
| 		msgClient:          rpcli.NewMsgClient(msgConn), | 		msgClient:          rpcli.NewMsgClient(msgConn), | ||||||
| 		conversationClient: rpcli.NewConversationClient(conversationConn), | 		conversationClient: rpcli.NewConversationClient(conversationConn), | ||||||
|  | 		adminUserIDs:       config.Share.IMAdminUser.UserIDs, | ||||||
| 	} | 	} | ||||||
| 	gs.db = controller.NewGroupDatabase(rdb, &config.LocalCacheConfig, groupDB, groupMemberDB, groupRequestDB, mgocli.GetTx(), grouphash.NewGroupHashFromGroupServer(&gs)) | 	gs.db = controller.NewGroupDatabase(rdb, &config.LocalCacheConfig, groupDB, groupMemberDB, groupRequestDB, mgocli.GetTx(), grouphash.NewGroupHashFromGroupServer(&gs)) | ||||||
| 	gs.notification = NewNotificationSender(gs.db, config, gs.userClient, gs.msgClient, gs.conversationClient) | 	gs.notification = NewNotificationSender(gs.db, config, gs.userClient, gs.msgClient, gs.conversationClient) | ||||||
| @ -152,11 +153,15 @@ func (g *groupServer) NotificationUserInfoUpdate(ctx context.Context, req *pbgro | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *groupServer) CheckGroupAdmin(ctx context.Context, groupID string) error { | func (g *groupServer) CheckGroupAdmin(ctx context.Context, groupID string) error { | ||||||
| 	if !authverify.IsAppManagerUid(ctx, g.config.Share.IMAdminUserID) { | 	if !authverify.IsAdmin(ctx) { | ||||||
| 		groupMember, err := g.db.TakeGroupMember(ctx, groupID, mcontext.GetOpUserID(ctx)) | 		members, err := g.db.FindGroupMembers(ctx, groupID, []string{mcontext.GetOpUserID(ctx)}) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
|  | 		if len(members) == 0 { | ||||||
|  | 			return errs.ErrNoPermission.WrapMsg("op user not in group") | ||||||
|  | 		} | ||||||
|  | 		groupMember := members[0] | ||||||
| 		if !(groupMember.RoleLevel == constant.GroupOwner || groupMember.RoleLevel == constant.GroupAdmin) { | 		if !(groupMember.RoleLevel == constant.GroupOwner || groupMember.RoleLevel == constant.GroupAdmin) { | ||||||
| 			return errs.ErrNoPermission.WrapMsg("no group owner or admin") | 			return errs.ErrNoPermission.WrapMsg("no group owner or admin") | ||||||
| 		} | 		} | ||||||
| @ -204,7 +209,7 @@ func (g *groupServer) CreateGroup(ctx context.Context, req *pbgroup.CreateGroupR | |||||||
| 	if req.OwnerUserID == "" { | 	if req.OwnerUserID == "" { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("no group owner") | 		return nil, errs.ErrArgs.WrapMsg("no group owner") | ||||||
| 	} | 	} | ||||||
| 	if err := authverify.CheckAccessV3(ctx, req.OwnerUserID, g.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAccess(ctx, req.OwnerUserID); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	userIDs := append(append(req.MemberUserIDs, req.AdminUserIDs...), req.OwnerUserID) | 	userIDs := append(append(req.MemberUserIDs, req.AdminUserIDs...), req.OwnerUserID) | ||||||
| @ -288,10 +293,11 @@ func (g *groupServer) CreateGroup(ctx context.Context, req *pbgroup.CreateGroupR | |||||||
| 	g.notification.GroupCreatedNotification(ctx, tips, req.SendMessage) | 	g.notification.GroupCreatedNotification(ctx, tips, req.SendMessage) | ||||||
| 
 | 
 | ||||||
| 	if req.GroupInfo.Notification != "" { | 	if req.GroupInfo.Notification != "" { | ||||||
|  | 		notificationFlag := true | ||||||
| 		g.notification.GroupInfoSetAnnouncementNotification(ctx, &sdkws.GroupInfoSetAnnouncementTips{ | 		g.notification.GroupInfoSetAnnouncementNotification(ctx, &sdkws.GroupInfoSetAnnouncementTips{ | ||||||
| 			Group:  tips.Group, | 			Group:  tips.Group, | ||||||
| 			OpUser: tips.OpUser, | 			OpUser: tips.OpUser, | ||||||
| 		}) | 		}, ¬ificationFlag) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	reqCallBackAfter := &pbgroup.CreateGroupReq{ | 	reqCallBackAfter := &pbgroup.CreateGroupReq{ | ||||||
| @ -307,7 +313,7 @@ func (g *groupServer) CreateGroup(ctx context.Context, req *pbgroup.CreateGroupR | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *groupServer) GetJoinedGroupList(ctx context.Context, req *pbgroup.GetJoinedGroupListReq) (*pbgroup.GetJoinedGroupListResp, error) { | func (g *groupServer) GetJoinedGroupList(ctx context.Context, req *pbgroup.GetJoinedGroupListReq) (*pbgroup.GetJoinedGroupListResp, error) { | ||||||
| 	if err := authverify.CheckAccessV3(ctx, req.FromUserID, g.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAccess(ctx, req.FromUserID); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	total, members, err := g.db.PageGetJoinGroup(ctx, req.FromUserID, req.Pagination) | 	total, members, err := g.db.PageGetJoinGroup(ctx, req.FromUserID, req.Pagination) | ||||||
| @ -368,6 +374,10 @@ func (g *groupServer) InviteUserToGroup(ctx context.Context, req *pbgroup.Invite | |||||||
| 		return nil, servererrs.ErrDismissedAlready.WrapMsg("group dismissed checking group status found it dismissed") | 		return nil, servererrs.ErrDismissedAlready.WrapMsg("group dismissed checking group status found it dismissed") | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	if err := g.checkAdminOrInGroup(ctx, req.GroupID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	userMap, err := g.userClient.GetUsersInfoMap(ctx, req.InvitedUserIDs) | 	userMap, err := g.userClient.GetUsersInfoMap(ctx, req.InvitedUserIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -378,9 +388,9 @@ func (g *groupServer) InviteUserToGroup(ctx context.Context, req *pbgroup.Invite | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	var groupMember *model.GroupMember | 	var groupMember *model.GroupMember | ||||||
| 	var opUserID string | 	opUserID := mcontext.GetOpUserID(ctx) | ||||||
| 	if !authverify.IsAppManagerUid(ctx, g.config.Share.IMAdminUserID) { | 
 | ||||||
| 		opUserID = mcontext.GetOpUserID(ctx) | 	if !authverify.IsAdmin(ctx) { | ||||||
| 		var err error | 		var err error | ||||||
| 		groupMember, err = g.db.TakeGroupMember(ctx, req.GroupID, opUserID) | 		groupMember, err = g.db.TakeGroupMember(ctx, req.GroupID, opUserID) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| @ -396,7 +406,7 @@ func (g *groupServer) InviteUserToGroup(ctx context.Context, req *pbgroup.Invite | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if group.NeedVerification == constant.AllNeedVerification { | 	if group.NeedVerification == constant.AllNeedVerification { | ||||||
| 		if !authverify.IsAppManagerUid(ctx, g.config.Share.IMAdminUserID) { | 		if !authverify.IsAdmin(ctx) { | ||||||
| 			if !(groupMember.RoleLevel == constant.GroupOwner || groupMember.RoleLevel == constant.GroupAdmin) { | 			if !(groupMember.RoleLevel == constant.GroupOwner || groupMember.RoleLevel == constant.GroupAdmin) { | ||||||
| 				var requests []*model.GroupRequest | 				var requests []*model.GroupRequest | ||||||
| 				for _, userID := range req.InvitedUserIDs { | 				for _, userID := range req.InvitedUserIDs { | ||||||
| @ -418,12 +428,13 @@ func (g *groupServer) InviteUserToGroup(ctx context.Context, req *pbgroup.Invite | |||||||
| 						ReqMessage:    request.ReqMsg, | 						ReqMessage:    request.ReqMsg, | ||||||
| 						JoinSource:    request.JoinSource, | 						JoinSource:    request.JoinSource, | ||||||
| 						InviterUserID: request.InviterUserID, | 						InviterUserID: request.InviterUserID, | ||||||
| 					}) | 					}, request) | ||||||
| 				} | 				} | ||||||
| 				return &pbgroup.InviteUserToGroupResp{}, nil | 				return &pbgroup.InviteUserToGroupResp{}, nil | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	var groupMembers []*model.GroupMember | 	var groupMembers []*model.GroupMember | ||||||
| 	for _, userID := range req.InvitedUserIDs { | 	for _, userID := range req.InvitedUserIDs { | ||||||
| 		member := &model.GroupMember{ | 		member := &model.GroupMember{ | ||||||
| @ -444,12 +455,22 @@ func (g *groupServer) InviteUserToGroup(ctx context.Context, req *pbgroup.Invite | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if err := g.db.CreateGroup(ctx, nil, groupMembers); err != nil { | 	const singleQuantity = 50 | ||||||
| 		return nil, err | 	for start := 0; start < len(groupMembers); start += singleQuantity { | ||||||
| 	} | 		end := min(start+singleQuantity, len(groupMembers)) | ||||||
|  | 		currentMembers := groupMembers[start:end] | ||||||
| 
 | 
 | ||||||
| 	if err = g.notification.GroupApplicationAgreeMemberEnterNotification(ctx, req.GroupID, req.SendMessage, opUserID, req.InvitedUserIDs...); err != nil { | 		if err := g.db.CreateGroup(ctx, nil, currentMembers); err != nil { | ||||||
| 		return nil, err | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		userIDs := datautil.Slice(currentMembers, func(e *model.GroupMember) string { | ||||||
|  | 			return e.UserID | ||||||
|  | 		}) | ||||||
|  | 
 | ||||||
|  | 		if len(userIDs) != 0 { | ||||||
|  | 			g.notification.GroupApplicationAgreeMemberEnterNotification(ctx, req.GroupID, req.SendMessage, opUserID, userIDs...) | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 	return &pbgroup.InviteUserToGroupResp{}, nil | 	return &pbgroup.InviteUserToGroupResp{}, nil | ||||||
| } | } | ||||||
| @ -459,6 +480,19 @@ func (g *groupServer) GetGroupAllMember(ctx context.Context, req *pbgroup.GetGro | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 	if !authverify.IsAdmin(ctx) { | ||||||
|  | 		var inGroup bool | ||||||
|  | 		opUserID := mcontext.GetOpUserID(ctx) | ||||||
|  | 		for _, member := range members { | ||||||
|  | 			if member.UserID == opUserID { | ||||||
|  | 				inGroup = true | ||||||
|  | 				break | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		if !inGroup { | ||||||
|  | 			return nil, errs.ErrNoPermission.WrapMsg("opuser not in group") | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
| 	if err := g.PopulateGroupMember(ctx, members...); err != nil { | 	if err := g.PopulateGroupMember(ctx, members...); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -469,7 +503,25 @@ func (g *groupServer) GetGroupAllMember(ctx context.Context, req *pbgroup.GetGro | |||||||
| 	return &resp, nil | 	return &resp, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func (g *groupServer) checkAdminOrInGroup(ctx context.Context, groupID string) error { | ||||||
|  | 	if authverify.IsAdmin(ctx) { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	opUserID := mcontext.GetOpUserID(ctx) | ||||||
|  | 	members, err := g.db.FindGroupMembers(ctx, groupID, []string{opUserID}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	if len(members) == 0 { | ||||||
|  | 		return errs.ErrNoPermission.WrapMsg("op user not in group") | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func (g *groupServer) GetGroupMemberList(ctx context.Context, req *pbgroup.GetGroupMemberListReq) (*pbgroup.GetGroupMemberListResp, error) { | func (g *groupServer) GetGroupMemberList(ctx context.Context, req *pbgroup.GetGroupMemberListReq) (*pbgroup.GetGroupMemberListResp, error) { | ||||||
|  | 	if err := g.checkAdminOrInGroup(ctx, req.GroupID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	var ( | 	var ( | ||||||
| 		total   int64 | 		total   int64 | ||||||
| 		members []*model.GroupMember | 		members []*model.GroupMember | ||||||
| @ -478,7 +530,7 @@ func (g *groupServer) GetGroupMemberList(ctx context.Context, req *pbgroup.GetGr | |||||||
| 	if req.Keyword == "" { | 	if req.Keyword == "" { | ||||||
| 		total, members, err = g.db.PageGetGroupMember(ctx, req.GroupID, req.Pagination) | 		total, members, err = g.db.PageGetGroupMember(ctx, req.GroupID, req.Pagination) | ||||||
| 	} else { | 	} else { | ||||||
| 		members, err = g.db.FindGroupMemberAll(ctx, req.GroupID) | 		total, members, err = g.db.SearchGroupMember(ctx, req.GroupID, req.Keyword, req.Pagination) | ||||||
| 	} | 	} | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -486,27 +538,6 @@ func (g *groupServer) GetGroupMemberList(ctx context.Context, req *pbgroup.GetGr | |||||||
| 	if err := g.PopulateGroupMember(ctx, members...); err != nil { | 	if err := g.PopulateGroupMember(ctx, members...); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	if req.Keyword != "" { |  | ||||||
| 		groupMembers := make([]*model.GroupMember, 0) |  | ||||||
| 		for _, member := range members { |  | ||||||
| 			if member.UserID == req.Keyword { |  | ||||||
| 				groupMembers = append(groupMembers, member) |  | ||||||
| 				total++ |  | ||||||
| 				continue |  | ||||||
| 			} |  | ||||||
| 			if member.Nickname == req.Keyword { |  | ||||||
| 				groupMembers = append(groupMembers, member) |  | ||||||
| 				total++ |  | ||||||
| 				continue |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		members := datautil.Paginate(groupMembers, int(req.Pagination.GetPageNumber()), int(req.Pagination.GetShowNumber())) |  | ||||||
| 		return &pbgroup.GetGroupMemberListResp{ |  | ||||||
| 			Total:   uint32(total), |  | ||||||
| 			Members: datautil.Batch(convert.Db2PbGroupMember, members), |  | ||||||
| 		}, nil |  | ||||||
| 	} |  | ||||||
| 	return &pbgroup.GetGroupMemberListResp{ | 	return &pbgroup.GetGroupMemberListResp{ | ||||||
| 		Total:   uint32(total), | 		Total:   uint32(total), | ||||||
| 		Members: datautil.Batch(convert.Db2PbGroupMember, members), | 		Members: datautil.Batch(convert.Db2PbGroupMember, members), | ||||||
| @ -547,7 +578,7 @@ func (g *groupServer) KickGroupMember(ctx context.Context, req *pbgroup.KickGrou | |||||||
| 	for i, member := range members { | 	for i, member := range members { | ||||||
| 		memberMap[member.UserID] = members[i] | 		memberMap[member.UserID] = members[i] | ||||||
| 	} | 	} | ||||||
| 	isAppManagerUid := authverify.IsAppManagerUid(ctx, g.config.Share.IMAdminUserID) | 	isAppManagerUid := authverify.IsAdmin(ctx) | ||||||
| 	opMember := memberMap[opUserID] | 	opMember := memberMap[opUserID] | ||||||
| 	for _, userID := range req.KickedUserIDs { | 	for _, userID := range req.KickedUserIDs { | ||||||
| 		member, ok := memberMap[userID] | 		member, ok := memberMap[userID] | ||||||
| @ -630,6 +661,9 @@ func (g *groupServer) GetGroupMembersInfo(ctx context.Context, req *pbgroup.GetG | |||||||
| 	if req.GroupID == "" { | 	if req.GroupID == "" { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("groupID empty") | 		return nil, errs.ErrArgs.WrapMsg("groupID empty") | ||||||
| 	} | 	} | ||||||
|  | 	if err := g.checkAdminOrInGroup(ctx, req.GroupID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	members, err := g.getGroupMembersInfo(ctx, req.GroupID, req.UserIDs) | 	members, err := g.getGroupMembersInfo(ctx, req.GroupID, req.UserIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -657,15 +691,37 @@ func (g *groupServer) getGroupMembersInfo(ctx context.Context, groupID string, u | |||||||
| 
 | 
 | ||||||
| // GetGroupApplicationList handles functions that get a list of group requests. | // GetGroupApplicationList handles functions that get a list of group requests. | ||||||
| func (g *groupServer) GetGroupApplicationList(ctx context.Context, req *pbgroup.GetGroupApplicationListReq) (*pbgroup.GetGroupApplicationListResp, error) { | func (g *groupServer) GetGroupApplicationList(ctx context.Context, req *pbgroup.GetGroupApplicationListReq) (*pbgroup.GetGroupApplicationListResp, error) { | ||||||
| 	groupIDs, err := g.db.FindUserManagedGroupID(ctx, req.FromUserID) | 	if err := authverify.CheckAccess(ctx, req.FromUserID); err != nil { | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 	var ( | ||||||
|  | 		groupIDs []string | ||||||
|  | 		err      error | ||||||
|  | 	) | ||||||
|  | 	if len(req.GroupIDs) == 0 { | ||||||
|  | 		groupIDs, err = g.db.FindUserManagedGroupID(ctx, req.FromUserID) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		req.GroupIDs = datautil.Distinct(req.GroupIDs) | ||||||
|  | 		if !authverify.IsAdmin(ctx) { | ||||||
|  | 			for _, groupID := range req.GroupIDs { | ||||||
|  | 				if err := g.CheckGroupAdmin(ctx, groupID); err != nil { | ||||||
|  | 					return nil, err | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		groupIDs = req.GroupIDs | ||||||
|  | 	} | ||||||
| 	resp := &pbgroup.GetGroupApplicationListResp{} | 	resp := &pbgroup.GetGroupApplicationListResp{} | ||||||
| 	if len(groupIDs) == 0 { | 	if len(groupIDs) == 0 { | ||||||
| 		return resp, nil | 		return resp, nil | ||||||
| 	} | 	} | ||||||
| 	total, groupRequests, err := g.db.PageGroupRequest(ctx, groupIDs, req.Pagination) | 	handleResults := datautil.Slice(req.HandleResults, func(e int32) int { | ||||||
|  | 		return int(e) | ||||||
|  | 	}) | ||||||
|  | 	total, groupRequests, err := g.db.PageGroupRequest(ctx, groupIDs, handleResults, req.Pagination) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -730,6 +786,23 @@ func (g *groupServer) GetGroupsInfo(ctx context.Context, req *pbgroup.GetGroupsI | |||||||
| 	}, nil | 	}, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func (g *groupServer) GetGroupApplicationUnhandledCount(ctx context.Context, req *pbgroup.GetGroupApplicationUnhandledCountReq) (*pbgroup.GetGroupApplicationUnhandledCountResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	groupIDs, err := g.db.FindUserManagedGroupID(ctx, req.UserID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	count, err := g.db.GetGroupApplicationUnhandledCount(ctx, groupIDs, req.Time) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	return &pbgroup.GetGroupApplicationUnhandledCountResp{ | ||||||
|  | 		Count: count, | ||||||
|  | 	}, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func (g *groupServer) getGroupsInfo(ctx context.Context, groupIDs []string) ([]*sdkws.GroupInfo, error) { | func (g *groupServer) getGroupsInfo(ctx context.Context, groupIDs []string) ([]*sdkws.GroupInfo, error) { | ||||||
| 	if len(groupIDs) == 0 { | 	if len(groupIDs) == 0 { | ||||||
| 		return nil, nil | 		return nil, nil | ||||||
| @ -765,7 +838,7 @@ func (g *groupServer) GroupApplicationResponse(ctx context.Context, req *pbgroup | |||||||
| 	if !datautil.Contain(req.HandleResult, constant.GroupResponseAgree, constant.GroupResponseRefuse) { | 	if !datautil.Contain(req.HandleResult, constant.GroupResponseAgree, constant.GroupResponseRefuse) { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("HandleResult unknown") | 		return nil, errs.ErrArgs.WrapMsg("HandleResult unknown") | ||||||
| 	} | 	} | ||||||
| 	if !authverify.IsAppManagerUid(ctx, g.config.Share.IMAdminUserID) { | 	if !authverify.IsAdmin(ctx) { | ||||||
| 		groupMember, err := g.db.TakeGroupMember(ctx, req.GroupID, mcontext.GetOpUserID(ctx)) | 		groupMember, err := g.db.TakeGroupMember(ctx, req.GroupID, mcontext.GetOpUserID(ctx)) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return nil, err | 			return nil, err | ||||||
| @ -823,8 +896,14 @@ 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.GroupApplicationAgreeMemberEnterNotification(ctx, req.GroupID, nil, groupRequest.InviterUserID, req.FromUserID); err != nil { | 			if groupRequest.InviterUserID == "" { | ||||||
| 				return nil, err | 				if err = g.notification.MemberEnterNotification(ctx, req.GroupID, req.FromUserID); err != nil { | ||||||
|  | 					return nil, err | ||||||
|  | 				} | ||||||
|  | 			} else { | ||||||
|  | 				if err = g.notification.GroupApplicationAgreeMemberEnterNotification(ctx, req.GroupID, nil, groupRequest.InviterUserID, req.FromUserID); err != nil { | ||||||
|  | 					return nil, err | ||||||
|  | 				} | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 	case constant.GroupResponseRefuse: | 	case constant.GroupResponseRefuse: | ||||||
| @ -905,7 +984,7 @@ func (g *groupServer) JoinGroup(ctx context.Context, req *pbgroup.JoinGroupReq) | |||||||
| 	if err = g.db.CreateGroupRequest(ctx, []*model.GroupRequest{&groupRequest}); err != nil { | 	if err = g.db.CreateGroupRequest(ctx, []*model.GroupRequest{&groupRequest}); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	g.notification.JoinGroupApplicationNotification(ctx, req) | 	g.notification.JoinGroupApplicationNotification(ctx, req, &groupRequest) | ||||||
| 	return &pbgroup.JoinGroupResp{}, nil | 	return &pbgroup.JoinGroupResp{}, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -913,7 +992,7 @@ func (g *groupServer) QuitGroup(ctx context.Context, req *pbgroup.QuitGroupReq) | |||||||
| 	if req.UserID == "" { | 	if req.UserID == "" { | ||||||
| 		req.UserID = mcontext.GetOpUserID(ctx) | 		req.UserID = mcontext.GetOpUserID(ctx) | ||||||
| 	} else { | 	} else { | ||||||
| 		if err := authverify.CheckAccessV3(ctx, req.UserID, g.config.Share.IMAdminUserID); err != nil { | 		if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
| 			return nil, err | 			return nil, err | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @ -951,7 +1030,7 @@ func (g *groupServer) deleteMemberAndSetConversationSeq(ctx context.Context, gro | |||||||
| 
 | 
 | ||||||
| func (g *groupServer) SetGroupInfo(ctx context.Context, req *pbgroup.SetGroupInfoReq) (*pbgroup.SetGroupInfoResp, error) { | func (g *groupServer) SetGroupInfo(ctx context.Context, req *pbgroup.SetGroupInfoReq) (*pbgroup.SetGroupInfoResp, error) { | ||||||
| 	var opMember *model.GroupMember | 	var opMember *model.GroupMember | ||||||
| 	if !authverify.IsAppManagerUid(ctx, g.config.Share.IMAdminUserID) { | 	if !authverify.IsAdmin(ctx) { | ||||||
| 		var err error | 		var err error | ||||||
| 		opMember, err = g.db.TakeGroupMember(ctx, req.GroupInfoForSet.GroupID, mcontext.GetOpUserID(ctx)) | 		opMember, err = g.db.TakeGroupMember(ctx, req.GroupInfoForSet.GroupID, mcontext.GetOpUserID(ctx)) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| @ -1026,7 +1105,8 @@ func (g *groupServer) SetGroupInfo(ctx context.Context, req *pbgroup.SetGroupInf | |||||||
| 				log.ZWarn(ctx, "SetConversations", err, "UserIDs", resp.UserIDs, "conversation", conversation) | 				log.ZWarn(ctx, "SetConversations", err, "UserIDs", resp.UserIDs, "conversation", conversation) | ||||||
| 			} | 			} | ||||||
| 		}() | 		}() | ||||||
| 		g.notification.GroupInfoSetAnnouncementNotification(ctx, &sdkws.GroupInfoSetAnnouncementTips{Group: tips.Group, OpUser: tips.OpUser}) | 		notficationFlag := true | ||||||
|  | 		g.notification.GroupInfoSetAnnouncementNotification(ctx, &sdkws.GroupInfoSetAnnouncementTips{Group: tips.Group, OpUser: tips.OpUser}, ¬ficationFlag) | ||||||
| 	} | 	} | ||||||
| 	if req.GroupInfoForSet.GroupName != "" { | 	if req.GroupInfoForSet.GroupName != "" { | ||||||
| 		num-- | 		num-- | ||||||
| @ -1044,7 +1124,7 @@ func (g *groupServer) SetGroupInfo(ctx context.Context, req *pbgroup.SetGroupInf | |||||||
| 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.IsAdmin(ctx) { | ||||||
| 		var err error | 		var err error | ||||||
| 
 | 
 | ||||||
| 		opMember, err = g.db.TakeGroupMember(ctx, req.GroupID, mcontext.GetOpUserID(ctx)) | 		opMember, err = g.db.TakeGroupMember(ctx, req.GroupID, mcontext.GetOpUserID(ctx)) | ||||||
| @ -1087,7 +1167,7 @@ func (g *groupServer) SetGroupInfoEx(ctx context.Context, req *pbgroup.SetGroupI | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	updatedData, err := UpdateGroupInfoExMap(ctx, req) | 	updatedData, normalFlag, groupNameFlag, notificationFlag, err := UpdateGroupInfoExMap(ctx, req) | ||||||
| 	if len(updatedData) == 0 { | 	if len(updatedData) == 0 { | ||||||
| 		return &pbgroup.SetGroupInfoExResp{}, nil | 		return &pbgroup.SetGroupInfoExResp{}, nil | ||||||
| 	} | 	} | ||||||
| @ -1115,41 +1195,38 @@ func (g *groupServer) SetGroupInfoEx(ctx context.Context, req *pbgroup.SetGroupI | |||||||
| 		tips.OpUser = g.groupMemberDB2PB(opMember, 0) | 		tips.OpUser = g.groupMemberDB2PB(opMember, 0) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	num := len(updatedData) | 	if notificationFlag { | ||||||
| 
 |  | ||||||
| 	if req.Notification != nil { |  | ||||||
| 		num -= 3 |  | ||||||
| 
 |  | ||||||
| 		if req.Notification.Value != "" { | 		if req.Notification.Value != "" { | ||||||
| 			func() { | 			conversation := &pbconv.ConversationReq{ | ||||||
| 				conversation := &pbconv.ConversationReq{ | 				ConversationID:   msgprocessor.GetConversationIDBySessionType(constant.ReadGroupChatType, req.GroupID), | ||||||
| 					ConversationID:   msgprocessor.GetConversationIDBySessionType(constant.ReadGroupChatType, req.GroupID), | 				ConversationType: constant.ReadGroupChatType, | ||||||
| 					ConversationType: constant.ReadGroupChatType, | 				GroupID:          req.GroupID, | ||||||
| 					GroupID:          req.GroupID, | 			} | ||||||
| 				} |  | ||||||
| 
 | 
 | ||||||
| 				resp, err := g.GetGroupMemberUserIDs(ctx, &pbgroup.GetGroupMemberUserIDsReq{GroupID: req.GroupID}) | 			resp, err := g.GetGroupMemberUserIDs(ctx, &pbgroup.GetGroupMemberUserIDsReq{GroupID: req.GroupID}) | ||||||
| 				if err != nil { | 			if err != nil { | ||||||
| 					log.ZWarn(ctx, "GetGroupMemberIDs is failed.", err) | 				log.ZWarn(ctx, "GetGroupMemberIDs is failed.", err) | ||||||
| 					return | 				return nil, err | ||||||
| 				} | 			} | ||||||
| 
 | 
 | ||||||
| 				conversation.GroupAtType = &wrapperspb.Int32Value{Value: constant.GroupNotification} | 			conversation.GroupAtType = &wrapperspb.Int32Value{Value: constant.GroupNotification} | ||||||
| 				if err := g.conversationClient.SetConversations(ctx, resp.UserIDs, conversation); err != nil { | 			if err := g.conversationClient.SetConversations(ctx, resp.UserIDs, conversation); err != nil { | ||||||
| 					log.ZWarn(ctx, "SetConversations", err, "UserIDs", resp.UserIDs, "conversation", 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}, ¬ificationFlag) | ||||||
|  | 		} else { | ||||||
|  | 			notificationFlag = false | ||||||
|  | 			g.notification.GroupInfoSetAnnouncementNotification(ctx, &sdkws.GroupInfoSetAnnouncementTips{Group: tips.Group, OpUser: tips.OpUser}, ¬ificationFlag) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if req.GroupName != nil { | 	if groupNameFlag { | ||||||
| 		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 updatedData > 0, send the normal notification | ||||||
|  | 	if normalFlag { | ||||||
| 		g.notification.GroupInfoSetNotification(ctx, tips) | 		g.notification.GroupInfoSetNotification(ctx, tips) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -1196,7 +1273,7 @@ func (g *groupServer) TransferGroupOwner(ctx context.Context, req *pbgroup.Trans | |||||||
| 		return nil, errs.ErrArgs.WrapMsg("NewOwnerUser not in group " + req.NewOwnerUserID) | 		return nil, errs.ErrArgs.WrapMsg("NewOwnerUser not in group " + req.NewOwnerUserID) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if !authverify.IsAppManagerUid(ctx, g.config.Share.IMAdminUserID) { | 	if !authverify.IsAdmin(ctx) { | ||||||
| 		if !(mcontext.GetOpUserID(ctx) == oldOwner.UserID && oldOwner.RoleLevel == constant.GroupOwner) { | 		if !(mcontext.GetOpUserID(ctx) == oldOwner.UserID && oldOwner.RoleLevel == constant.GroupOwner) { | ||||||
| 			return nil, errs.ErrNoPermission.WrapMsg("no permission transfer group owner") | 			return nil, errs.ErrNoPermission.WrapMsg("no permission transfer group owner") | ||||||
| 		} | 		} | ||||||
| @ -1271,6 +1348,9 @@ func (g *groupServer) GetGroups(ctx context.Context, req *pbgroup.GetGroupsReq) | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *groupServer) GetGroupMembersCMS(ctx context.Context, req *pbgroup.GetGroupMembersCMSReq) (*pbgroup.GetGroupMembersCMSResp, error) { | func (g *groupServer) GetGroupMembersCMS(ctx context.Context, req *pbgroup.GetGroupMembersCMSReq) (*pbgroup.GetGroupMembersCMSResp, error) { | ||||||
|  | 	if err := g.checkAdminOrInGroup(ctx, req.GroupID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	total, members, err := g.db.SearchGroupMember(ctx, req.UserName, req.GroupID, req.Pagination) | 	total, members, err := g.db.SearchGroupMember(ctx, req.UserName, req.GroupID, req.Pagination) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -1287,11 +1367,17 @@ func (g *groupServer) GetGroupMembersCMS(ctx context.Context, req *pbgroup.GetGr | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *groupServer) GetUserReqApplicationList(ctx context.Context, req *pbgroup.GetUserReqApplicationListReq) (*pbgroup.GetUserReqApplicationListResp, error) { | func (g *groupServer) GetUserReqApplicationList(ctx context.Context, req *pbgroup.GetUserReqApplicationListReq) (*pbgroup.GetUserReqApplicationListResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	user, err := g.userClient.GetUserInfo(ctx, req.UserID) | 	user, err := g.userClient.GetUserInfo(ctx, req.UserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	total, requests, err := g.db.PageGroupRequestUser(ctx, req.UserID, req.Pagination) | 	handleResults := datautil.Slice(req.HandleResults, func(e int32) int { | ||||||
|  | 		return int(e) | ||||||
|  | 	}) | ||||||
|  | 	total, requests, err := g.db.PageGroupRequestUser(ctx, req.UserID, req.GroupIDs, handleResults, req.Pagination) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -1339,7 +1425,7 @@ func (g *groupServer) DismissGroup(ctx context.Context, req *pbgroup.DismissGrou | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	if !authverify.IsAppManagerUid(ctx, g.config.Share.IMAdminUserID) { | 	if !authverify.IsAdmin(ctx) { | ||||||
| 		if owner.UserID != mcontext.GetOpUserID(ctx) { | 		if owner.UserID != mcontext.GetOpUserID(ctx) { | ||||||
| 			return nil, errs.ErrNoPermission.WrapMsg("not group owner") | 			return nil, errs.ErrNoPermission.WrapMsg("not group owner") | ||||||
| 		} | 		} | ||||||
| @ -1362,6 +1448,7 @@ func (g *groupServer) DismissGroup(ctx context.Context, req *pbgroup.DismissGrou | |||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return nil, err | 			return nil, err | ||||||
| 		} | 		} | ||||||
|  | 		group.Status = constant.GroupStatusDismissed | ||||||
| 		tips := &sdkws.GroupDismissedTips{ | 		tips := &sdkws.GroupDismissedTips{ | ||||||
| 			Group:  g.groupDB2PB(group, owner.UserID, num), | 			Group:  g.groupDB2PB(group, owner.UserID, num), | ||||||
| 			OpUser: &sdkws.GroupMemberFullInfo{}, | 			OpUser: &sdkws.GroupMemberFullInfo{}, | ||||||
| @ -1395,7 +1482,7 @@ func (g *groupServer) MuteGroupMember(ctx context.Context, req *pbgroup.MuteGrou | |||||||
| 	if err := g.PopulateGroupMember(ctx, member); err != nil { | 	if err := g.PopulateGroupMember(ctx, member); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	if !authverify.IsAppManagerUid(ctx, g.config.Share.IMAdminUserID) { | 	if !authverify.IsAdmin(ctx) { | ||||||
| 		opMember, err := g.db.TakeGroupMember(ctx, req.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 | ||||||
| @ -1431,7 +1518,7 @@ func (g *groupServer) CancelMuteGroupMember(ctx context.Context, req *pbgroup.Ca | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if !authverify.IsAppManagerUid(ctx, g.config.Share.IMAdminUserID) { | 	if !authverify.IsAdmin(ctx) { | ||||||
| 		opMember, err := g.db.TakeGroupMember(ctx, req.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 | ||||||
| @ -1491,7 +1578,7 @@ func (g *groupServer) SetGroupMemberInfo(ctx context.Context, req *pbgroup.SetGr | |||||||
| 	if opUserID == "" { | 	if opUserID == "" { | ||||||
| 		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.IsAdmin(ctx) | ||||||
| 	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 { | ||||||
| @ -1646,6 +1733,11 @@ func (g *groupServer) GetGroupAbstractInfo(ctx context.Context, req *pbgroup.Get | |||||||
| 	if datautil.Duplicate(req.GroupIDs) { | 	if datautil.Duplicate(req.GroupIDs) { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("groupIDs duplicate") | 		return nil, errs.ErrArgs.WrapMsg("groupIDs duplicate") | ||||||
| 	} | 	} | ||||||
|  | 	for _, groupID := range req.GroupIDs { | ||||||
|  | 		if err := g.checkAdminOrInGroup(ctx, groupID); err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
| 	groups, err := g.db.FindGroup(ctx, req.GroupIDs) | 	groups, err := g.db.FindGroup(ctx, req.GroupIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -1674,6 +1766,9 @@ func (g *groupServer) GetUserInGroupMembers(ctx context.Context, req *pbgroup.Ge | |||||||
| 	if len(req.GroupIDs) == 0 { | 	if len(req.GroupIDs) == 0 { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("groupIDs empty") | 		return nil, errs.ErrArgs.WrapMsg("groupIDs empty") | ||||||
| 	} | 	} | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	members, err := g.db.FindGroupMemberUser(ctx, req.GroupIDs, req.UserID) | 	members, err := g.db.FindGroupMemberUser(ctx, req.GroupIDs, req.UserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -1693,6 +1788,9 @@ func (g *groupServer) GetGroupMemberUserIDs(ctx context.Context, req *pbgroup.Ge | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 	if err := authverify.CheckAccessIn(ctx, userIDs...); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	return &pbgroup.GetGroupMemberUserIDsResp{ | 	return &pbgroup.GetGroupMemberUserIDsResp{ | ||||||
| 		UserIDs: userIDs, | 		UserIDs: userIDs, | ||||||
| 	}, nil | 	}, nil | ||||||
| @ -1702,6 +1800,9 @@ func (g *groupServer) GetGroupMemberRoleLevel(ctx context.Context, req *pbgroup. | |||||||
| 	if len(req.RoleLevels) == 0 { | 	if len(req.RoleLevels) == 0 { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("RoleLevels empty") | 		return nil, errs.ErrArgs.WrapMsg("RoleLevels empty") | ||||||
| 	} | 	} | ||||||
|  | 	if err := g.checkAdminOrInGroup(ctx, req.GroupID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	members, err := g.db.FindGroupMemberRoleLevels(ctx, req.GroupID, req.RoleLevels) | 	members, err := g.db.FindGroupMemberRoleLevels(ctx, req.GroupID, req.RoleLevels) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -1717,6 +1818,9 @@ func (g *groupServer) GetGroupMemberRoleLevel(ctx context.Context, req *pbgroup. | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *groupServer) GetGroupUsersReqApplicationList(ctx context.Context, req *pbgroup.GetGroupUsersReqApplicationListReq) (*pbgroup.GetGroupUsersReqApplicationListResp, error) { | func (g *groupServer) GetGroupUsersReqApplicationList(ctx context.Context, req *pbgroup.GetGroupUsersReqApplicationListReq) (*pbgroup.GetGroupUsersReqApplicationListResp, error) { | ||||||
|  | 	if err := g.CheckGroupAdmin(ctx, req.GroupID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	requests, err := g.db.FindGroupRequests(ctx, req.GroupID, req.UserIDs) | 	requests, err := g.db.FindGroupRequests(ctx, req.GroupID, req.UserIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -1799,7 +1903,7 @@ func (g *groupServer) GetSpecifiedUserGroupRequestInfo(ctx context.Context, req | |||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		adminIDs = append(adminIDs, owners[0].UserID) | 		adminIDs = append(adminIDs, owners[0].UserID) | ||||||
| 		adminIDs = append(adminIDs, g.config.Share.IMAdminUserID...) | 		adminIDs = append(adminIDs, g.adminUserIDs...) | ||||||
| 
 | 
 | ||||||
| 		if !datautil.Contain(opUserID, adminIDs...) { | 		if !datautil.Contain(opUserID, adminIDs...) { | ||||||
| 			return nil, errs.ErrNoPermission.WrapMsg("opUser no permission") | 			return nil, errs.ErrNoPermission.WrapMsg("opUser no permission") | ||||||
|  | |||||||
| @ -20,6 +20,7 @@ import ( | |||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"time" | 	"time" | ||||||
| 
 | 
 | ||||||
|  | 	"github.com/google/uuid" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | ||||||
| 
 | 
 | ||||||
| 	"go.mongodb.org/mongo-driver/mongo" | 	"go.mongodb.org/mongo-driver/mongo" | ||||||
| @ -234,17 +235,17 @@ func (g *NotificationSender) groupMemberDB2PB(member *model.GroupMember, appMang | |||||||
| 	return result, nil | 	return result, nil | ||||||
| } */ | } */ | ||||||
| 
 | 
 | ||||||
| func (g *NotificationSender) fillOpUser(ctx context.Context, opUser **sdkws.GroupMemberFullInfo, groupID string) (err error) { | func (g *NotificationSender) fillOpUser(ctx context.Context, targetUser **sdkws.GroupMemberFullInfo, groupID string) (err error) { | ||||||
| 	return g.fillOpUserByUserID(ctx, mcontext.GetOpUserID(ctx), opUser, groupID) | 	return g.fillUserByUserID(ctx, mcontext.GetOpUserID(ctx), targetUser, groupID) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *NotificationSender) fillOpUserByUserID(ctx context.Context, userID string, opUser **sdkws.GroupMemberFullInfo, groupID string) error { | func (g *NotificationSender) fillUserByUserID(ctx context.Context, userID string, targetUser **sdkws.GroupMemberFullInfo, groupID string) error { | ||||||
| 	if opUser == nil { | 	if targetUser == nil { | ||||||
| 		return errs.ErrInternalServer.WrapMsg("**sdkws.GroupMemberFullInfo is nil") | 		return errs.ErrInternalServer.WrapMsg("**sdkws.GroupMemberFullInfo is nil") | ||||||
| 	} | 	} | ||||||
| 	if groupID != "" { | 	if groupID != "" { | ||||||
| 		if authverify.IsManagerUserID(userID, g.config.Share.IMAdminUserID) { | 		if authverify.CheckUserIsAdmin(ctx, userID) { | ||||||
| 			*opUser = &sdkws.GroupMemberFullInfo{ | 			*targetUser = &sdkws.GroupMemberFullInfo{ | ||||||
| 				GroupID:        groupID, | 				GroupID:        groupID, | ||||||
| 				UserID:         userID, | 				UserID:         userID, | ||||||
| 				RoleLevel:      constant.GroupAdmin, | 				RoleLevel:      constant.GroupAdmin, | ||||||
| @ -253,7 +254,7 @@ func (g *NotificationSender) fillOpUserByUserID(ctx context.Context, userID stri | |||||||
| 		} else { | 		} else { | ||||||
| 			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) | 				*targetUser = g.groupMemberDB2PB(member, 0) | ||||||
| 			} else if !(errors.Is(err, mongo.ErrNoDocuments) || errs.ErrRecordNotFound.Is(err)) { | 			} else if !(errors.Is(err, mongo.ErrNoDocuments) || errs.ErrRecordNotFound.Is(err)) { | ||||||
| 				return err | 				return err | ||||||
| 			} | 			} | ||||||
| @ -263,8 +264,8 @@ func (g *NotificationSender) fillOpUserByUserID(ctx context.Context, userID stri | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	if *opUser == nil { | 	if *targetUser == nil { | ||||||
| 		*opUser = &sdkws.GroupMemberFullInfo{ | 		*targetUser = &sdkws.GroupMemberFullInfo{ | ||||||
| 			GroupID:        groupID, | 			GroupID:        groupID, | ||||||
| 			UserID:         userID, | 			UserID:         userID, | ||||||
| 			Nickname:       user.Nickname, | 			Nickname:       user.Nickname, | ||||||
| @ -272,11 +273,11 @@ func (g *NotificationSender) fillOpUserByUserID(ctx context.Context, userID stri | |||||||
| 			OperatorUserID: userID, | 			OperatorUserID: userID, | ||||||
| 		} | 		} | ||||||
| 	} else { | 	} else { | ||||||
| 		if (*opUser).Nickname == "" { | 		if (*targetUser).Nickname == "" { | ||||||
| 			(*opUser).Nickname = user.Nickname | 			(*targetUser).Nickname = user.Nickname | ||||||
| 		} | 		} | ||||||
| 		if (*opUser).FaceURL == "" { | 		if (*targetUser).FaceURL == "" { | ||||||
| 			(*opUser).FaceURL = user.FaceURL | 			(*targetUser).FaceURL = user.FaceURL | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	return nil | 	return nil | ||||||
| @ -284,7 +285,8 @@ func (g *NotificationSender) fillOpUserByUserID(ctx context.Context, userID stri | |||||||
| 
 | 
 | ||||||
| func (g *NotificationSender) setVersion(ctx context.Context, version *uint64, versionID *string, collName string, id string) { | func (g *NotificationSender) setVersion(ctx context.Context, version *uint64, versionID *string, collName string, id string) { | ||||||
| 	versions := versionctx.GetVersionLog(ctx).Get() | 	versions := versionctx.GetVersionLog(ctx).Get() | ||||||
| 	for _, coll := range versions { | 	for i := len(versions) - 1; i >= 0; i-- { | ||||||
|  | 		coll := versions[i] | ||||||
| 		if coll.Name == collName && coll.Doc.DID == id { | 		if coll.Name == collName && coll.Doc.DID == id { | ||||||
| 			*version = uint64(coll.Doc.Version) | 			*version = uint64(coll.Doc.Version) | ||||||
| 			*versionID = coll.Doc.ID.Hex() | 			*versionID = coll.Doc.ID.Hex() | ||||||
| @ -350,7 +352,7 @@ func (g *NotificationSender) GroupInfoSetNameNotification(ctx context.Context, t | |||||||
| 	g.Notification(ctx, mcontext.GetOpUserID(ctx), tips.Group.GroupID, constant.GroupInfoSetNameNotification, tips) | 	g.Notification(ctx, mcontext.GetOpUserID(ctx), tips.Group.GroupID, constant.GroupInfoSetNameNotification, tips) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *NotificationSender) GroupInfoSetAnnouncementNotification(ctx context.Context, tips *sdkws.GroupInfoSetAnnouncementTips) { | func (g *NotificationSender) GroupInfoSetAnnouncementNotification(ctx context.Context, tips *sdkws.GroupInfoSetAnnouncementTips, sendMessage *bool) { | ||||||
| 	var err error | 	var err error | ||||||
| 	defer func() { | 	defer func() { | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| @ -361,16 +363,49 @@ func (g *NotificationSender) GroupInfoSetAnnouncementNotification(ctx context.Co | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	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), tips.Group.GroupID, constant.GroupInfoSetAnnouncementNotification, tips, notification.WithRpcGetUserName()) | 	g.Notification(ctx, mcontext.GetOpUserID(ctx), tips.Group.GroupID, constant.GroupInfoSetAnnouncementNotification, tips, notification.WithRpcGetUserName(), notification.WithSendMessage(sendMessage)) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *NotificationSender) JoinGroupApplicationNotification(ctx context.Context, req *pbgroup.JoinGroupReq) { | func (g *NotificationSender) uuid() string { | ||||||
|  | 	return uuid.New().String() | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (g *NotificationSender) getGroupRequest(ctx context.Context, groupID string, userID string) (*sdkws.GroupRequest, error) { | ||||||
|  | 	request, err := g.db.TakeGroupRequest(ctx, groupID, userID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	users, err := g.getUsersInfo(ctx, []string{userID}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	if len(users) == 0 { | ||||||
|  | 		return nil, servererrs.ErrUserIDNotFound.WrapMsg(fmt.Sprintf("user %s not found", userID)) | ||||||
|  | 	} | ||||||
|  | 	info, ok := users[0].(*sdkws.UserInfo) | ||||||
|  | 	if !ok { | ||||||
|  | 		info = &sdkws.UserInfo{ | ||||||
|  | 			UserID:   users[0].GetUserID(), | ||||||
|  | 			Nickname: users[0].GetNickname(), | ||||||
|  | 			FaceURL:  users[0].GetFaceURL(), | ||||||
|  | 			Ex:       users[0].GetEx(), | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return convert.Db2PbGroupRequest(request, info, nil), nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (g *NotificationSender) JoinGroupApplicationNotification(ctx context.Context, req *pbgroup.JoinGroupReq, dbReq *model.GroupRequest) { | ||||||
| 	var err error | 	var err error | ||||||
| 	defer func() { | 	defer func() { | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.ZError(ctx, stringutil.GetFuncName(1)+" failed", err) | 			log.ZError(ctx, stringutil.GetFuncName(1)+" failed", err) | ||||||
| 		} | 		} | ||||||
| 	}() | 	}() | ||||||
|  | 	request, err := g.getGroupRequest(ctx, dbReq.GroupID, dbReq.UserID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.ZError(ctx, "JoinGroupApplicationNotification getGroupRequest", err, "dbReq", dbReq) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
| 	var group *sdkws.GroupInfo | 	var group *sdkws.GroupInfo | ||||||
| 	group, err = g.getGroupInfo(ctx, req.GroupID) | 	group, err = g.getGroupInfo(ctx, req.GroupID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -386,7 +421,13 @@ func (g *NotificationSender) JoinGroupApplicationNotification(ctx context.Contex | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	userIDs = append(userIDs, req.InviterUserID, mcontext.GetOpUserID(ctx)) | 	userIDs = append(userIDs, req.InviterUserID, mcontext.GetOpUserID(ctx)) | ||||||
| 	tips := &sdkws.JoinGroupApplicationTips{Group: group, Applicant: user, ReqMsg: req.ReqMessage} | 	tips := &sdkws.JoinGroupApplicationTips{ | ||||||
|  | 		Group:     group, | ||||||
|  | 		Applicant: user, | ||||||
|  | 		ReqMsg:    req.ReqMessage, | ||||||
|  | 		Uuid:      g.uuid(), | ||||||
|  | 		Request:   request, | ||||||
|  | 	} | ||||||
| 	for _, userID := range datautil.Distinct(userIDs) { | 	for _, userID := range datautil.Distinct(userIDs) { | ||||||
| 		g.Notification(ctx, mcontext.GetOpUserID(ctx), userID, constant.JoinGroupApplicationNotification, tips) | 		g.Notification(ctx, mcontext.GetOpUserID(ctx), userID, constant.JoinGroupApplicationNotification, tips) | ||||||
| 	} | 	} | ||||||
| @ -416,6 +457,11 @@ func (g *NotificationSender) GroupApplicationAcceptedNotification(ctx context.Co | |||||||
| 			log.ZError(ctx, stringutil.GetFuncName(1)+" failed", err) | 			log.ZError(ctx, stringutil.GetFuncName(1)+" failed", err) | ||||||
| 		} | 		} | ||||||
| 	}() | 	}() | ||||||
|  | 	request, err := g.getGroupRequest(ctx, req.GroupID, req.FromUserID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.ZError(ctx, "GroupApplicationAcceptedNotification getGroupRequest", err, "req", req) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
| 	var group *sdkws.GroupInfo | 	var group *sdkws.GroupInfo | ||||||
| 	group, err = g.getGroupInfo(ctx, req.GroupID) | 	group, err = g.getGroupInfo(ctx, req.GroupID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -431,8 +477,14 @@ func (g *NotificationSender) GroupApplicationAcceptedNotification(ctx context.Co | |||||||
| 	if err = g.fillOpUser(ctx, &opUser, group.GroupID); err != nil { | 	if err = g.fillOpUser(ctx, &opUser, group.GroupID); err != nil { | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  | 	tips := &sdkws.GroupApplicationAcceptedTips{ | ||||||
|  | 		Group:     group, | ||||||
|  | 		OpUser:    opUser, | ||||||
|  | 		HandleMsg: req.HandledMsg, | ||||||
|  | 		Uuid:      g.uuid(), | ||||||
|  | 		Request:   request, | ||||||
|  | 	} | ||||||
| 	for _, userID := range append(userIDs, req.FromUserID) { | 	for _, userID := range append(userIDs, req.FromUserID) { | ||||||
| 		tips := &sdkws.GroupApplicationAcceptedTips{Group: group, OpUser: opUser, HandleMsg: req.HandledMsg} |  | ||||||
| 		if userID == req.FromUserID { | 		if userID == req.FromUserID { | ||||||
| 			tips.ReceiverAs = applicantReceiver | 			tips.ReceiverAs = applicantReceiver | ||||||
| 		} else { | 		} else { | ||||||
| @ -449,6 +501,11 @@ func (g *NotificationSender) GroupApplicationRejectedNotification(ctx context.Co | |||||||
| 			log.ZError(ctx, stringutil.GetFuncName(1)+" failed", err) | 			log.ZError(ctx, stringutil.GetFuncName(1)+" failed", err) | ||||||
| 		} | 		} | ||||||
| 	}() | 	}() | ||||||
|  | 	request, err := g.getGroupRequest(ctx, req.GroupID, req.FromUserID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.ZError(ctx, "GroupApplicationAcceptedNotification getGroupRequest", err, "req", req) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
| 	var group *sdkws.GroupInfo | 	var group *sdkws.GroupInfo | ||||||
| 	group, err = g.getGroupInfo(ctx, req.GroupID) | 	group, err = g.getGroupInfo(ctx, req.GroupID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -464,8 +521,14 @@ func (g *NotificationSender) GroupApplicationRejectedNotification(ctx context.Co | |||||||
| 	if err = g.fillOpUser(ctx, &opUser, group.GroupID); err != nil { | 	if err = g.fillOpUser(ctx, &opUser, group.GroupID); err != nil { | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  | 	tips := &sdkws.GroupApplicationRejectedTips{ | ||||||
|  | 		Group:     group, | ||||||
|  | 		OpUser:    opUser, | ||||||
|  | 		HandleMsg: req.HandledMsg, | ||||||
|  | 		Uuid:      g.uuid(), | ||||||
|  | 		Request:   request, | ||||||
|  | 	} | ||||||
| 	for _, userID := range append(userIDs, req.FromUserID) { | 	for _, userID := range append(userIDs, req.FromUserID) { | ||||||
| 		tips := &sdkws.GroupApplicationAcceptedTips{Group: group, OpUser: opUser, HandleMsg: req.HandledMsg} |  | ||||||
| 		if userID == req.FromUserID { | 		if userID == req.FromUserID { | ||||||
| 			tips.ReceiverAs = applicantReceiver | 			tips.ReceiverAs = applicantReceiver | ||||||
| 		} else { | 		} else { | ||||||
| @ -521,6 +584,10 @@ func (g *NotificationSender) MemberKickedNotification(ctx context.Context, tips | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *NotificationSender) GroupApplicationAgreeMemberEnterNotification(ctx context.Context, groupID string, SendMessage *bool, invitedOpUserID string, entrantUserID ...string) error { | func (g *NotificationSender) GroupApplicationAgreeMemberEnterNotification(ctx context.Context, groupID string, SendMessage *bool, invitedOpUserID string, entrantUserID ...string) error { | ||||||
|  | 	return g.groupApplicationAgreeMemberEnterNotification(ctx, groupID, SendMessage, invitedOpUserID, entrantUserID...) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (g *NotificationSender) groupApplicationAgreeMemberEnterNotification(ctx context.Context, groupID string, SendMessage *bool, invitedOpUserID string, entrantUserID ...string) error { | ||||||
| 	var err error | 	var err error | ||||||
| 	defer func() { | 	defer func() { | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| @ -557,15 +624,13 @@ func (g *NotificationSender) GroupApplicationAgreeMemberEnterNotification(ctx co | |||||||
| 		InvitedUserList: users, | 		InvitedUserList: users, | ||||||
| 	} | 	} | ||||||
| 	opUserID := mcontext.GetOpUserID(ctx) | 	opUserID := mcontext.GetOpUserID(ctx) | ||||||
| 	if err = g.fillOpUserByUserID(ctx, opUserID, &tips.OpUser, tips.Group.GroupID); err != nil { | 	if err = g.fillUserByUserID(ctx, opUserID, &tips.OpUser, tips.Group.GroupID); err != nil { | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| 	switch { | 	if invitedOpUserID == opUserID { | ||||||
| 	case invitedOpUserID == "": |  | ||||||
| 	case invitedOpUserID == opUserID: |  | ||||||
| 		tips.InviterUser = tips.OpUser | 		tips.InviterUser = tips.OpUser | ||||||
| 	default: | 	} else { | ||||||
| 		if err = g.fillOpUserByUserID(ctx, invitedOpUserID, &tips.InviterUser, tips.Group.GroupID); err != nil { | 		if err = g.fillUserByUserID(ctx, invitedOpUserID, &tips.InviterUser, tips.Group.GroupID); err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @ -782,7 +847,7 @@ func (g *NotificationSender) GroupMemberSetToAdminNotification(ctx context.Conte | |||||||
| 	if err = g.fillOpUser(ctx, &tips.OpUser, tips.Group.GroupID); err != nil { | 	if err = g.fillOpUser(ctx, &tips.OpUser, tips.Group.GroupID); err != nil { | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	g.setVersion(ctx, &tips.GroupMemberVersion, &tips.GroupMemberVersionID, database.GroupMemberVersionName, tips.Group.GroupID) | 	g.setSortVersion(ctx, &tips.GroupMemberVersion, &tips.GroupMemberVersionID, database.GroupMemberVersionName, tips.Group.GroupID, &tips.GroupSortVersion) | ||||||
| 	g.Notification(ctx, mcontext.GetOpUserID(ctx), group.GroupID, constant.GroupMemberSetToAdminNotification, tips) | 	g.Notification(ctx, mcontext.GetOpUserID(ctx), group.GroupID, constant.GroupMemberSetToAdminNotification, tips) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -807,6 +872,6 @@ func (g *NotificationSender) GroupMemberSetToOrdinaryUserNotification(ctx contex | |||||||
| 	if err = g.fillOpUser(ctx, &tips.OpUser, tips.Group.GroupID); err != nil { | 	if err = g.fillOpUser(ctx, &tips.OpUser, tips.Group.GroupID); err != nil { | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	g.setVersion(ctx, &tips.GroupMemberVersion, &tips.GroupMemberVersionID, database.GroupMemberVersionName, tips.Group.GroupID) | 	g.setSortVersion(ctx, &tips.GroupMemberVersion, &tips.GroupMemberVersionID, database.GroupMemberVersionName, tips.Group.GroupID, &tips.GroupSortVersion) | ||||||
| 	g.Notification(ctx, mcontext.GetOpUserID(ctx), group.GroupID, constant.GroupMemberSetToOrdinaryUserNotification, tips) | 	g.Notification(ctx, mcontext.GetOpUserID(ctx), group.GroupID, constant.GroupMemberSetToOrdinaryUserNotification, tips) | ||||||
| } | } | ||||||
|  | |||||||
| @ -18,6 +18,7 @@ import ( | |||||||
| 	"context" | 	"context" | ||||||
| 	"time" | 	"time" | ||||||
| 
 | 
 | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||||
| 	"github.com/openimsdk/protocol/group" | 	"github.com/openimsdk/protocol/group" | ||||||
| 	"github.com/openimsdk/tools/errs" | 	"github.com/openimsdk/tools/errs" | ||||||
| ) | ) | ||||||
| @ -26,6 +27,9 @@ func (g *groupServer) GroupCreateCount(ctx context.Context, req *group.GroupCrea | |||||||
| 	if req.Start > req.End { | 	if req.Start > req.End { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("start > end: %d > %d", req.Start, req.End) | 		return nil, errs.ErrArgs.WrapMsg("start > end: %d > %d", req.Start, req.End) | ||||||
| 	} | 	} | ||||||
|  | 	if err := authverify.CheckAdmin(ctx); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	total, err := g.db.CountTotal(ctx, nil) | 	total, err := g.db.CountTotal(ctx, nil) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
|  | |||||||
| @ -11,16 +11,19 @@ import ( | |||||||
| 	"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/sdkws" | 	"github.com/openimsdk/protocol/sdkws" | ||||||
| 	"github.com/openimsdk/tools/errs" |  | ||||||
| 	"github.com/openimsdk/tools/log" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | const versionSyncLimit = 500 | ||||||
|  | 
 | ||||||
| func (g *groupServer) GetFullGroupMemberUserIDs(ctx context.Context, req *pbgroup.GetFullGroupMemberUserIDsReq) (*pbgroup.GetFullGroupMemberUserIDsResp, error) { | func (g *groupServer) GetFullGroupMemberUserIDs(ctx context.Context, req *pbgroup.GetFullGroupMemberUserIDsReq) (*pbgroup.GetFullGroupMemberUserIDsResp, error) { | ||||||
| 	vl, err := g.db.FindMaxGroupMemberVersionCache(ctx, req.GroupID) | 	userIDs, err := g.db.FindGroupMemberUserID(ctx, req.GroupID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	userIDs, err := g.db.FindGroupMemberUserID(ctx, req.GroupID) | 	if err := authverify.CheckAccessIn(ctx, userIDs...); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	vl, err := g.db.FindMaxGroupMemberVersionCache(ctx, req.GroupID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -37,6 +40,9 @@ func (g *groupServer) GetFullGroupMemberUserIDs(ctx context.Context, req *pbgrou | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *groupServer) GetFullJoinGroupIDs(ctx context.Context, req *pbgroup.GetFullJoinGroupIDsReq) (*pbgroup.GetFullJoinGroupIDsResp, error) { | func (g *groupServer) GetFullJoinGroupIDs(ctx context.Context, req *pbgroup.GetFullJoinGroupIDsReq) (*pbgroup.GetFullJoinGroupIDsResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	vl, err := g.db.FindMaxJoinGroupVersionCache(ctx, req.UserID) | 	vl, err := g.db.FindMaxJoinGroupVersionCache(ctx, req.UserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -58,6 +64,9 @@ func (g *groupServer) GetFullJoinGroupIDs(ctx context.Context, req *pbgroup.GetF | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *groupServer) GetIncrementalGroupMember(ctx context.Context, req *pbgroup.GetIncrementalGroupMemberReq) (*pbgroup.GetIncrementalGroupMemberResp, error) { | func (g *groupServer) GetIncrementalGroupMember(ctx context.Context, req *pbgroup.GetIncrementalGroupMemberReq) (*pbgroup.GetIncrementalGroupMemberResp, error) { | ||||||
|  | 	if err := g.checkAdminOrInGroup(ctx, req.GroupID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	group, err := g.db.TakeGroup(ctx, req.GroupID) | 	group, err := g.db.TakeGroup(ctx, req.GroupID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -132,152 +141,8 @@ func (g *groupServer) GetIncrementalGroupMember(ctx context.Context, req *pbgrou | |||||||
| 	return resp, nil | 	return resp, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (g *groupServer) BatchGetIncrementalGroupMember(ctx context.Context, req *pbgroup.BatchGetIncrementalGroupMemberReq) (resp *pbgroup.BatchGetIncrementalGroupMemberResp, err error) { |  | ||||||
| 	type VersionInfo struct { |  | ||||||
| 		GroupID       string |  | ||||||
| 		VersionID     string |  | ||||||
| 		VersionNumber uint64 |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	var groupIDs []string |  | ||||||
| 
 |  | ||||||
| 	groupsVersionMap := make(map[string]*VersionInfo) |  | ||||||
| 	groupsMap := make(map[string]*model.Group) |  | ||||||
| 	hasGroupUpdateMap := make(map[string]bool) |  | ||||||
| 	sortVersionMap := make(map[string]uint64) |  | ||||||
| 
 |  | ||||||
| 	var targetKeys, versionIDs []string |  | ||||||
| 	var versionNumbers []uint64 |  | ||||||
| 
 |  | ||||||
| 	var requestBodyLen int |  | ||||||
| 
 |  | ||||||
| 	for _, group := range req.ReqList { |  | ||||||
| 		groupsVersionMap[group.GroupID] = &VersionInfo{ |  | ||||||
| 			GroupID:       group.GroupID, |  | ||||||
| 			VersionID:     group.VersionID, |  | ||||||
| 			VersionNumber: group.Version, |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		groupIDs = append(groupIDs, group.GroupID) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	groups, err := g.db.FindGroup(ctx, groupIDs) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, errs.Wrap(err) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	for _, group := range groups { |  | ||||||
| 		if group.Status == constant.GroupStatusDismissed { |  | ||||||
| 			err = servererrs.ErrDismissedAlready.Wrap() |  | ||||||
| 			log.ZError(ctx, "This group is Dismissed Already", err, "group is", group.GroupID) |  | ||||||
| 
 |  | ||||||
| 			delete(groupsVersionMap, group.GroupID) |  | ||||||
| 		} else { |  | ||||||
| 			groupsMap[group.GroupID] = group |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	for groupID, vInfo := range groupsVersionMap { |  | ||||||
| 		targetKeys = append(targetKeys, groupID) |  | ||||||
| 		versionIDs = append(versionIDs, vInfo.VersionID) |  | ||||||
| 		versionNumbers = append(versionNumbers, vInfo.VersionNumber) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	opt := incrversion.BatchOption[[]*sdkws.GroupMemberFullInfo, pbgroup.BatchGetIncrementalGroupMemberResp]{ |  | ||||||
| 		Ctx:            ctx, |  | ||||||
| 		TargetKeys:     targetKeys, |  | ||||||
| 		VersionIDs:     versionIDs, |  | ||||||
| 		VersionNumbers: versionNumbers, |  | ||||||
| 		Versions: func(ctx context.Context, groupIDs []string, versions []uint64, limits []int) (map[string]*model.VersionLog, error) { |  | ||||||
| 			vLogs, err := g.db.BatchFindMemberIncrVersion(ctx, groupIDs, versions, limits) |  | ||||||
| 			if err != nil { |  | ||||||
| 				return nil, errs.Wrap(err) |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			for groupID, vlog := range vLogs { |  | ||||||
| 				vlogElems := make([]model.VersionLogElem, 0, len(vlog.Logs)) |  | ||||||
| 				for i, log := range vlog.Logs { |  | ||||||
| 					switch log.EID { |  | ||||||
| 					case model.VersionGroupChangeID: |  | ||||||
| 						vlog.LogLen-- |  | ||||||
| 						hasGroupUpdateMap[groupID] = true |  | ||||||
| 					case model.VersionSortChangeID: |  | ||||||
| 						vlog.LogLen-- |  | ||||||
| 						sortVersionMap[groupID] = uint64(log.Version) |  | ||||||
| 					default: |  | ||||||
| 						vlogElems = append(vlogElems, vlog.Logs[i]) |  | ||||||
| 					} |  | ||||||
| 				} |  | ||||||
| 				vlog.Logs = vlogElems |  | ||||||
| 				if vlog.LogLen > 0 { |  | ||||||
| 					hasGroupUpdateMap[groupID] = true |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			return vLogs, nil |  | ||||||
| 		}, |  | ||||||
| 		CacheMaxVersions: g.db.BatchFindMaxGroupMemberVersionCache, |  | ||||||
| 		Find: func(ctx context.Context, groupID string, ids []string) ([]*sdkws.GroupMemberFullInfo, error) { |  | ||||||
| 			memberInfo, err := g.getGroupMembersInfo(ctx, groupID, ids) |  | ||||||
| 			if err != nil { |  | ||||||
| 				return nil, err |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			return memberInfo, err |  | ||||||
| 		}, |  | ||||||
| 		Resp: func(versions map[string]*model.VersionLog, deleteIdsMap map[string][]string, insertListMap, updateListMap map[string][]*sdkws.GroupMemberFullInfo, fullMap map[string]bool) *pbgroup.BatchGetIncrementalGroupMemberResp { |  | ||||||
| 			resList := make(map[string]*pbgroup.GetIncrementalGroupMemberResp) |  | ||||||
| 
 |  | ||||||
| 			for groupID, versionLog := range versions { |  | ||||||
| 				resList[groupID] = &pbgroup.GetIncrementalGroupMemberResp{ |  | ||||||
| 					VersionID:   versionLog.ID.Hex(), |  | ||||||
| 					Version:     uint64(versionLog.Version), |  | ||||||
| 					Full:        fullMap[groupID], |  | ||||||
| 					Delete:      deleteIdsMap[groupID], |  | ||||||
| 					Insert:      insertListMap[groupID], |  | ||||||
| 					Update:      updateListMap[groupID], |  | ||||||
| 					SortVersion: sortVersionMap[groupID], |  | ||||||
| 				} |  | ||||||
| 
 |  | ||||||
| 				requestBodyLen += len(insertListMap[groupID]) + len(updateListMap[groupID]) + len(deleteIdsMap[groupID]) |  | ||||||
| 				if requestBodyLen > 200 { |  | ||||||
| 					break |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			return &pbgroup.BatchGetIncrementalGroupMemberResp{ |  | ||||||
| 				RespList: resList, |  | ||||||
| 			} |  | ||||||
| 		}, |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	resp, err = opt.Build() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, errs.Wrap(err) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	for groupID, val := range resp.RespList { |  | ||||||
| 		if val.Full || hasGroupUpdateMap[groupID] { |  | ||||||
| 			count, err := g.db.FindGroupMemberNum(ctx, groupID) |  | ||||||
| 			if err != nil { |  | ||||||
| 				return nil, err |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			owner, err := g.db.TakeGroupOwner(ctx, groupID) |  | ||||||
| 			if err != nil { |  | ||||||
| 				return nil, err |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			resp.RespList[groupID].Group = g.groupDB2PB(groupsMap[groupID], owner.UserID, count) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	return resp, nil |  | ||||||
| 
 |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (g *groupServer) GetIncrementalJoinGroup(ctx context.Context, req *pbgroup.GetIncrementalJoinGroupReq) (*pbgroup.GetIncrementalJoinGroupResp, error) { | func (g *groupServer) GetIncrementalJoinGroup(ctx context.Context, req *pbgroup.GetIncrementalJoinGroupReq) (*pbgroup.GetIncrementalJoinGroupResp, error) { | ||||||
| 	if err := authverify.CheckAccessV3(ctx, req.UserID, g.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	opt := incrversion.Option[*sdkws.GroupInfo, pbgroup.GetIncrementalJoinGroupResp]{ | 	opt := incrversion.Option[*sdkws.GroupInfo, pbgroup.GetIncrementalJoinGroupResp]{ | ||||||
| @ -301,3 +166,23 @@ func (g *groupServer) GetIncrementalJoinGroup(ctx context.Context, req *pbgroup. | |||||||
| 	} | 	} | ||||||
| 	return opt.Build() | 	return opt.Build() | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | func (g *groupServer) BatchGetIncrementalGroupMember(ctx context.Context, req *pbgroup.BatchGetIncrementalGroupMemberReq) (*pbgroup.BatchGetIncrementalGroupMemberResp, error) { | ||||||
|  | 	var num int | ||||||
|  | 	resp := make(map[string]*pbgroup.GetIncrementalGroupMemberResp) | ||||||
|  | 	for _, memberReq := range req.ReqList { | ||||||
|  | 		if _, ok := resp[memberReq.GroupID]; ok { | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  | 		memberResp, err := g.GetIncrementalGroupMember(ctx, memberReq) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 		resp[memberReq.GroupID] = memberResp | ||||||
|  | 		num += len(memberResp.Insert) + len(memberResp.Update) + len(memberResp.Delete) | ||||||
|  | 		if num >= versionSyncLimit { | ||||||
|  | 			break | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return &pbgroup.BatchGetIncrementalGroupMemberResp{RespList: resp}, nil | ||||||
|  | } | ||||||
|  | |||||||
| @ -18,6 +18,7 @@ import ( | |||||||
| 	"context" | 	"context" | ||||||
| 	"errors" | 	"errors" | ||||||
| 
 | 
 | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||||
| 	cbapi "github.com/openimsdk/open-im-server/v3/pkg/callbackstruct" | 	cbapi "github.com/openimsdk/open-im-server/v3/pkg/callbackstruct" | ||||||
| 	"github.com/openimsdk/protocol/constant" | 	"github.com/openimsdk/protocol/constant" | ||||||
| 	"github.com/openimsdk/protocol/msg" | 	"github.com/openimsdk/protocol/msg" | ||||||
| @ -29,6 +30,9 @@ import ( | |||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) GetConversationsHasReadAndMaxSeq(ctx context.Context, req *msg.GetConversationsHasReadAndMaxSeqReq) (*msg.GetConversationsHasReadAndMaxSeqResp, error) { | func (m *msgServer) GetConversationsHasReadAndMaxSeq(ctx context.Context, req *msg.GetConversationsHasReadAndMaxSeqReq) (*msg.GetConversationsHasReadAndMaxSeqResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	var conversationIDs []string | 	var conversationIDs []string | ||||||
| 	if len(req.ConversationIDs) == 0 { | 	if len(req.ConversationIDs) == 0 { | ||||||
| 		var err error | 		var err error | ||||||
| @ -61,6 +65,13 @@ func (m *msgServer) GetConversationsHasReadAndMaxSeq(ctx context.Context, req *m | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	resp := &msg.GetConversationsHasReadAndMaxSeqResp{Seqs: make(map[string]*msg.Seqs)} | 	resp := &msg.GetConversationsHasReadAndMaxSeqResp{Seqs: make(map[string]*msg.Seqs)} | ||||||
|  | 	if req.ReturnPinned { | ||||||
|  | 		pinnedConversationIDs, err := m.ConversationLocalCache.GetPinnedConversationIDs(ctx, req.UserID) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 		resp.PinnedConversationIDs = pinnedConversationIDs | ||||||
|  | 	} | ||||||
| 	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], | ||||||
| @ -75,6 +86,9 @@ func (m *msgServer) GetConversationsHasReadAndMaxSeq(ctx context.Context, req *m | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) SetConversationHasReadSeq(ctx context.Context, req *msg.SetConversationHasReadSeqReq) (*msg.SetConversationHasReadSeqResp, error) { | func (m *msgServer) SetConversationHasReadSeq(ctx context.Context, req *msg.SetConversationHasReadSeqReq) (*msg.SetConversationHasReadSeqResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	maxSeq, err := m.MsgDatabase.GetMaxSeq(ctx, req.ConversationID) | 	maxSeq, err := m.MsgDatabase.GetMaxSeq(ctx, req.ConversationID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -90,8 +104,8 @@ func (m *msgServer) SetConversationHasReadSeq(ctx context.Context, req *msg.SetC | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) MarkMsgsAsRead(ctx context.Context, req *msg.MarkMsgsAsReadReq) (*msg.MarkMsgsAsReadResp, error) { | func (m *msgServer) MarkMsgsAsRead(ctx context.Context, req *msg.MarkMsgsAsReadReq) (*msg.MarkMsgsAsReadResp, error) { | ||||||
| 	if len(req.Seqs) < 1 { | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("seqs must not be empty") | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	maxSeq, err := m.MsgDatabase.GetMaxSeq(ctx, req.ConversationID) | 	maxSeq, err := m.MsgDatabase.GetMaxSeq(ctx, req.ConversationID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -132,6 +146,9 @@ func (m *msgServer) MarkMsgsAsRead(ctx context.Context, req *msg.MarkMsgsAsReadR | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) MarkConversationAsRead(ctx context.Context, req *msg.MarkConversationAsReadReq) (*msg.MarkConversationAsReadResp, error) { | func (m *msgServer) MarkConversationAsRead(ctx context.Context, req *msg.MarkConversationAsReadReq) (*msg.MarkConversationAsReadResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	conversation, err := m.ConversationLocalCache.GetConversation(ctx, req.UserID, req.ConversationID) | 	conversation, err := m.ConversationLocalCache.GetConversation(ctx, req.UserID, req.ConversationID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -209,5 +226,4 @@ func (m *msgServer) sendMarkAsReadNotification(ctx context.Context, conversation | |||||||
| 		HasReadSeq:       hasReadSeq, | 		HasReadSeq:       hasReadSeq, | ||||||
| 	} | 	} | ||||||
| 	m.notificationSender.NotificationWithSessionType(ctx, sendID, recvID, constant.HasReadReceipt, sessionType, tips) | 	m.notificationSender.NotificationWithSessionType(ctx, sendID, recvID, constant.HasReadReceipt, sessionType, tips) | ||||||
| 
 |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -16,11 +16,10 @@ package msg | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
| 	"encoding/base64" | 	"encoding/json" | ||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/apistruct" |  | ||||||
| 	"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/stringutil" | 	"github.com/openimsdk/tools/errs" | ||||||
| 
 | 
 | ||||||
| 	cbapi "github.com/openimsdk/open-im-server/v3/pkg/callbackstruct" | 	cbapi "github.com/openimsdk/open-im-server/v3/pkg/callbackstruct" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||||
| @ -87,19 +86,20 @@ func (m *msgServer) webhookBeforeSendSingleMsg(ctx context.Context, before *conf | |||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) webhookAfterSendSingleMsg(ctx context.Context, after *config.AfterConfig, msg *pbchat.SendMsgReq) { | // Move to msgtransfer | ||||||
| 	if msg.MsgData.ContentType == constant.Typing { | // func (m *msgServer) webhookAfterSendSingleMsg(ctx context.Context, after *config.AfterConfig, msg *pbchat.SendMsgReq) { | ||||||
| 		return | // 	if msg.MsgData.ContentType == constant.Typing { | ||||||
| 	} | // 		return | ||||||
| 	if !filterAfterMsg(msg, after) { | // 	} | ||||||
| 		return | // 	if !filterAfterMsg(msg, after) { | ||||||
| 	} | // 		return | ||||||
| 	cbReq := &cbapi.CallbackAfterSendSingleMsgReq{ | // 	} | ||||||
| 		CommonCallbackReq: toCommonCallback(ctx, msg, cbapi.CallbackAfterSendSingleMsgCommand), | // 	cbReq := &cbapi.CallbackAfterSendSingleMsgReq{ | ||||||
| 		RecvID:            msg.MsgData.RecvID, | // 		CommonCallbackReq: toCommonCallback(ctx, msg, cbapi.CallbackAfterSendSingleMsgCommand), | ||||||
| 	} | // 		RecvID:            msg.MsgData.RecvID, | ||||||
| 	m.webhookClient.AsyncPostWithQuery(ctx, cbReq.GetCallbackCommand(), cbReq, &cbapi.CallbackAfterSendSingleMsgResp{}, after, buildKeyMsgDataQuery(msg.MsgData)) | // 	} | ||||||
| } | // 	m.webhookClient.AsyncPostWithQuery(ctx, cbReq.GetCallbackCommand(), cbReq, &cbapi.CallbackAfterSendSingleMsgResp{}, after, buildKeyMsgDataQuery(msg.MsgData)) | ||||||
|  | // } | ||||||
| 
 | 
 | ||||||
| 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 { | ||||||
| @ -121,26 +121,27 @@ func (m *msgServer) webhookBeforeSendGroupMsg(ctx context.Context, before *confi | |||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) webhookAfterSendGroupMsg(ctx context.Context, after *config.AfterConfig, msg *pbchat.SendMsgReq) { | // Move to msgtransfer | ||||||
| 	if msg.MsgData.ContentType == constant.Typing { | // func (m *msgServer) webhookAfterSendGroupMsg(ctx context.Context, after *config.AfterConfig, msg *pbchat.SendMsgReq) { | ||||||
| 		return | // 	if msg.MsgData.ContentType == constant.Typing { | ||||||
| 	} | // 		return | ||||||
| 	if !filterAfterMsg(msg, after) { | // 	} | ||||||
| 		return | // 	if !filterAfterMsg(msg, after) { | ||||||
| 	} | // 		return | ||||||
| 	cbReq := &cbapi.CallbackAfterSendGroupMsgReq{ | // 	} | ||||||
| 		CommonCallbackReq: toCommonCallback(ctx, msg, cbapi.CallbackAfterSendGroupMsgCommand), | // 	cbReq := &cbapi.CallbackAfterSendGroupMsgReq{ | ||||||
| 		GroupID:           msg.MsgData.GroupID, | // 		CommonCallbackReq: toCommonCallback(ctx, msg, cbapi.CallbackAfterSendGroupMsgCommand), | ||||||
| 	} | // 		GroupID:           msg.MsgData.GroupID, | ||||||
|  | // 	} | ||||||
| 
 | 
 | ||||||
| 	m.webhookClient.AsyncPostWithQuery(ctx, cbReq.GetCallbackCommand(), cbReq, &cbapi.CallbackAfterSendGroupMsgResp{}, after, buildKeyMsgDataQuery(msg.MsgData)) | // 	m.webhookClient.AsyncPostWithQuery(ctx, cbReq.GetCallbackCommand(), cbReq, &cbapi.CallbackAfterSendGroupMsgResp{}, after, buildKeyMsgDataQuery(msg.MsgData)) | ||||||
| } | // } | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) webhookBeforeMsgModify(ctx context.Context, before *config.BeforeConfig, msg *pbchat.SendMsgReq) error { | func (m *msgServer) webhookBeforeMsgModify(ctx context.Context, before *config.BeforeConfig, msg *pbchat.SendMsgReq, beforeMsgData **sdkws.MsgData) error { | ||||||
| 	return webhook.WithCondition(ctx, before, func(ctx context.Context) error { | 	return webhook.WithCondition(ctx, before, func(ctx context.Context) error { | ||||||
| 		if msg.MsgData.ContentType != constant.Text { | 		//if msg.MsgData.ContentType != constant.Text { | ||||||
| 			return nil | 		//	return nil | ||||||
| 		} | 		//} | ||||||
| 		if !filterBeforeMsg(msg, before) { | 		if !filterBeforeMsg(msg, before) { | ||||||
| 			return nil | 			return nil | ||||||
| 		} | 		} | ||||||
| @ -151,9 +152,14 @@ func (m *msgServer) webhookBeforeMsgModify(ctx context.Context, before *config.B | |||||||
| 		if err := m.webhookClient.SyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, resp, before); err != nil { | 		if err := m.webhookClient.SyncPost(ctx, cbReq.GetCallbackCommand(), cbReq, resp, before); err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| 
 | 		if beforeMsgData != nil { | ||||||
|  | 			*beforeMsgData = proto.Clone(msg.MsgData).(*sdkws.MsgData) | ||||||
|  | 		} | ||||||
| 		if resp.Content != nil { | 		if resp.Content != nil { | ||||||
| 			msg.MsgData.Content = []byte(*resp.Content) | 			msg.MsgData.Content = []byte(*resp.Content) | ||||||
|  | 			if err := json.Unmarshal(msg.MsgData.Content, &struct{}{}); err != nil { | ||||||
|  | 				return errs.ErrArgs.WrapMsg("webhook msg modify content is not json", "content", string(msg.MsgData.Content)) | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
| 		datautil.NotNilReplace(msg.MsgData.OfflinePushInfo, resp.OfflinePushInfo) | 		datautil.NotNilReplace(msg.MsgData.OfflinePushInfo, resp.OfflinePushInfo) | ||||||
| 		datautil.NotNilReplace(&msg.MsgData.RecvID, resp.RecvID) | 		datautil.NotNilReplace(&msg.MsgData.RecvID, resp.RecvID) | ||||||
| @ -198,14 +204,14 @@ func (m *msgServer) webhookAfterRevokeMsg(ctx context.Context, after *config.Aft | |||||||
| 	m.webhookClient.AsyncPost(ctx, callbackReq.GetCallbackCommand(), callbackReq, &cbapi.CallbackAfterRevokeMsgResp{}, after) | 	m.webhookClient.AsyncPost(ctx, callbackReq.GetCallbackCommand(), callbackReq, &cbapi.CallbackAfterRevokeMsgResp{}, after) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func buildKeyMsgDataQuery(msg *sdkws.MsgData) map[string]string { | // func buildKeyMsgDataQuery(msg *sdkws.MsgData) map[string]string { | ||||||
| 	keyMsgData := apistruct.KeyMsgData{ | // 	keyMsgData := apistruct.KeyMsgData{ | ||||||
| 		SendID:  msg.SendID, | // 		SendID:  msg.SendID, | ||||||
| 		RecvID:  msg.RecvID, | // 		RecvID:  msg.RecvID, | ||||||
| 		GroupID: msg.GroupID, | // 		GroupID: msg.GroupID, | ||||||
| 	} | // 	} | ||||||
| 
 | 
 | ||||||
| 	return map[string]string{ | // 	return map[string]string{ | ||||||
| 		webhook.Key: base64.StdEncoding.EncodeToString(stringutil.StructToJsonBytes(keyMsgData)), | // 		webhook.Key: base64.StdEncoding.EncodeToString(stringutil.StructToJsonBytes(keyMsgData)), | ||||||
| 	} | // 	} | ||||||
| } | // } | ||||||
|  | |||||||
| @ -2,15 +2,16 @@ package msg | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 	"strings" | ||||||
|  | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||||
| 	"github.com/openimsdk/protocol/msg" | 	"github.com/openimsdk/protocol/msg" | ||||||
| 	"github.com/openimsdk/tools/log" | 	"github.com/openimsdk/tools/log" | ||||||
| 	"strings" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| // DestructMsgs hard delete in Database. | // DestructMsgs hard delete in Database. | ||||||
| func (m *msgServer) DestructMsgs(ctx context.Context, req *msg.DestructMsgsReq) (*msg.DestructMsgsResp, error) { | func (m *msgServer) DestructMsgs(ctx context.Context, req *msg.DestructMsgsReq) (*msg.DestructMsgsResp, error) { | ||||||
| 	if err := authverify.CheckAdmin(ctx, m.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAdmin(ctx); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	docs, err := m.MsgDatabase.GetRandBeforeMsg(ctx, req.Timestamp, int(req.Limit)) | 	docs, err := m.MsgDatabase.GetRandBeforeMsg(ctx, req.Timestamp, int(req.Limit)) | ||||||
|  | |||||||
| @ -42,7 +42,7 @@ func (m *msgServer) validateDeleteSyncOpt(opt *msg.DeleteSyncOpt) (isSyncSelf, i | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) ClearConversationsMsg(ctx context.Context, req *msg.ClearConversationsMsgReq) (*msg.ClearConversationsMsgResp, error) { | func (m *msgServer) ClearConversationsMsg(ctx context.Context, req *msg.ClearConversationsMsgReq) (*msg.ClearConversationsMsgResp, error) { | ||||||
| 	if err := authverify.CheckAccessV3(ctx, req.UserID, m.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	if err := m.clearConversation(ctx, req.ConversationIDs, req.UserID, req.DeleteSyncOpt); err != nil { | 	if err := m.clearConversation(ctx, req.ConversationIDs, req.UserID, req.DeleteSyncOpt); err != nil { | ||||||
| @ -52,7 +52,7 @@ func (m *msgServer) ClearConversationsMsg(ctx context.Context, req *msg.ClearCon | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) UserClearAllMsg(ctx context.Context, req *msg.UserClearAllMsgReq) (*msg.UserClearAllMsgResp, error) { | func (m *msgServer) UserClearAllMsg(ctx context.Context, req *msg.UserClearAllMsgReq) (*msg.UserClearAllMsgResp, error) { | ||||||
| 	if err := authverify.CheckAccessV3(ctx, req.UserID, m.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	conversationIDs, err := m.ConversationLocalCache.GetConversationIDs(ctx, req.UserID) | 	conversationIDs, err := m.ConversationLocalCache.GetConversationIDs(ctx, req.UserID) | ||||||
| @ -66,7 +66,7 @@ func (m *msgServer) UserClearAllMsg(ctx context.Context, req *msg.UserClearAllMs | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) DeleteMsgs(ctx context.Context, req *msg.DeleteMsgsReq) (*msg.DeleteMsgsResp, error) { | func (m *msgServer) DeleteMsgs(ctx context.Context, req *msg.DeleteMsgsReq) (*msg.DeleteMsgsResp, error) { | ||||||
| 	if err := authverify.CheckAccessV3(ctx, req.UserID, m.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	isSyncSelf, isSyncOther := m.validateDeleteSyncOpt(req.DeleteSyncOpt) | 	isSyncSelf, isSyncOther := m.validateDeleteSyncOpt(req.DeleteSyncOpt) | ||||||
| @ -94,6 +94,9 @@ func (m *msgServer) DeleteMsgs(ctx context.Context, req *msg.DeleteMsgsReq) (*ms | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) DeleteMsgPhysicalBySeq(ctx context.Context, req *msg.DeleteMsgPhysicalBySeqReq) (*msg.DeleteMsgPhysicalBySeqResp, error) { | func (m *msgServer) DeleteMsgPhysicalBySeq(ctx context.Context, req *msg.DeleteMsgPhysicalBySeqReq) (*msg.DeleteMsgPhysicalBySeqResp, error) { | ||||||
|  | 	if err := authverify.CheckAdmin(ctx); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	err := m.MsgDatabase.DeleteMsgsPhysicalBySeqs(ctx, req.ConversationID, req.Seqs) | 	err := m.MsgDatabase.DeleteMsgsPhysicalBySeqs(ctx, req.ConversationID, req.Seqs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -102,7 +105,7 @@ func (m *msgServer) DeleteMsgPhysicalBySeq(ctx context.Context, req *msg.DeleteM | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) DeleteMsgPhysical(ctx context.Context, req *msg.DeleteMsgPhysicalReq) (*msg.DeleteMsgPhysicalResp, error) { | func (m *msgServer) DeleteMsgPhysical(ctx context.Context, req *msg.DeleteMsgPhysicalReq) (*msg.DeleteMsgPhysicalResp, error) { | ||||||
| 	if err := authverify.CheckAdmin(ctx, m.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAdmin(ctx); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	remainTime := timeutil.GetCurrentTimestampBySecond() - req.Timestamp | 	remainTime := timeutil.GetCurrentTimestampBySecond() - req.Timestamp | ||||||
|  | |||||||
| @ -48,7 +48,3 @@ func (m *MsgNotificationSender) MarkAsReadNotification(ctx context.Context, conv | |||||||
| 	} | 	} | ||||||
| 	m.NotificationWithSessionType(ctx, sendID, recvID, constant.HasReadReceipt, sessionType, tips) | 	m.NotificationWithSessionType(ctx, sendID, recvID, constant.HasReadReceipt, sessionType, tips) | ||||||
| } | } | ||||||
| 
 |  | ||||||
| func (m *MsgNotificationSender) StreamMsgNotification(ctx context.Context, sendID string, recvID string, sessionType int32, tips *sdkws.StreamMsgTips) { |  | ||||||
| 	m.NotificationWithSessionType(ctx, sendID, recvID, constant.StreamMsgNotification, sessionType, tips) |  | ||||||
| } |  | ||||||
|  | |||||||
| @ -42,7 +42,7 @@ func (m *msgServer) RevokeMsg(ctx context.Context, req *msg.RevokeMsgReq) (*msg. | |||||||
| 	if req.Seq < 0 { | 	if req.Seq < 0 { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("seq is invalid") | 		return nil, errs.ErrArgs.WrapMsg("seq is invalid") | ||||||
| 	} | 	} | ||||||
| 	if err := authverify.CheckAccessV3(ctx, req.UserID, m.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	user, err := m.UserLocalCache.GetUserInfo(ctx, req.UserID) | 	user, err := m.UserLocalCache.GetUserInfo(ctx, req.UserID) | ||||||
| @ -63,11 +63,11 @@ func (m *msgServer) RevokeMsg(ctx context.Context, req *msg.RevokeMsgReq) (*msg. | |||||||
| 	data, _ := json.Marshal(msgs[0]) | 	data, _ := json.Marshal(msgs[0]) | ||||||
| 	log.ZDebug(ctx, "GetMsgBySeqs", "conversationID", req.ConversationID, "seq", req.Seq, "msg", string(data)) | 	log.ZDebug(ctx, "GetMsgBySeqs", "conversationID", req.ConversationID, "seq", req.Seq, "msg", string(data)) | ||||||
| 	var role int32 | 	var role int32 | ||||||
| 	if !authverify.IsAppManagerUid(ctx, m.config.Share.IMAdminUserID) { | 	if !authverify.IsAdmin(ctx) { | ||||||
| 		sessionType := msgs[0].SessionType | 		sessionType := msgs[0].SessionType | ||||||
| 		switch sessionType { | 		switch sessionType { | ||||||
| 		case constant.SingleChatType: | 		case constant.SingleChatType: | ||||||
| 			if err := authverify.CheckAccessV3(ctx, msgs[0].SendID, m.config.Share.IMAdminUserID); err != nil { | 			if err := authverify.CheckAccess(ctx, msgs[0].SendID); err != nil { | ||||||
| 				return nil, err | 				return nil, err | ||||||
| 			} | 			} | ||||||
| 			role = user.AppMangerLevel | 			role = user.AppMangerLevel | ||||||
| @ -109,8 +109,8 @@ func (m *msgServer) RevokeMsg(ctx context.Context, req *msg.RevokeMsgReq) (*msg. | |||||||
| 	revokerUserID := mcontext.GetOpUserID(ctx) | 	revokerUserID := mcontext.GetOpUserID(ctx) | ||||||
| 	var flag bool | 	var flag bool | ||||||
| 
 | 
 | ||||||
| 	if len(m.config.Share.IMAdminUserID) > 0 { | 	if len(m.config.Share.IMAdminUser.UserIDs) > 0 { | ||||||
| 		flag = datautil.Contain(revokerUserID, m.config.Share.IMAdminUserID...) | 		flag = datautil.Contain(revokerUserID, m.adminUserIDs...) | ||||||
| 	} | 	} | ||||||
| 	tips := sdkws.RevokeMsgTips{ | 	tips := sdkws.RevokeMsgTips{ | ||||||
| 		RevokerUserID:  revokerUserID, | 		RevokerUserID:  revokerUserID, | ||||||
|  | |||||||
| @ -17,6 +17,9 @@ package msg | |||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
| 
 | 
 | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||||
|  | 	"google.golang.org/protobuf/proto" | ||||||
|  | 
 | ||||||
| 	"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/msgprocessor" | 	"github.com/openimsdk/open-im-server/v3/pkg/msgprocessor" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/util/conversationutil" | 	"github.com/openimsdk/open-im-server/v3/pkg/util/conversationutil" | ||||||
| @ -32,28 +35,38 @@ import ( | |||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) SendMsg(ctx context.Context, req *pbmsg.SendMsgReq) (*pbmsg.SendMsgResp, error) { | func (m *msgServer) SendMsg(ctx context.Context, req *pbmsg.SendMsgReq) (*pbmsg.SendMsgResp, error) { | ||||||
| 	if req.MsgData != nil { | 	if req.MsgData == nil { | ||||||
| 		m.encapsulateMsgData(req.MsgData) | 		return nil, errs.ErrArgs.WrapMsg("msgData is nil") | ||||||
| 		if req.MsgData.ContentType == constant.Stream { |  | ||||||
| 			if err := m.handlerStreamMsg(ctx, req.MsgData); err != nil { |  | ||||||
| 				return nil, err |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		switch req.MsgData.SessionType { |  | ||||||
| 		case constant.SingleChatType: |  | ||||||
| 			return m.sendMsgSingleChat(ctx, req) |  | ||||||
| 		case constant.NotificationChatType: |  | ||||||
| 			return m.sendMsgNotification(ctx, req) |  | ||||||
| 		case constant.ReadGroupChatType: |  | ||||||
| 			return m.sendMsgGroupChat(ctx, req) |  | ||||||
| 		default: |  | ||||||
| 			return nil, errs.ErrArgs.WrapMsg("unknown sessionType") |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
| 	return nil, errs.ErrArgs.WrapMsg("msgData is nil") | 	if err := authverify.CheckAccess(ctx, req.MsgData.SendID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	before := new(*sdkws.MsgData) | ||||||
|  | 	resp, err := m.sendMsg(ctx, req, before) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	if *before != nil && proto.Equal(*before, req.MsgData) == false { | ||||||
|  | 		resp.Modify = req.MsgData | ||||||
|  | 	} | ||||||
|  | 	return resp, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) sendMsgGroupChat(ctx context.Context, req *pbmsg.SendMsgReq) (resp *pbmsg.SendMsgResp, err error) { | func (m *msgServer) sendMsg(ctx context.Context, req *pbmsg.SendMsgReq, before **sdkws.MsgData) (*pbmsg.SendMsgResp, error) { | ||||||
|  | 	m.encapsulateMsgData(req.MsgData) | ||||||
|  | 	switch req.MsgData.SessionType { | ||||||
|  | 	case constant.SingleChatType: | ||||||
|  | 		return m.sendMsgSingleChat(ctx, req, before) | ||||||
|  | 	case constant.NotificationChatType: | ||||||
|  | 		return m.sendMsgNotification(ctx, req, before) | ||||||
|  | 	case constant.ReadGroupChatType: | ||||||
|  | 		return m.sendMsgGroupChat(ctx, req, before) | ||||||
|  | 	default: | ||||||
|  | 		return nil, errs.ErrArgs.WrapMsg("unknown sessionType") | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (m *msgServer) sendMsgGroupChat(ctx context.Context, req *pbmsg.SendMsgReq, before **sdkws.MsgData) (resp *pbmsg.SendMsgResp, err error) { | ||||||
| 	if err = m.messageVerification(ctx, req); err != nil { | 	if err = m.messageVerification(ctx, req); err != nil { | ||||||
| 		prommetrics.GroupChatMsgProcessFailedCounter.Inc() | 		prommetrics.GroupChatMsgProcessFailedCounter.Inc() | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -62,7 +75,7 @@ func (m *msgServer) sendMsgGroupChat(ctx context.Context, req *pbmsg.SendMsgReq) | |||||||
| 	if err = m.webhookBeforeSendGroupMsg(ctx, &m.config.WebhooksConfig.BeforeSendGroupMsg, req); err != nil { | 	if err = m.webhookBeforeSendGroupMsg(ctx, &m.config.WebhooksConfig.BeforeSendGroupMsg, req); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	if err := m.webhookBeforeMsgModify(ctx, &m.config.WebhooksConfig.BeforeMsgModify, req); err != nil { | 	if err := m.webhookBeforeMsgModify(ctx, &m.config.WebhooksConfig.BeforeMsgModify, req, before); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	err = m.MsgDatabase.MsgToMQ(ctx, conversationutil.GenConversationUniqueKeyForGroup(req.MsgData.GroupID), req.MsgData) | 	err = m.MsgDatabase.MsgToMQ(ctx, conversationutil.GenConversationUniqueKeyForGroup(req.MsgData.GroupID), req.MsgData) | ||||||
| @ -73,7 +86,8 @@ func (m *msgServer) sendMsgGroupChat(ctx context.Context, req *pbmsg.SendMsgReq) | |||||||
| 		go m.setConversationAtInfo(ctx, req.MsgData) | 		go m.setConversationAtInfo(ctx, req.MsgData) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	m.webhookAfterSendGroupMsg(ctx, &m.config.WebhooksConfig.AfterSendGroupMsg, req) | 	// m.webhookAfterSendGroupMsg(ctx, &m.config.WebhooksConfig.AfterSendGroupMsg, req) | ||||||
|  | 
 | ||||||
| 	prommetrics.GroupChatMsgProcessSuccessCounter.Inc() | 	prommetrics.GroupChatMsgProcessSuccessCounter.Inc() | ||||||
| 	resp = &pbmsg.SendMsgResp{} | 	resp = &pbmsg.SendMsgResp{} | ||||||
| 	resp.SendTime = req.MsgData.SendTime | 	resp.SendTime = req.MsgData.SendTime | ||||||
| @ -144,7 +158,7 @@ func (m *msgServer) setConversationAtInfo(nctx context.Context, msg *sdkws.MsgDa | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) sendMsgNotification(ctx context.Context, req *pbmsg.SendMsgReq) (resp *pbmsg.SendMsgResp, err error) { | func (m *msgServer) sendMsgNotification(ctx context.Context, req *pbmsg.SendMsgReq, before **sdkws.MsgData) (resp *pbmsg.SendMsgResp, err error) { | ||||||
| 	if err := m.MsgDatabase.MsgToMQ(ctx, conversationutil.GenConversationUniqueKeyForSingle(req.MsgData.SendID, req.MsgData.RecvID), req.MsgData); err != nil { | 	if err := m.MsgDatabase.MsgToMQ(ctx, conversationutil.GenConversationUniqueKeyForSingle(req.MsgData.SendID, req.MsgData.RecvID), req.MsgData); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -156,37 +170,31 @@ func (m *msgServer) sendMsgNotification(ctx context.Context, req *pbmsg.SendMsgR | |||||||
| 	return resp, nil | 	return resp, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) sendMsgSingleChat(ctx context.Context, req *pbmsg.SendMsgReq) (resp *pbmsg.SendMsgResp, err error) { | func (m *msgServer) sendMsgSingleChat(ctx context.Context, req *pbmsg.SendMsgReq, before **sdkws.MsgData) (resp *pbmsg.SendMsgResp, err error) { | ||||||
| 	if err := m.messageVerification(ctx, req); err != nil { | 	if err := m.messageVerification(ctx, req); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	isSend := true | 	isSend := true | ||||||
| 	isNotification := msgprocessor.IsNotificationByMsg(req.MsgData) | 	isNotification := msgprocessor.IsNotificationByMsg(req.MsgData) | ||||||
| 	if !isNotification { | 	if !isNotification { | ||||||
| 		isSend, err = m.modifyMessageByUserMessageReceiveOpt( | 		isSend, err = m.modifyMessageByUserMessageReceiveOpt(authverify.WithTempAdmin(ctx), req.MsgData.RecvID, conversationutil.GenConversationIDForSingle(req.MsgData.SendID, req.MsgData.RecvID), constant.SingleChatType, req) | ||||||
| 			ctx, |  | ||||||
| 			req.MsgData.RecvID, |  | ||||||
| 			conversationutil.GenConversationIDForSingle(req.MsgData.SendID, req.MsgData.RecvID), |  | ||||||
| 			constant.SingleChatType, |  | ||||||
| 			req, |  | ||||||
| 		) |  | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return nil, err | 			return nil, err | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	if !isSend { | 	if !isSend { | ||||||
| 		prommetrics.SingleChatMsgProcessFailedCounter.Inc() | 		prommetrics.SingleChatMsgProcessFailedCounter.Inc() | ||||||
| 		return nil, nil | 		return nil, errs.ErrArgs.WrapMsg("message is not sent") | ||||||
| 	} else { | 	} else { | ||||||
| 		if err := m.webhookBeforeMsgModify(ctx, &m.config.WebhooksConfig.BeforeMsgModify, req); err != nil { | 		if err := m.webhookBeforeMsgModify(ctx, &m.config.WebhooksConfig.BeforeMsgModify, req, before); err != nil { | ||||||
| 			return nil, err | 			return nil, err | ||||||
| 		} | 		} | ||||||
| 
 |  | ||||||
| 		if err := m.MsgDatabase.MsgToMQ(ctx, conversationutil.GenConversationUniqueKeyForSingle(req.MsgData.SendID, req.MsgData.RecvID), req.MsgData); err != nil { | 		if err := m.MsgDatabase.MsgToMQ(ctx, conversationutil.GenConversationUniqueKeyForSingle(req.MsgData.SendID, req.MsgData.RecvID), req.MsgData); err != nil { | ||||||
| 			prommetrics.SingleChatMsgProcessFailedCounter.Inc() | 			prommetrics.SingleChatMsgProcessFailedCounter.Inc() | ||||||
| 			return nil, err | 			return nil, err | ||||||
| 		} | 		} | ||||||
| 		m.webhookAfterSendSingleMsg(ctx, &m.config.WebhooksConfig.AfterSendSingleMsg, req) | 
 | ||||||
|  | 		// m.webhookAfterSendSingleMsg(ctx, &m.config.WebhooksConfig.AfterSendSingleMsg, req) | ||||||
| 		prommetrics.SingleChatMsgProcessSuccessCounter.Inc() | 		prommetrics.SingleChatMsgProcessSuccessCounter.Inc() | ||||||
| 		return &pbmsg.SendMsgResp{ | 		return &pbmsg.SendMsgResp{ | ||||||
| 			ServerMsgID: req.MsgData.ServerMsgID, | 			ServerMsgID: req.MsgData.ServerMsgID, | ||||||
| @ -195,3 +203,25 @@ func (m *msgServer) sendMsgSingleChat(ctx context.Context, req *pbmsg.SendMsgReq | |||||||
| 		}, nil | 		}, nil | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | func (m *msgServer) SendSimpleMsg(ctx context.Context, req *pbmsg.SendSimpleMsgReq) (*pbmsg.SendSimpleMsgResp, error) { | ||||||
|  | 	if req.MsgData == nil { | ||||||
|  | 		return nil, errs.ErrArgs.WrapMsg("msg data is nil") | ||||||
|  | 	} | ||||||
|  | 	sender, err := m.UserLocalCache.GetUserInfo(ctx, req.MsgData.SendID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	req.MsgData.SenderFaceURL = sender.FaceURL | ||||||
|  | 	req.MsgData.SenderNickname = sender.Nickname | ||||||
|  | 	resp, err := m.SendMsg(ctx, &pbmsg.SendMsgReq{MsgData: req.MsgData}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	return &pbmsg.SendSimpleMsgResp{ | ||||||
|  | 		ServerMsgID: resp.ServerMsgID, | ||||||
|  | 		ClientMsgID: resp.ClientMsgID, | ||||||
|  | 		SendTime:    resp.SendTime, | ||||||
|  | 		Modify:      resp.Modify, | ||||||
|  | 	}, nil | ||||||
|  | } | ||||||
|  | |||||||
| @ -17,9 +17,10 @@ package msg | |||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
| 	"errors" | 	"errors" | ||||||
|  | 	"sort" | ||||||
|  | 
 | ||||||
| 	pbmsg "github.com/openimsdk/protocol/msg" | 	pbmsg "github.com/openimsdk/protocol/msg" | ||||||
| 	"github.com/redis/go-redis/v9" | 	"github.com/redis/go-redis/v9" | ||||||
| 	"sort" |  | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| 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) { | ||||||
|  | |||||||
| @ -22,7 +22,6 @@ import ( | |||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/dbbuild" | 	"github.com/openimsdk/open-im-server/v3/pkg/dbbuild" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/mqbuild" | 	"github.com/openimsdk/open-im-server/v3/pkg/mqbuild" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | ||||||
| 
 |  | ||||||
| 	"google.golang.org/grpc" | 	"google.golang.org/grpc" | ||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/config" | ||||||
| @ -59,9 +58,8 @@ type Config struct { | |||||||
| // MsgServer encapsulates dependencies required for message handling. | // MsgServer encapsulates dependencies required for message handling. | ||||||
| type msgServer struct { | type msgServer struct { | ||||||
| 	msg.UnimplementedMsgServer | 	msg.UnimplementedMsgServer | ||||||
| 	RegisterCenter         discovery.Conn               // Service discovery registry for service registration. | 	RegisterCenter         discovery.Conn                   // Service discovery registry for service registration. | ||||||
| 	MsgDatabase            controller.CommonMsgDatabase // Interface for message database operations. | 	MsgDatabase            controller.CommonMsgDatabase     // Interface for message database operations. | ||||||
| 	StreamMsgDatabase      controller.StreamMsgDatabase |  | ||||||
| 	UserLocalCache         *rpccache.UserLocalCache         // Local cache for user data. | 	UserLocalCache         *rpccache.UserLocalCache         // Local cache for user data. | ||||||
| 	FriendLocalCache       *rpccache.FriendLocalCache       // Local cache for friend data. | 	FriendLocalCache       *rpccache.FriendLocalCache       // Local cache for friend data. | ||||||
| 	GroupLocalCache        *rpccache.GroupLocalCache        // Local cache for group data. | 	GroupLocalCache        *rpccache.GroupLocalCache        // Local cache for group data. | ||||||
| @ -72,6 +70,8 @@ type msgServer struct { | |||||||
| 	config                 *Config                          // Global configuration settings. | 	config                 *Config                          // Global configuration settings. | ||||||
| 	webhookClient          *webhook.Client | 	webhookClient          *webhook.Client | ||||||
| 	conversationClient     *rpcli.ConversationClient | 	conversationClient     *rpcli.ConversationClient | ||||||
|  | 
 | ||||||
|  | 	adminUserIDs []string | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) addInterceptorHandler(interceptorFunc ...MessageInterceptorFunc) { | func (m *msgServer) addInterceptorHandler(interceptorFunc ...MessageInterceptorFunc) { | ||||||
| @ -79,7 +79,7 @@ func (m *msgServer) addInterceptorHandler(interceptorFunc ...MessageInterceptorF | |||||||
| 
 | 
 | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func Start(ctx context.Context, config *Config, client discovery.Conn, server grpc.ServiceRegistrar) error { | func Start(ctx context.Context, config *Config, client discovery.SvcDiscoveryRegistry, server grpc.ServiceRegistrar) error { | ||||||
| 	builder := mqbuild.NewBuilder(&config.KafkaConfig) | 	builder := mqbuild.NewBuilder(&config.KafkaConfig) | ||||||
| 	redisProducer, err := builder.GetTopicProducer(ctx, config.KafkaConfig.ToRedisTopic) | 	redisProducer, err := builder.GetTopicProducer(ctx, config.KafkaConfig.ToRedisTopic) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -117,10 +117,6 @@ func Start(ctx context.Context, config *Config, client discovery.Conn, server gr | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	streamMsg, err := mgo.NewStreamMsgMongo(mgocli.GetDB()) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	seqUserCache := redis.NewSeqUserCacheRedis(rdb, seqUser) | 	seqUserCache := redis.NewSeqUserCacheRedis(rdb, seqUser) | ||||||
| 	userConn, err := client.GetConn(ctx, config.Discovery.RpcService.User) | 	userConn, err := client.GetConn(ctx, config.Discovery.RpcService.User) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -142,7 +138,6 @@ func Start(ctx context.Context, config *Config, client discovery.Conn, server gr | |||||||
| 	msgDatabase := controller.NewCommonMsgDatabase(msgDocModel, msgModel, seqUserCache, seqConversationCache, redisProducer) | 	msgDatabase := controller.NewCommonMsgDatabase(msgDocModel, msgModel, seqUserCache, seqConversationCache, redisProducer) | ||||||
| 	s := &msgServer{ | 	s := &msgServer{ | ||||||
| 		MsgDatabase:            msgDatabase, | 		MsgDatabase:            msgDatabase, | ||||||
| 		StreamMsgDatabase:      controller.NewStreamMsgDatabase(streamMsg), |  | ||||||
| 		RegisterCenter:         client, | 		RegisterCenter:         client, | ||||||
| 		UserLocalCache:         rpccache.NewUserLocalCache(rpcli.NewUserClient(userConn), &config.LocalCacheConfig, rdb), | 		UserLocalCache:         rpccache.NewUserLocalCache(rpcli.NewUserClient(userConn), &config.LocalCacheConfig, rdb), | ||||||
| 		GroupLocalCache:        rpccache.NewGroupLocalCache(rpcli.NewGroupClient(groupConn), &config.LocalCacheConfig, rdb), | 		GroupLocalCache:        rpccache.NewGroupLocalCache(rpcli.NewGroupClient(groupConn), &config.LocalCacheConfig, rdb), | ||||||
| @ -151,6 +146,7 @@ func Start(ctx context.Context, config *Config, client discovery.Conn, server gr | |||||||
| 		config:                 config, | 		config:                 config, | ||||||
| 		webhookClient:          webhook.NewWebhookClient(config.WebhooksConfig.URL), | 		webhookClient:          webhook.NewWebhookClient(config.WebhooksConfig.URL), | ||||||
| 		conversationClient:     conversationClient, | 		conversationClient:     conversationClient, | ||||||
|  | 		adminUserIDs:           config.Share.IMAdminUser.UserIDs, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	s.notificationSender = notification.NewNotificationSender(&config.NotificationConfig, notification.WithLocalSendMsg(s.SendMsg)) | 	s.notificationSender = notification.NewNotificationSender(&config.NotificationConfig, notification.WithLocalSendMsg(s.SendMsg)) | ||||||
|  | |||||||
| @ -16,15 +16,20 @@ package msg | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" |  | ||||||
| 	"time" | 	"time" | ||||||
| 
 | 
 | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||||
|  | 
 | ||||||
| 	"github.com/openimsdk/protocol/msg" | 	"github.com/openimsdk/protocol/msg" | ||||||
| 	"github.com/openimsdk/protocol/sdkws" | 	"github.com/openimsdk/protocol/sdkws" | ||||||
| 	"github.com/openimsdk/tools/utils/datautil" | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) GetActiveUser(ctx context.Context, req *msg.GetActiveUserReq) (*msg.GetActiveUserResp, error) { | func (m *msgServer) GetActiveUser(ctx context.Context, req *msg.GetActiveUserReq) (*msg.GetActiveUserResp, error) { | ||||||
|  | 	if err := authverify.CheckAdmin(ctx); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	msgCount, userCount, users, dateCount, err := m.MsgDatabase.RangeUserSendCount(ctx, time.UnixMilli(req.Start), time.UnixMilli(req.End), req.Group, req.Ase, req.Pagination.PageNumber, req.Pagination.ShowNumber) | 	msgCount, userCount, users, dateCount, err := m.MsgDatabase.RangeUserSendCount(ctx, time.UnixMilli(req.Start), time.UnixMilli(req.End), req.Group, req.Ase, req.Pagination.PageNumber, req.Pagination.ShowNumber) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -60,6 +65,9 @@ func (m *msgServer) GetActiveUser(ctx context.Context, req *msg.GetActiveUserReq | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (m *msgServer) GetActiveGroup(ctx context.Context, req *msg.GetActiveGroupReq) (*msg.GetActiveGroupResp, error) { | func (m *msgServer) GetActiveGroup(ctx context.Context, req *msg.GetActiveGroupReq) (*msg.GetActiveGroupResp, error) { | ||||||
|  | 	if err := authverify.CheckAdmin(ctx); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	msgCount, groupCount, groups, dateCount, err := m.MsgDatabase.RangeGroupSendCount(ctx, time.UnixMilli(req.Start), time.UnixMilli(req.End), req.Ase, req.Pagination.PageNumber, req.Pagination.ShowNumber) | 	msgCount, groupCount, groups, dateCount, err := m.MsgDatabase.RangeGroupSendCount(ctx, time.UnixMilli(req.Start), time.UnixMilli(req.End), req.Ase, req.Pagination.PageNumber, req.Pagination.ShowNumber) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
|  | |||||||
| @ -1,115 +0,0 @@ | |||||||
| package msg |  | ||||||
| 
 |  | ||||||
| import ( |  | ||||||
| 	"context" |  | ||||||
| 	"fmt" |  | ||||||
| 	"time" |  | ||||||
| 
 |  | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" |  | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/msgprocessor" |  | ||||||
| 	"github.com/openimsdk/protocol/constant" |  | ||||||
| 	"github.com/openimsdk/protocol/msg" |  | ||||||
| 	"github.com/openimsdk/protocol/sdkws" |  | ||||||
| 	"github.com/openimsdk/tools/errs" |  | ||||||
| ) |  | ||||||
| 
 |  | ||||||
| const StreamDeadlineTime = time.Second * 60 * 10 |  | ||||||
| 
 |  | ||||||
| func (m *msgServer) handlerStreamMsg(ctx context.Context, msgData *sdkws.MsgData) error { |  | ||||||
| 	now := time.Now() |  | ||||||
| 	val := &model.StreamMsg{ |  | ||||||
| 		ClientMsgID:    msgData.ClientMsgID, |  | ||||||
| 		ConversationID: msgprocessor.GetConversationIDByMsg(msgData), |  | ||||||
| 		UserID:         msgData.SendID, |  | ||||||
| 		CreateTime:     now, |  | ||||||
| 		DeadlineTime:   now.Add(StreamDeadlineTime), |  | ||||||
| 	} |  | ||||||
| 	return m.StreamMsgDatabase.CreateStreamMsg(ctx, val) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (m *msgServer) getStreamMsg(ctx context.Context, clientMsgID string) (*model.StreamMsg, error) { |  | ||||||
| 	res, err := m.StreamMsgDatabase.GetStreamMsg(ctx, clientMsgID) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	now := time.Now() |  | ||||||
| 	if !res.End && res.DeadlineTime.Before(now) { |  | ||||||
| 		res.End = true |  | ||||||
| 		res.DeadlineTime = now |  | ||||||
| 		_ = m.StreamMsgDatabase.AppendStreamMsg(ctx, res.ClientMsgID, 0, nil, true, now) |  | ||||||
| 	} |  | ||||||
| 	return res, nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (m *msgServer) AppendStreamMsg(ctx context.Context, req *msg.AppendStreamMsgReq) (*msg.AppendStreamMsgResp, error) { |  | ||||||
| 	res, err := m.getStreamMsg(ctx, req.ClientMsgID) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	if res.End { |  | ||||||
| 		return nil, errs.ErrNoPermission.WrapMsg("stream msg is end") |  | ||||||
| 	} |  | ||||||
| 	if len(res.Packets) < int(req.StartIndex) { |  | ||||||
| 		return nil, errs.ErrNoPermission.WrapMsg("start index is invalid") |  | ||||||
| 	} |  | ||||||
| 	if val := len(res.Packets) - int(req.StartIndex); val > 0 { |  | ||||||
| 		exist := res.Packets[int(req.StartIndex):] |  | ||||||
| 		for i, s := range exist { |  | ||||||
| 			if len(req.Packets) == 0 { |  | ||||||
| 				break |  | ||||||
| 			} |  | ||||||
| 			if s != req.Packets[i] { |  | ||||||
| 				return nil, errs.ErrNoPermission.WrapMsg(fmt.Sprintf("packet %d has been written and is inconsistent", i)) |  | ||||||
| 			} |  | ||||||
| 			req.StartIndex++ |  | ||||||
| 			req.Packets = req.Packets[1:] |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	if len(req.Packets) == 0 && res.End == req.End { |  | ||||||
| 		return &msg.AppendStreamMsgResp{}, nil |  | ||||||
| 	} |  | ||||||
| 	deadlineTime := time.Now().Add(StreamDeadlineTime) |  | ||||||
| 	if err := m.StreamMsgDatabase.AppendStreamMsg(ctx, req.ClientMsgID, int(req.StartIndex), req.Packets, req.End, deadlineTime); err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	conversation, err := m.conversationClient.GetConversation(ctx, res.ConversationID, res.UserID) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	tips := &sdkws.StreamMsgTips{ |  | ||||||
| 		ConversationID: res.ConversationID, |  | ||||||
| 		ClientMsgID:    res.ClientMsgID, |  | ||||||
| 		StartIndex:     req.StartIndex, |  | ||||||
| 		Packets:        req.Packets, |  | ||||||
| 		End:            req.End, |  | ||||||
| 	} |  | ||||||
| 	var ( |  | ||||||
| 		recvID      string |  | ||||||
| 		sessionType int32 |  | ||||||
| 	) |  | ||||||
| 	if conversation.GroupID == "" { |  | ||||||
| 		sessionType = constant.SingleChatType |  | ||||||
| 		recvID = conversation.UserID |  | ||||||
| 	} else { |  | ||||||
| 		sessionType = constant.ReadGroupChatType |  | ||||||
| 		recvID = conversation.GroupID |  | ||||||
| 	} |  | ||||||
| 	m.msgNotificationSender.StreamMsgNotification(ctx, res.UserID, recvID, sessionType, tips) |  | ||||||
| 	return &msg.AppendStreamMsgResp{}, nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func (m *msgServer) GetStreamMsg(ctx context.Context, req *msg.GetStreamMsgReq) (*msg.GetStreamMsgResp, error) { |  | ||||||
| 	res, err := m.getStreamMsg(ctx, req.ClientMsgID) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	return &msg.GetStreamMsgResp{ |  | ||||||
| 		ClientMsgID:    res.ClientMsgID, |  | ||||||
| 		ConversationID: res.ConversationID, |  | ||||||
| 		UserID:         res.UserID, |  | ||||||
| 		Packets:        res.Packets, |  | ||||||
| 		End:            res.End, |  | ||||||
| 		CreateTime:     res.CreateTime.UnixMilli(), |  | ||||||
| 		DeadlineTime:   res.DeadlineTime.UnixMilli(), |  | ||||||
| 	}, nil |  | ||||||
| } |  | ||||||
| @ -29,6 +29,9 @@ import ( | |||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| 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) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	resp := &sdkws.PullMessageBySeqsResp{} | 	resp := &sdkws.PullMessageBySeqsResp{} | ||||||
| 	resp.Msgs = make(map[string]*sdkws.PullMsgs) | 	resp.Msgs = make(map[string]*sdkws.PullMsgs) | ||||||
| 	resp.NotificationMsgs = make(map[string]*sdkws.PullMsgs) | 	resp.NotificationMsgs = make(map[string]*sdkws.PullMsgs) | ||||||
| @ -118,7 +121,7 @@ func (m *msgServer) GetSeqMessage(ctx context.Context, req *msg.GetSeqMessageReq | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 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.CheckAccess(ctx, req.UserID); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	conversationIDs, err := m.ConversationLocalCache.GetConversationIDs(ctx, req.UserID) | 	conversationIDs, err := m.ConversationLocalCache.GetConversationIDs(ctx, req.UserID) | ||||||
|  | |||||||
| @ -20,6 +20,7 @@ import ( | |||||||
| 	"strconv" | 	"strconv" | ||||||
| 	"time" | 	"time" | ||||||
| 
 | 
 | ||||||
|  | 	"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/tools/utils/datautil" | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
| 	"github.com/openimsdk/tools/utils/encrypt" | 	"github.com/openimsdk/tools/utils/encrypt" | ||||||
| @ -53,7 +54,7 @@ type MessageRevoked struct { | |||||||
| func (m *msgServer) messageVerification(ctx context.Context, data *msg.SendMsgReq) error { | func (m *msgServer) messageVerification(ctx context.Context, data *msg.SendMsgReq) error { | ||||||
| 	switch data.MsgData.SessionType { | 	switch data.MsgData.SessionType { | ||||||
| 	case constant.SingleChatType: | 	case constant.SingleChatType: | ||||||
| 		if datautil.Contain(data.MsgData.SendID, m.config.Share.IMAdminUserID...) { | 		if datautil.Contain(data.MsgData.SendID, m.adminUserIDs...) { | ||||||
| 			return nil | 			return nil | ||||||
| 		} | 		} | ||||||
| 		if data.MsgData.ContentType <= constant.NotificationEnd && | 		if data.MsgData.ContentType <= constant.NotificationEnd && | ||||||
| @ -63,6 +64,13 @@ func (m *msgServer) messageVerification(ctx context.Context, data *msg.SendMsgRe | |||||||
| 		if err := m.webhookBeforeSendSingleMsg(ctx, &m.config.WebhooksConfig.BeforeSendSingleMsg, data); err != nil { | 		if err := m.webhookBeforeSendSingleMsg(ctx, &m.config.WebhooksConfig.BeforeSendSingleMsg, data); err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
|  | 		u, err := m.UserLocalCache.GetUserInfo(ctx, data.MsgData.SendID) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 		if authverify.CheckSystemAccount(ctx, u.AppMangerLevel) { | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
| 		black, err := m.FriendLocalCache.IsBlack(ctx, data.MsgData.SendID, data.MsgData.RecvID) | 		black, err := m.FriendLocalCache.IsBlack(ctx, data.MsgData.SendID, data.MsgData.RecvID) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return err | ||||||
| @ -94,7 +102,7 @@ func (m *msgServer) messageVerification(ctx context.Context, data *msg.SendMsgRe | |||||||
| 			return nil | 			return nil | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		if datautil.Contain(data.MsgData.SendID, m.config.Share.IMAdminUserID...) { | 		if datautil.Contain(data.MsgData.SendID, m.adminUserIDs...) { | ||||||
| 			return nil | 			return nil | ||||||
| 		} | 		} | ||||||
| 		if data.MsgData.ContentType <= constant.NotificationEnd && | 		if data.MsgData.ContentType <= constant.NotificationEnd && | ||||||
|  | |||||||
| @ -29,10 +29,9 @@ import ( | |||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| 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 := authverify.CheckAccessV3(ctx, req.UserID, s.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAccess(ctx, req.UserID); 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 | ||||||
| @ -47,6 +46,9 @@ func (s *friendServer) GetPaginationBlacks(ctx context.Context, req *relation.Ge | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *friendServer) IsBlack(ctx context.Context, req *relation.IsBlackReq) (*relation.IsBlackResp, error) { | func (s *friendServer) IsBlack(ctx context.Context, req *relation.IsBlackReq) (*relation.IsBlackResp, error) { | ||||||
|  | 	if err := authverify.CheckAccessIn(ctx, req.UserID1, req.UserID2); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	in1, in2, err := s.blackDatabase.CheckIn(ctx, req.UserID1, req.UserID2) | 	in1, in2, err := s.blackDatabase.CheckIn(ctx, req.UserID1, req.UserID2) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -58,7 +60,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 := authverify.CheckAccessV3(ctx, req.OwnerUserID, s.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAccess(ctx, req.OwnerUserID); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -73,7 +75,7 @@ func (s *friendServer) RemoveBlack(ctx context.Context, req *relation.RemoveBlac | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *friendServer) AddBlack(ctx context.Context, req *relation.AddBlackReq) (*relation.AddBlackResp, error) { | func (s *friendServer) AddBlack(ctx context.Context, req *relation.AddBlackReq) (*relation.AddBlackResp, error) { | ||||||
| 	if err := authverify.CheckAccessV3(ctx, req.OwnerUserID, s.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAccess(ctx, req.OwnerUserID); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -99,7 +101,7 @@ func (s *friendServer) AddBlack(ctx context.Context, req *relation.AddBlackReq) | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *friendServer) GetSpecifiedBlacks(ctx context.Context, req *relation.GetSpecifiedBlacksReq) (*relation.GetSpecifiedBlacksResp, error) { | 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 { | 	if err := authverify.CheckAccess(ctx, req.OwnerUserID); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -18,6 +18,7 @@ import ( | |||||||
| 	"context" | 	"context" | ||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/dbbuild" | 	"github.com/openimsdk/open-im-server/v3/pkg/dbbuild" | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/notification/common_user" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | ||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/tools/mq/memamq" | 	"github.com/openimsdk/tools/mq/memamq" | ||||||
| @ -65,7 +66,7 @@ type Config struct { | |||||||
| 	Discovery          config.Discovery | 	Discovery          config.Discovery | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func Start(ctx context.Context, config *Config, client discovery.Conn, server grpc.ServiceRegistrar) error { | func Start(ctx context.Context, config *Config, client discovery.SvcDiscoveryRegistry, server grpc.ServiceRegistrar) error { | ||||||
| 	dbb := dbbuild.NewBuilder(&config.MongodbConfig, &config.RedisConfig) | 	dbb := dbbuild.NewBuilder(&config.MongodbConfig, &config.RedisConfig) | ||||||
| 	mgocli, err := dbb.Mongo(ctx) | 	mgocli, err := dbb.Mongo(ctx) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -100,23 +101,24 @@ func Start(ctx context.Context, config *Config, client discovery.Conn, server gr | |||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	userClient := rpcli.NewUserClient(userConn) | 	userClient := rpcli.NewUserClient(userConn) | ||||||
| 
 | 	database := controller.NewFriendDatabase( | ||||||
|  | 		friendMongoDB, | ||||||
|  | 		friendRequestMongoDB, | ||||||
|  | 		redis.NewFriendCacheRedis(rdb, &config.LocalCacheConfig, friendMongoDB), | ||||||
|  | 		mgocli.GetTx(), | ||||||
|  | 	) | ||||||
| 	// Initialize notification sender | 	// Initialize notification sender | ||||||
| 	notificationSender := NewFriendNotificationSender( | 	notificationSender := NewFriendNotificationSender( | ||||||
| 		&config.NotificationConfig, | 		&config.NotificationConfig, | ||||||
| 		rpcli.NewMsgClient(msgConn), | 		rpcli.NewMsgClient(msgConn), | ||||||
| 		WithRpcFunc(userClient.GetUsersInfo), | 		WithRpcFunc(userClient.GetUsersInfo), | ||||||
|  | 		WithFriendDB(database), | ||||||
| 	) | 	) | ||||||
| 	localcache.InitLocalCache(&config.LocalCacheConfig) | 	localcache.InitLocalCache(&config.LocalCacheConfig) | ||||||
| 
 | 
 | ||||||
| 	// Register Friend server with refactored MongoDB and Redis integrations | 	// Register Friend server with refactored MongoDB and Redis integrations | ||||||
| 	relation.RegisterFriendServer(server, &friendServer{ | 	relation.RegisterFriendServer(server, &friendServer{ | ||||||
| 		db: controller.NewFriendDatabase( | 		db: database, | ||||||
| 			friendMongoDB, |  | ||||||
| 			friendRequestMongoDB, |  | ||||||
| 			redis.NewFriendCacheRedis(rdb, &config.LocalCacheConfig, friendMongoDB), |  | ||||||
| 			mgocli.GetTx(), |  | ||||||
| 		), |  | ||||||
| 		blackDatabase: controller.NewBlackDatabase( | 		blackDatabase: controller.NewBlackDatabase( | ||||||
| 			blackMongoDB, | 			blackMongoDB, | ||||||
| 			redis.NewBlackCacheRedis(rdb, &config.LocalCacheConfig, blackMongoDB), | 			redis.NewBlackCacheRedis(rdb, &config.LocalCacheConfig, blackMongoDB), | ||||||
| @ -134,7 +136,7 @@ func Start(ctx context.Context, config *Config, client discovery.Conn, server gr | |||||||
| // ok. | // ok. | ||||||
| func (s *friendServer) ApplyToAddFriend(ctx context.Context, req *relation.ApplyToAddFriendReq) (resp *relation.ApplyToAddFriendResp, err error) { | func (s *friendServer) ApplyToAddFriend(ctx context.Context, req *relation.ApplyToAddFriendReq) (resp *relation.ApplyToAddFriendResp, err error) { | ||||||
| 	resp = &relation.ApplyToAddFriendResp{} | 	resp = &relation.ApplyToAddFriendResp{} | ||||||
| 	if err := authverify.CheckAccessV3(ctx, req.FromUserID, s.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAccess(ctx, req.FromUserID); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	if req.ToUserID == req.FromUserID { | 	if req.ToUserID == req.FromUserID { | ||||||
| @ -164,7 +166,7 @@ func (s *friendServer) ApplyToAddFriend(ctx context.Context, req *relation.Apply | |||||||
| 
 | 
 | ||||||
| // ok. | // ok. | ||||||
| func (s *friendServer) ImportFriends(ctx context.Context, req *relation.ImportFriendReq) (resp *relation.ImportFriendResp, err error) { | func (s *friendServer) ImportFriends(ctx context.Context, req *relation.ImportFriendReq) (resp *relation.ImportFriendResp, err error) { | ||||||
| 	if err := authverify.CheckAdmin(ctx, s.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAdmin(ctx); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -190,7 +192,7 @@ func (s *friendServer) ImportFriends(ctx context.Context, req *relation.ImportFr | |||||||
| 			FromUserID:   req.OwnerUserID, | 			FromUserID:   req.OwnerUserID, | ||||||
| 			ToUserID:     userID, | 			ToUserID:     userID, | ||||||
| 			HandleResult: constant.FriendResponseAgree, | 			HandleResult: constant.FriendResponseAgree, | ||||||
| 		}) | 		}, false) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	s.webhookAfterImportFriends(ctx, &s.config.WebhooksConfig.AfterImportFriends, req) | 	s.webhookAfterImportFriends(ctx, &s.config.WebhooksConfig.AfterImportFriends, req) | ||||||
| @ -200,7 +202,7 @@ func (s *friendServer) ImportFriends(ctx context.Context, req *relation.ImportFr | |||||||
| // ok. | // ok. | ||||||
| func (s *friendServer) RespondFriendApply(ctx context.Context, req *relation.RespondFriendApplyReq) (resp *relation.RespondFriendApplyResp, err error) { | func (s *friendServer) RespondFriendApply(ctx context.Context, req *relation.RespondFriendApplyReq) (resp *relation.RespondFriendApplyResp, err error) { | ||||||
| 	resp = &relation.RespondFriendApplyResp{} | 	resp = &relation.RespondFriendApplyResp{} | ||||||
| 	if err := authverify.CheckAccessV3(ctx, req.ToUserID, s.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAccess(ctx, req.ToUserID); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -219,7 +221,7 @@ func (s *friendServer) RespondFriendApply(ctx context.Context, req *relation.Res | |||||||
| 			return nil, err | 			return nil, err | ||||||
| 		} | 		} | ||||||
| 		s.webhookAfterAddFriendAgree(ctx, &s.config.WebhooksConfig.AfterAddFriendAgree, req) | 		s.webhookAfterAddFriendAgree(ctx, &s.config.WebhooksConfig.AfterAddFriendAgree, req) | ||||||
| 		s.notificationSender.FriendApplicationAgreedNotification(ctx, req) | 		s.notificationSender.FriendApplicationAgreedNotification(ctx, req, true) | ||||||
| 		return resp, nil | 		return resp, nil | ||||||
| 	} | 	} | ||||||
| 	if req.HandleResult == constant.FriendResponseRefuse { | 	if req.HandleResult == constant.FriendResponseRefuse { | ||||||
| @ -235,7 +237,7 @@ 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) { | ||||||
| 	if err := authverify.CheckAccessV3(ctx, req.OwnerUserID, s.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAccess(ctx, req.OwnerUserID); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -260,7 +262,7 @@ func (s *friendServer) SetFriendRemark(ctx context.Context, req *relation.SetFri | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if err := authverify.CheckAccessV3(ctx, req.OwnerUserID, s.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAccess(ctx, req.OwnerUserID); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -280,6 +282,9 @@ func (s *friendServer) SetFriendRemark(ctx context.Context, req *relation.SetFri | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *friendServer) GetFriendInfo(ctx context.Context, req *relation.GetFriendInfoReq) (*relation.GetFriendInfoResp, error) { | func (s *friendServer) GetFriendInfo(ctx context.Context, req *relation.GetFriendInfoReq) (*relation.GetFriendInfoResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.OwnerUserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	friends, err := s.db.FindFriendsWithError(ctx, req.OwnerUserID, req.FriendUserIDs) | 	friends, err := s.db.FindFriendsWithError(ctx, req.OwnerUserID, req.FriendUserIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -288,6 +293,9 @@ func (s *friendServer) GetFriendInfo(ctx context.Context, req *relation.GetFrien | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 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) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.OwnerUserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	resp = &relation.GetDesignatedFriendsResp{} | 	resp = &relation.GetDesignatedFriendsResp{} | ||||||
| 	if datautil.Duplicate(req.FriendUserIDs) { | 	if datautil.Duplicate(req.FriendUserIDs) { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("friend userID repeated") | 		return nil, errs.ErrArgs.WrapMsg("friend userID repeated") | ||||||
| @ -313,15 +321,16 @@ func (s *friendServer) getFriend(ctx context.Context, ownerUserID string, friend | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Get the list of friend requests sent out proactively. | // Get the list of friend requests sent out proactively. | ||||||
| func (s *friendServer) GetDesignatedFriendsApply(ctx context.Context, | func (s *friendServer) GetDesignatedFriendsApply(ctx context.Context, req *relation.GetDesignatedFriendsApplyReq) (resp *relation.GetDesignatedFriendsApplyResp, err error) { | ||||||
| 	req *relation.GetDesignatedFriendsApplyReq, | 	if err := authverify.CheckAccessIn(ctx, req.FromUserID, req.ToUserID); err != nil { | ||||||
| ) (resp *relation.GetDesignatedFriendsApplyResp, err error) { | 		return nil, err | ||||||
|  | 	} | ||||||
| 	friendRequests, err := s.db.FindBothFriendRequests(ctx, req.FromUserID, req.ToUserID) | 	friendRequests, err := s.db.FindBothFriendRequests(ctx, req.FromUserID, req.ToUserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	resp = &relation.GetDesignatedFriendsApplyResp{} | 	resp = &relation.GetDesignatedFriendsApplyResp{} | ||||||
| 	resp.FriendRequests, err = convert.FriendRequestDB2Pb(ctx, friendRequests, s.userClient.GetUsersInfoMap) | 	resp.FriendRequests, err = convert.FriendRequestDB2Pb(ctx, friendRequests, s.getCommonUserMap) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -330,17 +339,20 @@ 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 := authverify.CheckAccessV3(ctx, req.UserID, s.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	total, friendRequests, err := s.db.PageFriendRequestToMe(ctx, req.UserID, req.Pagination) | 	handleResults := datautil.Slice(req.HandleResults, func(e int32) int { | ||||||
|  | 		return int(e) | ||||||
|  | 	}) | ||||||
|  | 	total, friendRequests, err := s.db.PageFriendRequestToMe(ctx, req.UserID, handleResults, 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.userClient.GetUsersInfoMap) | 	resp.FriendRequests, err = convert.FriendRequestDB2Pb(ctx, friendRequests, s.getCommonUserMap) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -351,18 +363,20 @@ func (s *friendServer) GetPaginationFriendsApplyTo(ctx context.Context, req *rel | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 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{} | 	if err := authverify.CheckAccess(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) | 	handleResults := datautil.Slice(req.HandleResults, func(e int32) int { | ||||||
|  | 		return int(e) | ||||||
|  | 	}) | ||||||
|  | 	total, friendRequests, err := s.db.PageFriendRequestFromMe(ctx, req.UserID, handleResults, req.Pagination) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	resp.FriendRequests, err = convert.FriendRequestDB2Pb(ctx, friendRequests, s.userClient.GetUsersInfoMap) | 	resp = &relation.GetPaginationFriendsApplyFromResp{} | ||||||
|  | 	resp.FriendRequests, err = convert.FriendRequestDB2Pb(ctx, friendRequests, s.getCommonUserMap) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -374,6 +388,9 @@ func (s *friendServer) GetPaginationFriendsApplyFrom(ctx context.Context, req *r | |||||||
| 
 | 
 | ||||||
| // ok. | // ok. | ||||||
| func (s *friendServer) IsFriend(ctx context.Context, req *relation.IsFriendReq) (resp *relation.IsFriendResp, err error) { | func (s *friendServer) IsFriend(ctx context.Context, req *relation.IsFriendReq) (resp *relation.IsFriendResp, err error) { | ||||||
|  | 	if err := authverify.CheckAccessIn(ctx, req.UserID1, req.UserID2); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	resp = &relation.IsFriendResp{} | 	resp = &relation.IsFriendResp{} | ||||||
| 	resp.InUser1Friends, resp.InUser2Friends, err = s.db.CheckIn(ctx, req.UserID1, req.UserID2) | 	resp.InUser1Friends, resp.InUser2Friends, err = s.db.CheckIn(ctx, req.UserID1, req.UserID2) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -383,7 +400,7 @@ 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 := authverify.CheckAccessV3(ctx, req.UserID, s.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -404,7 +421,7 @@ func (s *friendServer) GetPaginationFriends(ctx context.Context, req *relation.G | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 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 := authverify.CheckAccessV3(ctx, req.UserID, s.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -426,6 +443,9 @@ func (s *friendServer) GetSpecifiedFriendsInfo(ctx context.Context, req *relatio | |||||||
| 		return nil, errs.ErrArgs.WrapMsg("userIDList repeated") | 		return nil, errs.ErrArgs.WrapMsg("userIDList repeated") | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.OwnerUserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	userMap, err := s.userClient.GetUsersInfoMap(ctx, req.UserIDList) | 	userMap, err := s.userClient.GetUsersInfoMap(ctx, req.UserIDList) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -494,10 +514,7 @@ func (s *friendServer) GetSpecifiedFriendsInfo(ctx context.Context, req *relatio | |||||||
| 	return resp, nil | 	return resp, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *friendServer) UpdateFriends( | func (s *friendServer) UpdateFriends(ctx context.Context, req *relation.UpdateFriendsReq) (*relation.UpdateFriendsResp, error) { | ||||||
| 	ctx context.Context, |  | ||||||
| 	req *relation.UpdateFriendsReq, |  | ||||||
| ) (*relation.UpdateFriendsResp, error) { |  | ||||||
| 	if len(req.FriendUserIDs) == 0 { | 	if len(req.FriendUserIDs) == 0 { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("friendIDList is empty") | 		return nil, errs.ErrArgs.WrapMsg("friendIDList is empty") | ||||||
| 	} | 	} | ||||||
| @ -505,6 +522,10 @@ func (s *friendServer) UpdateFriends( | |||||||
| 		return nil, errs.ErrArgs.WrapMsg("friendIDList repeated") | 		return nil, errs.ErrArgs.WrapMsg("friendIDList repeated") | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.OwnerUserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	_, err := s.db.FindFriendsWithError(ctx, req.OwnerUserID, req.FriendUserIDs) | 	_, err := s.db.FindFriendsWithError(ctx, req.OwnerUserID, req.FriendUserIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -530,3 +551,28 @@ func (s *friendServer) UpdateFriends( | |||||||
| 	s.notificationSender.FriendsInfoUpdateNotification(ctx, req.OwnerUserID, req.FriendUserIDs) | 	s.notificationSender.FriendsInfoUpdateNotification(ctx, req.OwnerUserID, req.FriendUserIDs) | ||||||
| 	return resp, nil | 	return resp, nil | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | func (s *friendServer) GetSelfUnhandledApplyCount(ctx context.Context, req *relation.GetSelfUnhandledApplyCountReq) (*relation.GetSelfUnhandledApplyCountResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	count, err := s.db.GetUnhandledCount(ctx, req.UserID, req.Time) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return &relation.GetSelfUnhandledApplyCountResp{ | ||||||
|  | 		Count: count, | ||||||
|  | 	}, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *friendServer) getCommonUserMap(ctx context.Context, userIDs []string) (map[string]common_user.CommonUser, error) { | ||||||
|  | 	users, err := s.userClient.GetUsersInfo(ctx, userIDs) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	return datautil.SliceToMapAny(users, func(e *sdkws.UserInfo) (string, common_user.CommonUser) { | ||||||
|  | 		return e.UserID, e | ||||||
|  | 	}), nil | ||||||
|  | } | ||||||
|  | |||||||
| @ -19,6 +19,9 @@ import ( | |||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli" | ||||||
| 	"github.com/openimsdk/protocol/msg" | 	"github.com/openimsdk/protocol/msg" | ||||||
|  | 	"github.com/openimsdk/tools/errs" | ||||||
|  | 	"github.com/openimsdk/tools/log" | ||||||
|  | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
| 
 | 
 | ||||||
| 	"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/versionctx" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/versionctx" | ||||||
| @ -52,9 +55,7 @@ func WithFriendDB(db controller.FriendDatabase) friendNotificationSenderOptions | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func WithDBFunc( | func WithDBFunc(fn func(ctx context.Context, userIDs []string) (users []*relationtb.User, err error)) friendNotificationSenderOptions { | ||||||
| 	fn func(ctx context.Context, userIDs []string) (users []*relationtb.User, err error), |  | ||||||
| ) friendNotificationSenderOptions { |  | ||||||
| 	return func(s *FriendNotificationSender) { | 	return func(s *FriendNotificationSender) { | ||||||
| 		f := func(ctx context.Context, userIDs []string) (result []common_user.CommonUser, err error) { | 		f := func(ctx context.Context, userIDs []string) (result []common_user.CommonUser, err error) { | ||||||
| 			users, err := fn(ctx, userIDs) | 			users, err := fn(ctx, userIDs) | ||||||
| @ -70,9 +71,7 @@ func WithDBFunc( | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func WithRpcFunc( | func WithRpcFunc(fn func(ctx context.Context, userIDs []string) ([]*sdkws.UserInfo, error)) friendNotificationSenderOptions { | ||||||
| 	fn func(ctx context.Context, userIDs []string) ([]*sdkws.UserInfo, error), |  | ||||||
| ) friendNotificationSenderOptions { |  | ||||||
| 	return func(s *FriendNotificationSender) { | 	return func(s *FriendNotificationSender) { | ||||||
| 		f := func(ctx context.Context, userIDs []string) (result []common_user.CommonUser, err error) { | 		f := func(ctx context.Context, userIDs []string) (result []common_user.CommonUser, err error) { | ||||||
| 			users, err := fn(ctx, userIDs) | 			users, err := fn(ctx, userIDs) | ||||||
| @ -100,10 +99,7 @@ func NewFriendNotificationSender(conf *config.Notification, msgClient *rpcli.Msg | |||||||
| 	return f | 	return f | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (f *FriendNotificationSender) getUsersInfoMap( | func (f *FriendNotificationSender) getUsersInfoMap(ctx context.Context, userIDs []string) (map[string]*sdkws.UserInfo, error) { | ||||||
| 	ctx context.Context, |  | ||||||
| 	userIDs []string, |  | ||||||
| ) (map[string]*sdkws.UserInfo, error) { |  | ||||||
| 	users, err := f.getUsersInfo(ctx, userIDs) | 	users, err := f.getUsersInfo(ctx, userIDs) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -116,10 +112,7 @@ func (f *FriendNotificationSender) getUsersInfoMap( | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| //nolint:unused | //nolint:unused | ||||||
| func (f *FriendNotificationSender) getFromToUserNickname( | func (f *FriendNotificationSender) getFromToUserNickname(ctx context.Context, fromUserID, toUserID string) (string, string, error) { | ||||||
| 	ctx context.Context, |  | ||||||
| 	fromUserID, toUserID string, |  | ||||||
| ) (string, string, error) { |  | ||||||
| 	users, err := f.getUsersInfoMap(ctx, []string{fromUserID, toUserID}) | 	users, err := f.getUsersInfoMap(ctx, []string{fromUserID, toUserID}) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return "", "", nil | 		return "", "", nil | ||||||
| @ -132,60 +125,113 @@ func (f *FriendNotificationSender) UserInfoUpdatedNotification(ctx context.Conte | |||||||
| 	f.Notification(ctx, mcontext.GetOpUserID(ctx), changedUserID, constant.UserInfoUpdatedNotification, &tips) | 	f.Notification(ctx, mcontext.GetOpUserID(ctx), changedUserID, constant.UserInfoUpdatedNotification, &tips) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func (f *FriendNotificationSender) getCommonUserMap(ctx context.Context, userIDs []string) (map[string]common_user.CommonUser, error) { | ||||||
|  | 	users, err := f.getUsersInfo(ctx, userIDs) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	return datautil.SliceToMap(users, func(e common_user.CommonUser) string { | ||||||
|  | 		return e.GetUserID() | ||||||
|  | 	}), nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (f *FriendNotificationSender) getFriendRequests(ctx context.Context, fromUserID, toUserID string) (*sdkws.FriendRequest, error) { | ||||||
|  | 	if f.db == nil { | ||||||
|  | 		return nil, errs.ErrInternalServer.WithDetail("db is nil") | ||||||
|  | 	} | ||||||
|  | 	friendRequests, err := f.db.FindBothFriendRequests(ctx, fromUserID, toUserID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	requests, err := convert.FriendRequestDB2Pb(ctx, friendRequests, f.getCommonUserMap) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	for _, request := range requests { | ||||||
|  | 		if request.FromUserID == fromUserID && request.ToUserID == toUserID { | ||||||
|  | 			return request, nil | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return nil, errs.ErrRecordNotFound.WrapMsg("friend request not found", "fromUserID", fromUserID, "toUserID", toUserID) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func (f *FriendNotificationSender) FriendApplicationAddNotification(ctx context.Context, req *relation.ApplyToAddFriendReq) { | func (f *FriendNotificationSender) FriendApplicationAddNotification(ctx context.Context, req *relation.ApplyToAddFriendReq) { | ||||||
| 	tips := sdkws.FriendApplicationTips{FromToUserID: &sdkws.FromToUserID{ | 	request, err := f.getFriendRequests(ctx, req.FromUserID, req.ToUserID) | ||||||
| 		FromUserID: req.FromUserID, | 	if err != nil { | ||||||
| 		ToUserID:   req.ToUserID, | 		log.ZError(ctx, "FriendApplicationAddNotification get friend request", err, "fromUserID", req.FromUserID, "toUserID", req.ToUserID) | ||||||
| 	}} | 		return | ||||||
|  | 	} | ||||||
|  | 	tips := sdkws.FriendApplicationTips{ | ||||||
|  | 		FromToUserID: &sdkws.FromToUserID{ | ||||||
|  | 			FromUserID: req.FromUserID, | ||||||
|  | 			ToUserID:   req.ToUserID, | ||||||
|  | 		}, | ||||||
|  | 		Request: request, | ||||||
|  | 	} | ||||||
| 	f.Notification(ctx, req.FromUserID, req.ToUserID, constant.FriendApplicationNotification, &tips) | 	f.Notification(ctx, req.FromUserID, req.ToUserID, constant.FriendApplicationNotification, &tips) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (f *FriendNotificationSender) FriendApplicationAgreedNotification( | func (f *FriendNotificationSender) FriendApplicationAgreedNotification(ctx context.Context, req *relation.RespondFriendApplyReq, checkReq bool) { | ||||||
| 	ctx context.Context, | 	var ( | ||||||
| 	req *relation.RespondFriendApplyReq, | 		request *sdkws.FriendRequest | ||||||
| ) { | 		err     error | ||||||
| 	tips := sdkws.FriendApplicationApprovedTips{FromToUserID: &sdkws.FromToUserID{ | 	) | ||||||
| 		FromUserID: req.FromUserID, | 	if checkReq { | ||||||
| 		ToUserID:   req.ToUserID, | 		request, err = f.getFriendRequests(ctx, req.FromUserID, req.ToUserID) | ||||||
| 	}, HandleMsg: req.HandleMsg} | 		if err != nil { | ||||||
|  | 			log.ZError(ctx, "FriendApplicationAgreedNotification get friend request", err, "fromUserID", req.FromUserID, "toUserID", req.ToUserID) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	tips := sdkws.FriendApplicationApprovedTips{ | ||||||
|  | 		FromToUserID: &sdkws.FromToUserID{ | ||||||
|  | 			FromUserID: req.FromUserID, | ||||||
|  | 			ToUserID:   req.ToUserID, | ||||||
|  | 		}, | ||||||
|  | 		HandleMsg: req.HandleMsg, | ||||||
|  | 		Request:   request, | ||||||
|  | 	} | ||||||
| 	f.Notification(ctx, req.ToUserID, req.FromUserID, constant.FriendApplicationApprovedNotification, &tips) | 	f.Notification(ctx, req.ToUserID, req.FromUserID, constant.FriendApplicationApprovedNotification, &tips) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (f *FriendNotificationSender) FriendApplicationRefusedNotification( | func (f *FriendNotificationSender) FriendApplicationRefusedNotification(ctx context.Context, req *relation.RespondFriendApplyReq) { | ||||||
| 	ctx context.Context, | 	request, err := f.getFriendRequests(ctx, req.FromUserID, req.ToUserID) | ||||||
| 	req *relation.RespondFriendApplyReq, | 	if err != nil { | ||||||
| ) { | 		log.ZError(ctx, "FriendApplicationRefusedNotification get friend request", err, "fromUserID", req.FromUserID, "toUserID", req.ToUserID) | ||||||
| 	tips := sdkws.FriendApplicationApprovedTips{FromToUserID: &sdkws.FromToUserID{ | 		return | ||||||
| 		FromUserID: req.FromUserID, | 	} | ||||||
| 		ToUserID:   req.ToUserID, | 	tips := sdkws.FriendApplicationRejectedTips{ | ||||||
| 	}, HandleMsg: req.HandleMsg} | 		FromToUserID: &sdkws.FromToUserID{ | ||||||
|  | 			FromUserID: req.FromUserID, | ||||||
|  | 			ToUserID:   req.ToUserID, | ||||||
|  | 		}, | ||||||
|  | 		HandleMsg: req.HandleMsg, | ||||||
|  | 		Request:   request, | ||||||
|  | 	} | ||||||
| 	f.Notification(ctx, req.ToUserID, req.FromUserID, constant.FriendApplicationRejectedNotification, &tips) | 	f.Notification(ctx, req.ToUserID, req.FromUserID, constant.FriendApplicationRejectedNotification, &tips) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (f *FriendNotificationSender) FriendAddedNotification( | //func (f *FriendNotificationSender) FriendAddedNotification(ctx context.Context, operationID, opUserID, fromUserID, toUserID string) error { | ||||||
| 	ctx context.Context, | //	tips := sdkws.FriendAddedTips{Friend: &sdkws.FriendInfo{}, OpUser: &sdkws.PublicUserInfo{}} | ||||||
| 	operationID, opUserID, fromUserID, toUserID string, | //	user, err := f.getUsersInfo(ctx, []string{opUserID}) | ||||||
| ) error { | //	if err != nil { | ||||||
| 	tips := sdkws.FriendAddedTips{Friend: &sdkws.FriendInfo{}, OpUser: &sdkws.PublicUserInfo{}} | //		return err | ||||||
| 	user, err := f.getUsersInfo(ctx, []string{opUserID}) | //	} | ||||||
| 	if err != nil { | //	tips.OpUser.UserID = user[0].GetUserID() | ||||||
| 		return err | //	tips.OpUser.Ex = user[0].GetEx() | ||||||
| 	} | //	tips.OpUser.Nickname = user[0].GetNickname() | ||||||
| 	tips.OpUser.UserID = user[0].GetUserID() | //	tips.OpUser.FaceURL = user[0].GetFaceURL() | ||||||
| 	tips.OpUser.Ex = user[0].GetEx() | //	friends, err := f.db.FindFriendsWithError(ctx, fromUserID, []string{toUserID}) | ||||||
| 	tips.OpUser.Nickname = user[0].GetNickname() | //	if err != nil { | ||||||
| 	tips.OpUser.FaceURL = user[0].GetFaceURL() | //		return err | ||||||
| 	friends, err := f.db.FindFriendsWithError(ctx, fromUserID, []string{toUserID}) | //	} | ||||||
| 	if err != nil { | //	tips.Friend, err = convert.FriendDB2Pb(ctx, friends[0], f.getUsersInfoMap) | ||||||
| 		return err | //	if err != nil { | ||||||
| 	} | //		return err | ||||||
| 	tips.Friend, err = convert.FriendDB2Pb(ctx, friends[0], f.getUsersInfoMap) | //	} | ||||||
| 	if err != nil { | //	f.Notification(ctx, fromUserID, toUserID, constant.FriendAddedNotification, &tips) | ||||||
| 		return err | //	return nil | ||||||
| 	} | //} | ||||||
| 	f.Notification(ctx, fromUserID, toUserID, constant.FriendAddedNotification, &tips) |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
| func (f *FriendNotificationSender) FriendDeletedNotification(ctx context.Context, req *relation.DeleteFriendReq) { | func (f *FriendNotificationSender) FriendDeletedNotification(ctx context.Context, req *relation.DeleteFriendReq) { | ||||||
| 	tips := sdkws.FriendDeletedTips{FromToUserID: &sdkws.FromToUserID{ | 	tips := sdkws.FriendDeletedTips{FromToUserID: &sdkws.FromToUserID{ | ||||||
|  | |||||||
| @ -2,10 +2,11 @@ package relation | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 	"slices" | ||||||
|  | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/util/hashutil" | 	"github.com/openimsdk/open-im-server/v3/pkg/util/hashutil" | ||||||
| 	"github.com/openimsdk/protocol/sdkws" | 	"github.com/openimsdk/protocol/sdkws" | ||||||
| 	"github.com/openimsdk/tools/log" | 	"github.com/openimsdk/tools/log" | ||||||
| 	"slices" |  | ||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/internal/rpc/incrversion" | 	"github.com/openimsdk/open-im-server/v3/internal/rpc/incrversion" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||||
| @ -39,6 +40,9 @@ func (s *friendServer) NotificationUserInfoUpdate(ctx context.Context, req *rela | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *friendServer) GetFullFriendUserIDs(ctx context.Context, req *relation.GetFullFriendUserIDsReq) (*relation.GetFullFriendUserIDsResp, error) { | func (s *friendServer) GetFullFriendUserIDs(ctx context.Context, req *relation.GetFullFriendUserIDsReq) (*relation.GetFullFriendUserIDsResp, error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	vl, err := s.db.FindMaxFriendVersionCache(ctx, req.UserID) | 	vl, err := s.db.FindMaxFriendVersionCache(ctx, req.UserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -60,7 +64,7 @@ func (s *friendServer) GetFullFriendUserIDs(ctx context.Context, req *relation.G | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *friendServer) GetIncrementalFriends(ctx context.Context, req *relation.GetIncrementalFriendsReq) (*relation.GetIncrementalFriendsResp, error) { | func (s *friendServer) GetIncrementalFriends(ctx context.Context, req *relation.GetIncrementalFriendsReq) (*relation.GetIncrementalFriendsResp, error) { | ||||||
| 	if err := authverify.CheckAccessV3(ctx, req.UserID, s.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	var sortVersion uint64 | 	var sortVersion uint64 | ||||||
|  | |||||||
| @ -25,6 +25,7 @@ import ( | |||||||
| 	"github.com/openimsdk/protocol/constant" | 	"github.com/openimsdk/protocol/constant" | ||||||
| 	"github.com/openimsdk/protocol/third" | 	"github.com/openimsdk/protocol/third" | ||||||
| 	"github.com/openimsdk/tools/errs" | 	"github.com/openimsdk/tools/errs" | ||||||
|  | 	"github.com/openimsdk/tools/mcontext" | ||||||
| 	"github.com/openimsdk/tools/utils/datautil" | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| @ -45,7 +46,7 @@ func genLogID() string { | |||||||
| 
 | 
 | ||||||
| func (t *thirdServer) UploadLogs(ctx context.Context, req *third.UploadLogsReq) (*third.UploadLogsResp, error) { | func (t *thirdServer) UploadLogs(ctx context.Context, req *third.UploadLogsReq) (*third.UploadLogsResp, error) { | ||||||
| 	var dbLogs []*relationtb.Log | 	var dbLogs []*relationtb.Log | ||||||
| 	userID := ctx.Value(constant.OpUserID).(string) | 	userID := mcontext.GetOpUserID(ctx) | ||||||
| 	platform := constant.PlatformID2Name[int(req.Platform)] | 	platform := constant.PlatformID2Name[int(req.Platform)] | ||||||
| 	for _, fileURL := range req.FileURLs { | 	for _, fileURL := range req.FileURLs { | ||||||
| 		log := relationtb.Log{ | 		log := relationtb.Log{ | ||||||
| @ -82,7 +83,7 @@ func (t *thirdServer) UploadLogs(ctx context.Context, req *third.UploadLogsReq) | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (t *thirdServer) DeleteLogs(ctx context.Context, req *third.DeleteLogsReq) (*third.DeleteLogsResp, error) { | func (t *thirdServer) DeleteLogs(ctx context.Context, req *third.DeleteLogsReq) (*third.DeleteLogsResp, error) { | ||||||
| 	if err := authverify.CheckAdmin(ctx, t.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAdmin(ctx); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	userID := "" | 	userID := "" | ||||||
| @ -123,7 +124,7 @@ func dbToPbLogInfos(logs []*relationtb.Log) []*third.LogInfo { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (t *thirdServer) SearchLogs(ctx context.Context, req *third.SearchLogsReq) (*third.SearchLogsResp, error) { | func (t *thirdServer) SearchLogs(ctx context.Context, req *third.SearchLogsReq) (*third.SearchLogsResp, error) { | ||||||
| 	if err := authverify.CheckAdmin(ctx, t.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAdmin(ctx); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	var ( | 	var ( | ||||||
|  | |||||||
| @ -62,7 +62,7 @@ func (t *thirdServer) InitiateMultipartUpload(ctx context.Context, req *third.In | |||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	expireTime := time.Now().Add(t.defaultExpire) | 	expireTime := time.Now().Add(t.defaultExpire) | ||||||
| 	result, err := t.s3dataBase.InitiateMultipartUpload(ctx, req.Hash, req.Size, t.defaultExpire, int(req.MaxParts)) | 	result, err := t.s3dataBase.InitiateMultipartUpload(ctx, req.Hash, req.Size, t.defaultExpire, int(req.MaxParts), req.ContentType) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		if haErr, ok := errs.Unwrap(err).(*cont.HashAlreadyExistsError); ok { | 		if haErr, ok := errs.Unwrap(err).(*cont.HashAlreadyExistsError); ok { | ||||||
| 			obj := &model.Object{ | 			obj := &model.Object{ | ||||||
| @ -198,7 +198,7 @@ func (t *thirdServer) InitiateFormData(ctx context.Context, req *third.InitiateF | |||||||
| 	var duration time.Duration | 	var duration time.Duration | ||||||
| 	opUserID := mcontext.GetOpUserID(ctx) | 	opUserID := mcontext.GetOpUserID(ctx) | ||||||
| 	var key string | 	var key string | ||||||
| 	if t.IsManagerUserID(opUserID) { | 	if authverify.CheckUserIsAdmin(ctx, opUserID) { | ||||||
| 		if req.Millisecond <= 0 { | 		if req.Millisecond <= 0 { | ||||||
| 			duration = time.Minute * 10 | 			duration = time.Minute * 10 | ||||||
| 		} else { | 		} else { | ||||||
| @ -289,7 +289,7 @@ func (t *thirdServer) apiAddress(prefix, name string) string { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (t *thirdServer) DeleteOutdatedData(ctx context.Context, req *third.DeleteOutdatedDataReq) (*third.DeleteOutdatedDataResp, error) { | func (t *thirdServer) DeleteOutdatedData(ctx context.Context, req *third.DeleteOutdatedDataReq) (*third.DeleteOutdatedDataResp, error) { | ||||||
| 	if err := authverify.CheckAdmin(ctx, t.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAdmin(ctx); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	engine := t.config.RpcConfig.Object.Enable | 	engine := t.config.RpcConfig.Object.Enable | ||||||
|  | |||||||
| @ -19,6 +19,7 @@ import ( | |||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"time" | 	"time" | ||||||
| 
 | 
 | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||||
| 	"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/mcache" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/cache/mcache" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/pkg/dbbuild" | 	"github.com/openimsdk/open-im-server/v3/pkg/dbbuild" | ||||||
| @ -63,7 +64,7 @@ type Config struct { | |||||||
| 	Discovery          config.Discovery | 	Discovery          config.Discovery | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func Start(ctx context.Context, config *Config, client discovery.Conn, server grpc.ServiceRegistrar) error { | func Start(ctx context.Context, config *Config, client discovery.SvcDiscoveryRegistry, server grpc.ServiceRegistrar) error { | ||||||
| 	dbb := dbbuild.NewBuilder(&config.MongodbConfig, &config.RedisConfig) | 	dbb := dbbuild.NewBuilder(&config.MongodbConfig, &config.RedisConfig) | ||||||
| 	mgocli, err := dbb.Mongo(ctx) | 	mgocli, err := dbb.Mongo(ctx) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -148,6 +149,9 @@ func (t *thirdServer) FcmUpdateToken(ctx context.Context, req *third.FcmUpdateTo | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (t *thirdServer) SetAppBadge(ctx context.Context, req *third.SetAppBadgeReq) (resp *third.SetAppBadgeResp, err error) { | func (t *thirdServer) SetAppBadge(ctx context.Context, req *third.SetAppBadgeReq) (resp *third.SetAppBadgeResp, err error) { | ||||||
|  | 	if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 	err = t.thirdDatabase.SetAppBadge(ctx, req.UserID, int(req.AppUnreadCount)) | 	err = t.thirdDatabase.SetAppBadge(ctx, req.UserID, int(req.AppUnreadCount)) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
|  | |||||||
| @ -54,7 +54,7 @@ func (t *thirdServer) checkUploadName(ctx context.Context, name string) error { | |||||||
| 	if opUserID == "" { | 	if opUserID == "" { | ||||||
| 		return errs.ErrNoPermission.WrapMsg("opUserID is empty") | 		return errs.ErrNoPermission.WrapMsg("opUserID is empty") | ||||||
| 	} | 	} | ||||||
| 	if !authverify.IsManagerUserID(opUserID, t.config.Share.IMAdminUserID) { | 	if !authverify.CheckUserIsAdmin(ctx, opUserID) { | ||||||
| 		if !strings.HasPrefix(name, opUserID+"/") { | 		if !strings.HasPrefix(name, opUserID+"/") { | ||||||
| 			return errs.ErrNoPermission.WrapMsg(fmt.Sprintf("name must start with `%s/`", opUserID)) | 			return errs.ErrNoPermission.WrapMsg(fmt.Sprintf("name must start with `%s/`", opUserID)) | ||||||
| 		} | 		} | ||||||
| @ -79,10 +79,6 @@ func checkValidObjectName(objectName string) error { | |||||||
| 	return checkValidObjectNamePrefix(objectName) | 	return checkValidObjectNamePrefix(objectName) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (t *thirdServer) IsManagerUserID(opUserID string) bool { |  | ||||||
| 	return authverify.IsManagerUserID(opUserID, t.config.Share.IMAdminUserID) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func putUpdate[T any](update map[string]any, name string, val interface{ GetValuePtr() *T }) { | func putUpdate[T any](update map[string]any, name string, val interface{ GetValuePtr() *T }) { | ||||||
| 	ptrVal := val.GetValuePtr() | 	ptrVal := val.GetValuePtr() | ||||||
| 	if ptrVal == nil { | 	if ptrVal == nil { | ||||||
|  | |||||||
							
								
								
									
										71
									
								
								internal/rpc/user/config.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								internal/rpc/user/config.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,71 @@ | |||||||
|  | package user | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"context" | ||||||
|  | 
 | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/authverify" | ||||||
|  | 	"github.com/openimsdk/open-im-server/v3/pkg/common/storage/model" | ||||||
|  | 	pbuser "github.com/openimsdk/protocol/user" | ||||||
|  | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func (s *userServer) GetUserClientConfig(ctx context.Context, req *pbuser.GetUserClientConfigReq) (*pbuser.GetUserClientConfigResp, error) { | ||||||
|  | 	if req.UserID != "" { | ||||||
|  | 		if err := authverify.CheckAccess(ctx, req.UserID); err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 		if _, err := s.db.GetUserByID(ctx, req.UserID); err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	res, err := s.clientConfig.GetUserConfig(ctx, req.UserID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	return &pbuser.GetUserClientConfigResp{Configs: res}, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *userServer) SetUserClientConfig(ctx context.Context, req *pbuser.SetUserClientConfigReq) (*pbuser.SetUserClientConfigResp, error) { | ||||||
|  | 	if err := authverify.CheckAdmin(ctx); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	if req.UserID != "" { | ||||||
|  | 		if _, err := s.db.GetUserByID(ctx, req.UserID); err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	if err := s.clientConfig.SetUserConfig(ctx, req.UserID, req.Configs); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	return &pbuser.SetUserClientConfigResp{}, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *userServer) DelUserClientConfig(ctx context.Context, req *pbuser.DelUserClientConfigReq) (*pbuser.DelUserClientConfigResp, error) { | ||||||
|  | 	if err := authverify.CheckAdmin(ctx); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	if err := s.clientConfig.DelUserConfig(ctx, req.UserID, req.Keys); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	return &pbuser.DelUserClientConfigResp{}, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *userServer) PageUserClientConfig(ctx context.Context, req *pbuser.PageUserClientConfigReq) (*pbuser.PageUserClientConfigResp, error) { | ||||||
|  | 	if err := authverify.CheckAdmin(ctx); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	total, res, err := s.clientConfig.GetUserConfigPage(ctx, req.UserID, req.Key, req.Pagination) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	return &pbuser.PageUserClientConfigResp{ | ||||||
|  | 		Total: total, | ||||||
|  | 		Configs: datautil.Slice(res, func(e *model.ClientConfig) *pbuser.ClientConfig { | ||||||
|  | 			return &pbuser.ClientConfig{ | ||||||
|  | 				UserID: e.UserID, | ||||||
|  | 				Key:    e.Key, | ||||||
|  | 				Value:  e.Value, | ||||||
|  | 			} | ||||||
|  | 		}), | ||||||
|  | 	}, nil | ||||||
|  | } | ||||||
| @ -49,6 +49,10 @@ import ( | |||||||
| 	"google.golang.org/grpc" | 	"google.golang.org/grpc" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | const ( | ||||||
|  | 	defaultSecret = "openIM123" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
| type userServer struct { | type userServer struct { | ||||||
| 	pbuser.UnimplementedUserServer | 	pbuser.UnimplementedUserServer | ||||||
| 	online                   cache.OnlineCache | 	online                   cache.OnlineCache | ||||||
| @ -60,6 +64,9 @@ type userServer struct { | |||||||
| 	webhookClient            *webhook.Client | 	webhookClient            *webhook.Client | ||||||
| 	groupClient              *rpcli.GroupClient | 	groupClient              *rpcli.GroupClient | ||||||
| 	relationClient           *rpcli.RelationClient | 	relationClient           *rpcli.RelationClient | ||||||
|  | 	clientConfig             controller.ClientConfigDatabase | ||||||
|  | 
 | ||||||
|  | 	adminUserIDs []string | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| type Config struct { | type Config struct { | ||||||
| @ -74,7 +81,7 @@ type Config struct { | |||||||
| 	Discovery          config.Discovery | 	Discovery          config.Discovery | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func Start(ctx context.Context, config *Config, client discovery.Conn, server grpc.ServiceRegistrar) error { | func Start(ctx context.Context, config *Config, client discovery.SvcDiscoveryRegistry, server grpc.ServiceRegistrar) error { | ||||||
| 	dbb := dbbuild.NewBuilder(&config.MongodbConfig, &config.RedisConfig) | 	dbb := dbbuild.NewBuilder(&config.MongodbConfig, &config.RedisConfig) | ||||||
| 	mgocli, err := dbb.Mongo(ctx) | 	mgocli, err := dbb.Mongo(ctx) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -87,13 +94,21 @@ func Start(ctx context.Context, config *Config, client discovery.Conn, server gr | |||||||
| 
 | 
 | ||||||
| 	users := make([]*tablerelation.User, 0) | 	users := make([]*tablerelation.User, 0) | ||||||
| 
 | 
 | ||||||
| 	for _, v := range config.Share.IMAdminUserID { | 	for i := range config.Share.IMAdminUser.UserIDs { | ||||||
| 		users = append(users, &tablerelation.User{UserID: v, Nickname: v, AppMangerLevel: constant.AppNotificationAdmin}) | 		users = append(users, &tablerelation.User{ | ||||||
|  | 			UserID:         config.Share.IMAdminUser.UserIDs[i], | ||||||
|  | 			Nickname:       config.Share.IMAdminUser.Nicknames[i], | ||||||
|  | 			AppMangerLevel: constant.AppAdmin, | ||||||
|  | 		}) | ||||||
| 	} | 	} | ||||||
| 	userDB, err := mgo.NewUserMongo(mgocli.GetDB()) | 	userDB, err := mgo.NewUserMongo(mgocli.GetDB()) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  | 	clientConfigDB, err := mgo.NewClientConfig(mgocli.GetDB()) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
| 	msgConn, err := client.GetConn(ctx, config.Discovery.RpcService.Msg) | 	msgConn, err := client.GetConn(ctx, config.Discovery.RpcService.Msg) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| @ -118,9 +133,10 @@ func Start(ctx context.Context, config *Config, client discovery.Conn, server gr | |||||||
| 		userNotificationSender:   NewUserNotificationSender(config, msgClient, WithUserFunc(database.FindWithError)), | 		userNotificationSender:   NewUserNotificationSender(config, msgClient, WithUserFunc(database.FindWithError)), | ||||||
| 		config:                   config, | 		config:                   config, | ||||||
| 		webhookClient:            webhook.NewWebhookClient(config.WebhooksConfig.URL), | 		webhookClient:            webhook.NewWebhookClient(config.WebhooksConfig.URL), | ||||||
| 
 | 		clientConfig:             controller.NewClientConfigDatabase(clientConfigDB, redis.NewClientConfigCache(rdb, clientConfigDB), mgocli.GetTx()), | ||||||
| 		groupClient:    rpcli.NewGroupClient(groupConn), | 		groupClient:              rpcli.NewGroupClient(groupConn), | ||||||
| 		relationClient: rpcli.NewRelationClient(friendConn), | 		relationClient:           rpcli.NewRelationClient(friendConn), | ||||||
|  | 		adminUserIDs:             config.Share.IMAdminUser.UserIDs, | ||||||
| 	} | 	} | ||||||
| 	pbuser.RegisterUserServer(server, u) | 	pbuser.RegisterUserServer(server, u) | ||||||
| 	return u.db.InitOnce(context.Background(), users) | 	return u.db.InitOnce(context.Background(), users) | ||||||
| @ -141,7 +157,7 @@ func (s *userServer) GetDesignateUsers(ctx context.Context, req *pbuser.GetDesig | |||||||
| // UpdateUserInfo | // UpdateUserInfo | ||||||
| func (s *userServer) UpdateUserInfo(ctx context.Context, req *pbuser.UpdateUserInfoReq) (resp *pbuser.UpdateUserInfoResp, err error) { | func (s *userServer) UpdateUserInfo(ctx context.Context, req *pbuser.UpdateUserInfoReq) (resp *pbuser.UpdateUserInfoResp, err error) { | ||||||
| 	resp = &pbuser.UpdateUserInfoResp{} | 	resp = &pbuser.UpdateUserInfoResp{} | ||||||
| 	err = authverify.CheckAccessV3(ctx, req.UserInfo.UserID, s.config.Share.IMAdminUserID) | 	err = authverify.CheckAccess(ctx, req.UserInfo.UserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -168,7 +184,7 @@ func (s *userServer) UpdateUserInfo(ctx context.Context, req *pbuser.UpdateUserI | |||||||
| 
 | 
 | ||||||
| func (s *userServer) UpdateUserInfoEx(ctx context.Context, req *pbuser.UpdateUserInfoExReq) (resp *pbuser.UpdateUserInfoExResp, err error) { | func (s *userServer) UpdateUserInfoEx(ctx context.Context, req *pbuser.UpdateUserInfoExReq) (resp *pbuser.UpdateUserInfoExResp, err error) { | ||||||
| 	resp = &pbuser.UpdateUserInfoExResp{} | 	resp = &pbuser.UpdateUserInfoExResp{} | ||||||
| 	err = authverify.CheckAccessV3(ctx, req.UserInfo.UserID, s.config.Share.IMAdminUserID) | 	err = authverify.CheckAccess(ctx, req.UserInfo.UserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -188,6 +204,7 @@ func (s *userServer) UpdateUserInfoEx(ctx context.Context, req *pbuser.UpdateUse | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	s.friendNotificationSender.UserInfoUpdatedNotification(ctx, req.UserInfo.UserID) | 	s.friendNotificationSender.UserInfoUpdatedNotification(ctx, req.UserInfo.UserID) | ||||||
|  | 
 | ||||||
| 	//friends, err := s.friendRpcClient.GetFriendIDs(ctx, req.UserInfo.UserID) | 	//friends, err := s.friendRpcClient.GetFriendIDs(ctx, req.UserInfo.UserID) | ||||||
| 	//if err != nil { | 	//if err != nil { | ||||||
| 	//	return nil, err | 	//	return nil, err | ||||||
| @ -200,6 +217,7 @@ func (s *userServer) UpdateUserInfoEx(ctx context.Context, req *pbuser.UpdateUse | |||||||
| 	//for _, friendID := range friends { | 	//for _, friendID := range friends { | ||||||
| 	//	s.friendNotificationSender.FriendInfoUpdatedNotification(ctx, req.UserInfo.UserID, friendID) | 	//	s.friendNotificationSender.FriendInfoUpdatedNotification(ctx, req.UserInfo.UserID, friendID) | ||||||
| 	//} | 	//} | ||||||
|  | 
 | ||||||
| 	s.webhookAfterUpdateUserInfoEx(ctx, &s.config.WebhooksConfig.AfterUpdateUserInfoEx, req) | 	s.webhookAfterUpdateUserInfoEx(ctx, &s.config.WebhooksConfig.AfterUpdateUserInfoEx, req) | ||||||
| 	if err := s.NotificationUserInfoUpdate(ctx, req.UserInfo.UserID, oldUser); err != nil { | 	if err := s.NotificationUserInfoUpdate(ctx, req.UserInfo.UserID, oldUser); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @ -226,8 +244,7 @@ func (s *userServer) AccountCheck(ctx context.Context, req *pbuser.AccountCheckR | |||||||
| 	if datautil.Duplicate(req.CheckUserIDs) { | 	if datautil.Duplicate(req.CheckUserIDs) { | ||||||
| 		return nil, errs.ErrArgs.WrapMsg("userID repeated") | 		return nil, errs.ErrArgs.WrapMsg("userID repeated") | ||||||
| 	} | 	} | ||||||
| 	err = authverify.CheckAdmin(ctx, s.config.Share.IMAdminUserID) | 	if err = authverify.CheckAdmin(ctx); err != nil { | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	users, err := s.db.Find(ctx, req.CheckUserIDs) | 	users, err := s.db.Find(ctx, req.CheckUserIDs) | ||||||
| @ -273,11 +290,13 @@ 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") | ||||||
| 	} | 	} | ||||||
| 
 | 	// check if secret is changed | ||||||
| 	if err = authverify.CheckAdmin(ctx, s.config.Share.IMAdminUserID); err != nil { | 	//if s.config.Share.Secret == defaultSecret { | ||||||
|  | 	//	return nil, servererrs.ErrSecretNotChanged.Wrap() | ||||||
|  | 	//} | ||||||
|  | 	if err = authverify.CheckAdmin(ctx); err != nil { | ||||||
| 		return nil, err | 		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") | ||||||
| 	} | 	} | ||||||
| @ -343,7 +362,7 @@ func (s *userServer) GetAllUserID(ctx context.Context, req *pbuser.GetAllUserIDR | |||||||
| 
 | 
 | ||||||
| // ProcessUserCommandAdd user general function add. | // ProcessUserCommandAdd user general function add. | ||||||
| func (s *userServer) ProcessUserCommandAdd(ctx context.Context, req *pbuser.ProcessUserCommandAddReq) (*pbuser.ProcessUserCommandAddResp, error) { | func (s *userServer) ProcessUserCommandAdd(ctx context.Context, req *pbuser.ProcessUserCommandAddReq) (*pbuser.ProcessUserCommandAddResp, error) { | ||||||
| 	err := authverify.CheckAccessV3(ctx, req.UserID, s.config.Share.IMAdminUserID) | 	err := authverify.CheckAccess(ctx, req.UserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -371,7 +390,7 @@ func (s *userServer) ProcessUserCommandAdd(ctx context.Context, req *pbuser.Proc | |||||||
| 
 | 
 | ||||||
| // ProcessUserCommandDelete user general function delete. | // ProcessUserCommandDelete user general function delete. | ||||||
| func (s *userServer) ProcessUserCommandDelete(ctx context.Context, req *pbuser.ProcessUserCommandDeleteReq) (*pbuser.ProcessUserCommandDeleteResp, error) { | func (s *userServer) ProcessUserCommandDelete(ctx context.Context, req *pbuser.ProcessUserCommandDeleteReq) (*pbuser.ProcessUserCommandDeleteResp, error) { | ||||||
| 	err := authverify.CheckAccessV3(ctx, req.UserID, s.config.Share.IMAdminUserID) | 	err := authverify.CheckAccess(ctx, req.UserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -390,7 +409,7 @@ func (s *userServer) ProcessUserCommandDelete(ctx context.Context, req *pbuser.P | |||||||
| 
 | 
 | ||||||
| // ProcessUserCommandUpdate user general function update. | // ProcessUserCommandUpdate user general function update. | ||||||
| func (s *userServer) ProcessUserCommandUpdate(ctx context.Context, req *pbuser.ProcessUserCommandUpdateReq) (*pbuser.ProcessUserCommandUpdateResp, error) { | func (s *userServer) ProcessUserCommandUpdate(ctx context.Context, req *pbuser.ProcessUserCommandUpdateReq) (*pbuser.ProcessUserCommandUpdateResp, error) { | ||||||
| 	err := authverify.CheckAccessV3(ctx, req.UserID, s.config.Share.IMAdminUserID) | 	err := authverify.CheckAccess(ctx, req.UserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -419,7 +438,7 @@ func (s *userServer) ProcessUserCommandUpdate(ctx context.Context, req *pbuser.P | |||||||
| 
 | 
 | ||||||
| func (s *userServer) ProcessUserCommandGet(ctx context.Context, req *pbuser.ProcessUserCommandGetReq) (*pbuser.ProcessUserCommandGetResp, error) { | func (s *userServer) ProcessUserCommandGet(ctx context.Context, req *pbuser.ProcessUserCommandGetReq) (*pbuser.ProcessUserCommandGetResp, error) { | ||||||
| 
 | 
 | ||||||
| 	err := authverify.CheckAccessV3(ctx, req.UserID, s.config.Share.IMAdminUserID) | 	err := authverify.CheckAccess(ctx, req.UserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -448,7 +467,7 @@ func (s *userServer) ProcessUserCommandGet(ctx context.Context, req *pbuser.Proc | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *userServer) ProcessUserCommandGetAll(ctx context.Context, req *pbuser.ProcessUserCommandGetAllReq) (*pbuser.ProcessUserCommandGetAllResp, error) { | func (s *userServer) ProcessUserCommandGetAll(ctx context.Context, req *pbuser.ProcessUserCommandGetAllReq) (*pbuser.ProcessUserCommandGetAllResp, error) { | ||||||
| 	err := authverify.CheckAccessV3(ctx, req.UserID, s.config.Share.IMAdminUserID) | 	err := authverify.CheckAccess(ctx, req.UserID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| @ -477,7 +496,7 @@ func (s *userServer) ProcessUserCommandGetAll(ctx context.Context, req *pbuser.P | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *userServer) AddNotificationAccount(ctx context.Context, req *pbuser.AddNotificationAccountReq) (*pbuser.AddNotificationAccountResp, error) { | func (s *userServer) AddNotificationAccount(ctx context.Context, req *pbuser.AddNotificationAccountReq) (*pbuser.AddNotificationAccountResp, error) { | ||||||
| 	if err := authverify.CheckAdmin(ctx, s.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAdmin(ctx); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	if req.AppMangerLevel < constant.AppNotificationAdmin { | 	if req.AppMangerLevel < constant.AppNotificationAdmin { | ||||||
| @ -523,7 +542,7 @@ func (s *userServer) AddNotificationAccount(ctx context.Context, req *pbuser.Add | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *userServer) UpdateNotificationAccountInfo(ctx context.Context, req *pbuser.UpdateNotificationAccountInfoReq) (*pbuser.UpdateNotificationAccountInfoResp, error) { | func (s *userServer) UpdateNotificationAccountInfo(ctx context.Context, req *pbuser.UpdateNotificationAccountInfoReq) (*pbuser.UpdateNotificationAccountInfoResp, error) { | ||||||
| 	if err := authverify.CheckAdmin(ctx, s.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAdmin(ctx); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -550,7 +569,7 @@ func (s *userServer) UpdateNotificationAccountInfo(ctx context.Context, req *pbu | |||||||
| 
 | 
 | ||||||
| func (s *userServer) SearchNotificationAccount(ctx context.Context, req *pbuser.SearchNotificationAccountReq) (*pbuser.SearchNotificationAccountResp, error) { | func (s *userServer) SearchNotificationAccount(ctx context.Context, req *pbuser.SearchNotificationAccountReq) (*pbuser.SearchNotificationAccountResp, error) { | ||||||
| 	// Check if user is an admin | 	// Check if user is an admin | ||||||
| 	if err := authverify.CheckAdmin(ctx, s.config.Share.IMAdminUserID); err != nil { | 	if err := authverify.CheckAdmin(ctx); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -605,7 +624,7 @@ func (s *userServer) GetNotificationAccount(ctx context.Context, req *pbuser.Get | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, servererrs.ErrUserIDNotFound.Wrap() | 		return nil, servererrs.ErrUserIDNotFound.Wrap() | ||||||
| 	} | 	} | ||||||
| 	if user.AppMangerLevel == constant.AppAdmin || user.AppMangerLevel >= constant.AppNotificationAdmin { | 	if user.AppMangerLevel >= constant.AppAdmin { | ||||||
| 		return &pbuser.GetNotificationAccountResp{Account: &pbuser.NotificationAccountInfo{ | 		return &pbuser.GetNotificationAccountResp{Account: &pbuser.NotificationAccountInfo{ | ||||||
| 			UserID:         user.UserID, | 			UserID:         user.UserID, | ||||||
| 			FaceURL:        user.FaceURL, | 			FaceURL:        user.FaceURL, | ||||||
| @ -636,7 +655,7 @@ func (s *userServer) userModelToResp(users []*tablerelation.User, pagination pag | |||||||
| 	accounts := make([]*pbuser.NotificationAccountInfo, 0) | 	accounts := make([]*pbuser.NotificationAccountInfo, 0) | ||||||
| 	var total int64 | 	var total int64 | ||||||
| 	for _, v := range users { | 	for _, v := range users { | ||||||
| 		if v.AppMangerLevel >= constant.AppNotificationAdmin && !datautil.Contain(v.UserID, s.config.Share.IMAdminUserID...) { | 		if v.AppMangerLevel >= constant.AppNotificationAdmin && !datautil.Contain(v.UserID, s.adminUserIDs...) { | ||||||
| 			if appManagerLevel != nil { | 			if appManagerLevel != nil { | ||||||
| 				if v.AppMangerLevel != *appManagerLevel { | 				if v.AppMangerLevel != *appManagerLevel { | ||||||
| 					continue | 					continue | ||||||
|  | |||||||
| @ -10,7 +10,6 @@ import ( | |||||||
| 	"github.com/openimsdk/protocol/third" | 	"github.com/openimsdk/protocol/third" | ||||||
| 	"github.com/openimsdk/tools/discovery" | 	"github.com/openimsdk/tools/discovery" | ||||||
| 	"github.com/openimsdk/tools/discovery/etcd" | 	"github.com/openimsdk/tools/discovery/etcd" | ||||||
| 
 |  | ||||||
| 	"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" | ||||||
| @ -25,14 +24,14 @@ type Config struct { | |||||||
| 	Discovery config.Discovery | 	Discovery config.Discovery | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func Start(ctx context.Context, conf *Config, client discovery.Conn, service grpc.ServiceRegistrar) error { | func Start(ctx context.Context, conf *Config, client discovery.SvcDiscoveryRegistry, service grpc.ServiceRegistrar) error { | ||||||
| 	log.CInfo(ctx, "CRON-TASK server is initializing", "runTimeEnv", runtimeenv.RuntimeEnvironment(), "chatRecordsClearTime", conf.CronTask.CronExecuteTime, "msgDestructTime", conf.CronTask.RetainChatRecords) | 	log.CInfo(ctx, "CRON-TASK server is initializing", "runTimeEnv", runtimeenv.RuntimeEnvironment(), "chatRecordsClearTime", conf.CronTask.CronExecuteTime, "msgDestructTime", conf.CronTask.RetainChatRecords) | ||||||
| 	if conf.CronTask.RetainChatRecords < 1 { | 	if conf.CronTask.RetainChatRecords < 1 { | ||||||
| 		log.ZInfo(ctx, "disable cron") | 		log.ZInfo(ctx, "disable cron") | ||||||
| 		<-ctx.Done() | 		<-ctx.Done() | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| 	ctx = mcontext.SetOpUserID(ctx, conf.Share.IMAdminUserID[0]) | 	ctx = mcontext.SetOpUserID(ctx, conf.Share.IMAdminUser.UserIDs[0]) | ||||||
| 
 | 
 | ||||||
| 	msgConn, err := client.GetConn(ctx, conf.Discovery.RpcService.Msg) | 	msgConn, err := client.GetConn(ctx, conf.Discovery.RpcService.Msg) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -49,6 +48,7 @@ func Start(ctx context.Context, conf *Config, client discovery.Conn, service grp | |||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	var locker Locker | ||||||
| 	if conf.Discovery.Enable == config.ETCD { | 	if conf.Discovery.Enable == config.ETCD { | ||||||
| 		cm := disetcd.NewConfigManager(client.(*etcd.SvcDiscoveryRegistryImpl).GetClient(), []string{ | 		cm := disetcd.NewConfigManager(client.(*etcd.SvcDiscoveryRegistryImpl).GetClient(), []string{ | ||||||
| 			conf.CronTask.GetConfigFileName(), | 			conf.CronTask.GetConfigFileName(), | ||||||
| @ -56,6 +56,14 @@ func Start(ctx context.Context, conf *Config, client discovery.Conn, service grp | |||||||
| 			conf.Discovery.GetConfigFileName(), | 			conf.Discovery.GetConfigFileName(), | ||||||
| 		}) | 		}) | ||||||
| 		cm.Watch(ctx) | 		cm.Watch(ctx) | ||||||
|  | 		locker, err = NewEtcdLocker(client.(*etcd.SvcDiscoveryRegistryImpl).GetClient()) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if locker == nil { | ||||||
|  | 		locker = emptyLocker{} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	srv := &cronServer{ | 	srv := &cronServer{ | ||||||
| @ -65,6 +73,7 @@ func Start(ctx context.Context, conf *Config, client discovery.Conn, service grp | |||||||
| 		msgClient:          msg.NewMsgClient(msgConn), | 		msgClient:          msg.NewMsgClient(msgConn), | ||||||
| 		conversationClient: pbconversation.NewConversationClient(conversationConn), | 		conversationClient: pbconversation.NewConversationClient(conversationConn), | ||||||
| 		thirdClient:        third.NewThirdClient(thirdConn), | 		thirdClient:        third.NewThirdClient(thirdConn), | ||||||
|  | 		locker:             locker, | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if err := srv.registerClearS3(); err != nil { | 	if err := srv.registerClearS3(); err != nil { | ||||||
| @ -81,9 +90,21 @@ func Start(ctx context.Context, conf *Config, client discovery.Conn, service grp | |||||||
| 	log.ZDebug(ctx, "cron task server is running") | 	log.ZDebug(ctx, "cron task server is running") | ||||||
| 	<-ctx.Done() | 	<-ctx.Done() | ||||||
| 	log.ZDebug(ctx, "cron task server is shutting down") | 	log.ZDebug(ctx, "cron task server is shutting down") | ||||||
|  | 	srv.cron.Stop() | ||||||
|  | 
 | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | type Locker interface { | ||||||
|  | 	ExecuteWithLock(ctx context.Context, taskName string, task func()) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type emptyLocker struct{} | ||||||
|  | 
 | ||||||
|  | func (emptyLocker) ExecuteWithLock(ctx context.Context, taskName string, task func()) { | ||||||
|  | 	task() | ||||||
|  | } | ||||||
|  | 
 | ||||||
| type cronServer struct { | type cronServer struct { | ||||||
| 	ctx                context.Context | 	ctx                context.Context | ||||||
| 	config             *Config | 	config             *Config | ||||||
| @ -91,6 +112,7 @@ type cronServer struct { | |||||||
| 	msgClient          msg.MsgClient | 	msgClient          msg.MsgClient | ||||||
| 	conversationClient pbconversation.ConversationClient | 	conversationClient pbconversation.ConversationClient | ||||||
| 	thirdClient        third.ThirdClient | 	thirdClient        third.ThirdClient | ||||||
|  | 	locker             Locker | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *cronServer) registerClearS3() error { | func (c *cronServer) registerClearS3() error { | ||||||
| @ -98,7 +120,9 @@ func (c *cronServer) registerClearS3() error { | |||||||
| 		log.ZInfo(c.ctx, "disable scheduled cleanup of s3", "fileExpireTime", c.config.CronTask.FileExpireTime, "deleteObjectType", c.config.CronTask.DeleteObjectType) | 		log.ZInfo(c.ctx, "disable scheduled cleanup of s3", "fileExpireTime", c.config.CronTask.FileExpireTime, "deleteObjectType", c.config.CronTask.DeleteObjectType) | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| 	_, err := c.cron.AddFunc(c.config.CronTask.CronExecuteTime, c.clearS3) | 	_, err := c.cron.AddFunc(c.config.CronTask.CronExecuteTime, func() { | ||||||
|  | 		c.locker.ExecuteWithLock(c.ctx, "clearS3", c.clearS3) | ||||||
|  | 	}) | ||||||
| 	return errs.WrapMsg(err, "failed to register clear s3 cron task") | 	return errs.WrapMsg(err, "failed to register clear s3 cron task") | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -107,11 +131,15 @@ func (c *cronServer) registerDeleteMsg() error { | |||||||
| 		log.ZInfo(c.ctx, "disable scheduled cleanup of chat records", "retainChatRecords", c.config.CronTask.RetainChatRecords) | 		log.ZInfo(c.ctx, "disable scheduled cleanup of chat records", "retainChatRecords", c.config.CronTask.RetainChatRecords) | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| 	_, err := c.cron.AddFunc(c.config.CronTask.CronExecuteTime, c.deleteMsg) | 	_, err := c.cron.AddFunc(c.config.CronTask.CronExecuteTime, func() { | ||||||
|  | 		c.locker.ExecuteWithLock(c.ctx, "deleteMsg", c.deleteMsg) | ||||||
|  | 	}) | ||||||
| 	return errs.WrapMsg(err, "failed to register delete msg cron task") | 	return errs.WrapMsg(err, "failed to register delete msg cron task") | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (c *cronServer) registerClearUserMsg() error { | func (c *cronServer) registerClearUserMsg() error { | ||||||
| 	_, err := c.cron.AddFunc(c.config.CronTask.CronExecuteTime, c.clearUserMsg) | 	_, err := c.cron.AddFunc(c.config.CronTask.CronExecuteTime, func() { | ||||||
|  | 		c.locker.ExecuteWithLock(c.ctx, "clearUserMsg", c.clearUserMsg) | ||||||
|  | 	}) | ||||||
| 	return errs.WrapMsg(err, "failed to register clear user msg cron task") | 	return errs.WrapMsg(err, "failed to register clear user msg cron task") | ||||||
| } | } | ||||||
|  | |||||||
| @ -46,7 +46,7 @@ func TestName(t *testing.T) { | |||||||
| 
 | 
 | ||||||
| 	srv := &cronServer{ | 	srv := &cronServer{ | ||||||
| 		ctx: ctx, | 		ctx: ctx, | ||||||
| 		config: &CronTaskConfig{ | 		config: &Config{ | ||||||
| 			CronTask: config.CronTask{ | 			CronTask: config.CronTask{ | ||||||
| 				RetainChatRecords: 1, | 				RetainChatRecords: 1, | ||||||
| 				FileExpireTime:    1, | 				FileExpireTime:    1, | ||||||
|  | |||||||
							
								
								
									
										86
									
								
								internal/tools/cron/dist_look.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										86
									
								
								internal/tools/cron/dist_look.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,86 @@ | |||||||
|  | package cron | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"context" | ||||||
|  | 	"fmt" | ||||||
|  | 	"os" | ||||||
|  | 	"time" | ||||||
|  | 
 | ||||||
|  | 	"github.com/openimsdk/tools/log" | ||||||
|  | 	clientv3 "go.etcd.io/etcd/client/v3" | ||||||
|  | 	"go.etcd.io/etcd/client/v3/concurrency" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | const ( | ||||||
|  | 	lockLeaseTTL = 300 | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | type EtcdLocker struct { | ||||||
|  | 	client     *clientv3.Client | ||||||
|  | 	instanceID string | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // NewEtcdLocker creates a new etcd distributed lock | ||||||
|  | func NewEtcdLocker(client *clientv3.Client) (*EtcdLocker, error) { | ||||||
|  | 	hostname, _ := os.Hostname() | ||||||
|  | 	pid := os.Getpid() | ||||||
|  | 	instanceID := fmt.Sprintf("%s-pid-%d-%d", hostname, pid, time.Now().UnixNano()) | ||||||
|  | 
 | ||||||
|  | 	locker := &EtcdLocker{ | ||||||
|  | 		client:     client, | ||||||
|  | 		instanceID: instanceID, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return locker, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (e *EtcdLocker) ExecuteWithLock(ctx context.Context, taskName string, task func()) { | ||||||
|  | 	session, err := concurrency.NewSession(e.client, concurrency.WithTTL(lockLeaseTTL)) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.ZWarn(ctx, "Failed to create etcd session", err, | ||||||
|  | 			"taskName", taskName, | ||||||
|  | 			"instanceID", e.instanceID) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	defer session.Close() | ||||||
|  | 
 | ||||||
|  | 	lockKey := fmt.Sprintf("openim/crontask/%s", taskName) | ||||||
|  | 	mutex := concurrency.NewMutex(session, lockKey) | ||||||
|  | 
 | ||||||
|  | 	ctxWithTimeout, cancel := context.WithTimeout(ctx, 100*time.Millisecond) | ||||||
|  | 	defer cancel() | ||||||
|  | 
 | ||||||
|  | 	err = mutex.TryLock(ctxWithTimeout) | ||||||
|  | 	if err != nil { | ||||||
|  | 		// errors.Is(err, concurrency.ErrLocked) | ||||||
|  | 		log.ZDebug(ctx, "Task is being executed by another instance, skipping", | ||||||
|  | 			"taskName", taskName, | ||||||
|  | 			"instanceID", e.instanceID, | ||||||
|  | 			"error", err.Error()) | ||||||
|  | 
 | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	defer func() { | ||||||
|  | 		if err := mutex.Unlock(ctx); err != nil { | ||||||
|  | 			log.ZWarn(ctx, "Failed to release task lock", err, | ||||||
|  | 				"taskName", taskName, | ||||||
|  | 				"instanceID", e.instanceID) | ||||||
|  | 		} else { | ||||||
|  | 			log.ZInfo(ctx, "Successfully released task lock", | ||||||
|  | 				"taskName", taskName, | ||||||
|  | 				"instanceID", e.instanceID) | ||||||
|  | 		} | ||||||
|  | 	}() | ||||||
|  | 
 | ||||||
|  | 	log.ZInfo(ctx, "Successfully acquired task lock, starting execution", | ||||||
|  | 		"taskName", taskName, | ||||||
|  | 		"instanceID", e.instanceID, | ||||||
|  | 		"sessionID", session.Lease()) | ||||||
|  | 
 | ||||||
|  | 	task() | ||||||
|  | 
 | ||||||
|  | 	log.ZInfo(ctx, "Task execution completed", | ||||||
|  | 		"taskName", taskName, | ||||||
|  | 		"instanceID", e.instanceID) | ||||||
|  | } | ||||||
							
								
								
									
										69
									
								
								magefile.go
									
									
									
									
									
								
							
							
						
						
									
										69
									
								
								magefile.go
									
									
									
									
									
								
							| @ -12,15 +12,48 @@ import ( | |||||||
| 
 | 
 | ||||||
| var Default = Build | var Default = Build | ||||||
| 
 | 
 | ||||||
|  | var Aliases = map[string]any{ | ||||||
|  | 	"buildcc": BuildWithCustomConfig, | ||||||
|  | 	"startcc": StartWithCustomConfig, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var ( | ||||||
|  | 	customRootDir   = "."       // workDir in mage, default is "./"(project root directory) | ||||||
|  | 	customSrcDir    = "cmd"     // source code directory, default is "cmd" | ||||||
|  | 	customOutputDir = "_output" // output directory, default is "_output" | ||||||
|  | 	customConfigDir = "config"  // configuration directory, default is "config" | ||||||
|  | 	customToolsDir  = "tools"   // tools source code directory, default is "tools" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | // Build support specifical binary build. | ||||||
|  | // | ||||||
|  | // Example: `mage build openim-api openim-rpc-user seq` | ||||||
| func Build() { | func Build() { | ||||||
| 	flag.Parse() | 	flag.Parse() | ||||||
| 
 |  | ||||||
| 	bin := flag.Args() | 	bin := flag.Args() | ||||||
| 	if len(bin) != 0 { | 	if len(bin) != 0 { | ||||||
| 		bin = bin[1:] | 		bin = bin[1:] | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	mageutil.Build(bin) | 	mageutil.Build(bin, nil) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func BuildWithCustomConfig() { | ||||||
|  | 	flag.Parse() | ||||||
|  | 	bin := flag.Args() | ||||||
|  | 	if len(bin) != 0 { | ||||||
|  | 		bin = bin[1:] | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	config := &mageutil.PathOptions{ | ||||||
|  | 		RootDir:   &customRootDir, | ||||||
|  | 		OutputDir: &customOutputDir, | ||||||
|  | 		SrcDir:    &customSrcDir, | ||||||
|  | 		ToolsDir:  &customToolsDir, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	mageutil.Build(bin, config) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func Start() { | func Start() { | ||||||
| @ -30,7 +63,37 @@ func Start() { | |||||||
| 		mageutil.PrintRed("setMaxOpenFiles failed " + err.Error()) | 		mageutil.PrintRed("setMaxOpenFiles failed " + err.Error()) | ||||||
| 		os.Exit(1) | 		os.Exit(1) | ||||||
| 	} | 	} | ||||||
| 	mageutil.StartToolsAndServices() | 
 | ||||||
|  | 	flag.Parse() | ||||||
|  | 	bin := flag.Args() | ||||||
|  | 	if len(bin) != 0 { | ||||||
|  | 		bin = bin[1:] | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	mageutil.StartToolsAndServices(bin, nil) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func StartWithCustomConfig() { | ||||||
|  | 	mageutil.InitForSSC() | ||||||
|  | 	err := setMaxOpenFiles() | ||||||
|  | 	if err != nil { | ||||||
|  | 		mageutil.PrintRed("setMaxOpenFiles failed " + err.Error()) | ||||||
|  | 		os.Exit(1) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	flag.Parse() | ||||||
|  | 	bin := flag.Args() | ||||||
|  | 	if len(bin) != 0 { | ||||||
|  | 		bin = bin[1:] | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	config := &mageutil.PathOptions{ | ||||||
|  | 		RootDir:   &customRootDir, | ||||||
|  | 		OutputDir: &customOutputDir, | ||||||
|  | 		ConfigDir: &customConfigDir, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	mageutil.StartToolsAndServices(bin, config) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func Stop() { | func Stop() { | ||||||
|  | |||||||
| @ -15,6 +15,10 @@ type SetConfigReq struct { | |||||||
| 	Data       string `json:"data"` | 	Data       string `json:"data"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | type SetConfigsReq struct { | ||||||
|  | 	Configs []SetConfigReq `json:"configs"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
| type SetEnableConfigManagerReq struct { | type SetEnableConfigManagerReq struct { | ||||||
| 	Enable bool `json:"enable"` | 	Enable bool `json:"enable"` | ||||||
| } | } | ||||||
|  | |||||||
| @ -15,6 +15,7 @@ | |||||||
| package apistruct | package apistruct | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
|  | 	pbmsg "github.com/openimsdk/protocol/msg" | ||||||
| 	"github.com/openimsdk/protocol/sdkws" | 	"github.com/openimsdk/protocol/sdkws" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| @ -139,4 +140,15 @@ type SingleReturnResult struct { | |||||||
| 
 | 
 | ||||||
| 	// RecvID uniquely identifies the receiver of the message. | 	// RecvID uniquely identifies the receiver of the message. | ||||||
| 	RecvID string `json:"recvID"` | 	RecvID string `json:"recvID"` | ||||||
|  | 
 | ||||||
|  | 	// Modify fields modified via webhook. | ||||||
|  | 	Modify map[string]any `json:"modify,omitempty"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type SendMsgResp struct { | ||||||
|  | 	// SendMsgResp original response. | ||||||
|  | 	*pbmsg.SendMsgResp | ||||||
|  | 
 | ||||||
|  | 	// Modify fields modified via webhook. | ||||||
|  | 	Modify map[string]any `json:"modify,omitempty"` | ||||||
| } | } | ||||||
|  | |||||||
| @ -1 +0,0 @@ | |||||||
| package apistruct |  | ||||||
| @ -20,6 +20,7 @@ import ( | |||||||
| 
 | 
 | ||||||
| 	"github.com/golang-jwt/jwt/v4" | 	"github.com/golang-jwt/jwt/v4" | ||||||
| 	"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/tools/mcontext" | 	"github.com/openimsdk/tools/mcontext" | ||||||
| 	"github.com/openimsdk/tools/utils/datautil" | 	"github.com/openimsdk/tools/utils/datautil" | ||||||
| ) | ) | ||||||
| @ -30,28 +31,90 @@ func Secret(secret string) jwt.Keyfunc { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func CheckAccessV3(ctx context.Context, ownerUserID string, imAdminUserID []string) (err error) { | func CheckAdmin(ctx context.Context) error { | ||||||
| 	opUserID := mcontext.GetOpUserID(ctx) | 	if IsAdmin(ctx) { | ||||||
| 	if datautil.Contain(opUserID, imAdminUserID...) { |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
| 	if opUserID == ownerUserID { |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
| 	return servererrs.ErrNoPermission.WrapMsg("ownerUserID", ownerUserID) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func IsAppManagerUid(ctx context.Context, imAdminUserID []string) bool { |  | ||||||
| 	return datautil.Contain(mcontext.GetOpUserID(ctx), imAdminUserID...) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func CheckAdmin(ctx context.Context, imAdminUserID []string) error { |  | ||||||
| 	if datautil.Contain(mcontext.GetOpUserID(ctx), imAdminUserID...) { |  | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| 	return servererrs.ErrNoPermission.WrapMsg(fmt.Sprintf("user %s is not admin userID", mcontext.GetOpUserID(ctx))) | 	return servererrs.ErrNoPermission.WrapMsg(fmt.Sprintf("user %s is not admin userID", mcontext.GetOpUserID(ctx))) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func IsManagerUserID(opUserID string, imAdminUserID []string) bool { | //func IsManagerUserID(opUserID string, imAdminUserID []string) bool { | ||||||
| 	return datautil.Contain(opUserID, imAdminUserID...) | //	return datautil.Contain(opUserID, imAdminUserID...) | ||||||
|  | //} | ||||||
|  | 
 | ||||||
|  | func CheckUserIsAdmin(ctx context.Context, userID string) bool { | ||||||
|  | 	return datautil.Contain(userID, GetIMAdminUserIDs(ctx)...) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func CheckSystemAccount(ctx context.Context, level int32) bool { | ||||||
|  | 	return level >= constant.AppAdmin | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const ( | ||||||
|  | 	CtxAdminUserIDsKey = "CtxAdminUserIDsKey" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func WithIMAdminUserIDs(ctx context.Context, imAdminUserID []string) context.Context { | ||||||
|  | 	return context.WithValue(ctx, CtxAdminUserIDsKey, imAdminUserID) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func GetIMAdminUserIDs(ctx context.Context) []string { | ||||||
|  | 	imAdminUserID, _ := ctx.Value(CtxAdminUserIDsKey).([]string) | ||||||
|  | 	return imAdminUserID | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func IsAdmin(ctx context.Context) bool { | ||||||
|  | 	return IsTempAdmin(ctx) || IsSystemAdmin(ctx) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func CheckAccess(ctx context.Context, ownerUserID string) error { | ||||||
|  | 	if mcontext.GetOpUserID(ctx) == ownerUserID { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	if IsAdmin(ctx) { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	return servererrs.ErrNoPermission.WrapMsg("ownerUserID", ownerUserID) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func CheckAccessIn(ctx context.Context, ownerUserIDs ...string) error { | ||||||
|  | 	opUserID := mcontext.GetOpUserID(ctx) | ||||||
|  | 	for _, userID := range ownerUserIDs { | ||||||
|  | 		if opUserID == userID { | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	if IsAdmin(ctx) { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	return servererrs.ErrNoPermission.WrapMsg("opUser in ownerUserIDs") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var tempAdminValue = []string{"1"} | ||||||
|  | 
 | ||||||
|  | const ctxTempAdminKey = "ctxImTempAdminKey" | ||||||
|  | 
 | ||||||
|  | func WithTempAdmin(ctx context.Context) context.Context { | ||||||
|  | 	keys, _ := ctx.Value(constant.RpcCustomHeader).([]string) | ||||||
|  | 	if datautil.Contain(ctxTempAdminKey, keys...) { | ||||||
|  | 		return ctx | ||||||
|  | 	} | ||||||
|  | 	if len(keys) > 0 { | ||||||
|  | 		temp := make([]string, 0, len(keys)+1) | ||||||
|  | 		temp = append(temp, keys...) | ||||||
|  | 		keys = append(temp, ctxTempAdminKey) | ||||||
|  | 	} else { | ||||||
|  | 		keys = []string{ctxTempAdminKey} | ||||||
|  | 	} | ||||||
|  | 	ctx = context.WithValue(ctx, constant.RpcCustomHeader, keys) | ||||||
|  | 	return context.WithValue(ctx, ctxTempAdminKey, tempAdminValue) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func IsTempAdmin(ctx context.Context) bool { | ||||||
|  | 	values, _ := ctx.Value(ctxTempAdminKey).([]string) | ||||||
|  | 	return datautil.Equal(tempAdminValue, values) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func IsSystemAdmin(ctx context.Context) bool { | ||||||
|  | 	return datautil.Contain(mcontext.GetOpUserID(ctx), GetIMAdminUserIDs(ctx)...) | ||||||
| } | } | ||||||
|  | |||||||
| @ -15,51 +15,55 @@ | |||||||
| package callbackstruct | package callbackstruct | ||||||
| 
 | 
 | ||||||
| const ( | const ( | ||||||
| 	CallbackBeforeInviteJoinGroupCommand    = "callbackBeforeInviteJoinGroupCommand" | 	CallbackBeforeInviteJoinGroupCommand               = "callbackBeforeInviteJoinGroupCommand" | ||||||
| 	CallbackAfterJoinGroupCommand           = "callbackAfterJoinGroupCommand" | 	CallbackAfterJoinGroupCommand                      = "callbackAfterJoinGroupCommand" | ||||||
| 	CallbackAfterSetGroupInfoCommand        = "callbackAfterSetGroupInfoCommand" | 	CallbackAfterSetGroupInfoCommand                   = "callbackAfterSetGroupInfoCommand" | ||||||
| 	CallbackAfterSetGroupInfoExCommand      = "callbackAfterSetGroupInfoExCommand" | 	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" | 	CallbackAfterAddFriendAgreeCommand                 = "callbackAfterAddFriendAgreeCommand" | ||||||
| 	CallbackAfterDeleteFriendCommand        = "callbackAfterDeleteFriendCommand" | 	CallbackAfterDeleteFriendCommand                   = "callbackAfterDeleteFriendCommand" | ||||||
| 	CallbackBeforeImportFriendsCommand      = "callbackBeforeImportFriendsCommand" | 	CallbackBeforeImportFriendsCommand                 = "callbackBeforeImportFriendsCommand" | ||||||
| 	CallbackAfterImportFriendsCommand       = "callbackAfterImportFriendsCommand" | 	CallbackAfterImportFriendsCommand                  = "callbackAfterImportFriendsCommand" | ||||||
| 	CallbackAfterRemoveBlackCommand         = "callbackAfterRemoveBlackCommand" | 	CallbackAfterRemoveBlackCommand                    = "callbackAfterRemoveBlackCommand" | ||||||
| 	CallbackAfterQuitGroupCommand           = "callbackAfterQuitGroupCommand" | 	CallbackAfterQuitGroupCommand                      = "callbackAfterQuitGroupCommand" | ||||||
| 	CallbackAfterKickGroupCommand           = "callbackAfterKickGroupCommand" | 	CallbackAfterKickGroupCommand                      = "callbackAfterKickGroupCommand" | ||||||
| 	CallbackAfterDisMissGroupCommand        = "callbackAfterDisMissGroupCommand" | 	CallbackAfterDisMissGroupCommand                   = "callbackAfterDisMissGroupCommand" | ||||||
| 	CallbackBeforeJoinGroupCommand          = "callbackBeforeJoinGroupCommand" | 	CallbackBeforeJoinGroupCommand                     = "callbackBeforeJoinGroupCommand" | ||||||
| 	CallbackAfterGroupMsgReadCommand        = "callbackAfterGroupMsgReadCommand" | 	CallbackAfterGroupMsgReadCommand                   = "callbackAfterGroupMsgReadCommand" | ||||||
| 	CallbackBeforeMsgModifyCommand          = "callbackBeforeMsgModifyCommand" | 	CallbackBeforeMsgModifyCommand                     = "callbackBeforeMsgModifyCommand" | ||||||
| 	CallbackAfterUpdateUserInfoCommand      = "callbackAfterUpdateUserInfoCommand" | 	CallbackAfterUpdateUserInfoCommand                 = "callbackAfterUpdateUserInfoCommand" | ||||||
| 	CallbackAfterUpdateUserInfoExCommand    = "callbackAfterUpdateUserInfoExCommand" | 	CallbackAfterUpdateUserInfoExCommand               = "callbackAfterUpdateUserInfoExCommand" | ||||||
| 	CallbackBeforeUpdateUserInfoExCommand   = "callbackBeforeUpdateUserInfoExCommand" | 	CallbackBeforeUpdateUserInfoExCommand              = "callbackBeforeUpdateUserInfoExCommand" | ||||||
| 	CallbackBeforeUserRegisterCommand       = "callbackBeforeUserRegisterCommand" | 	CallbackBeforeUserRegisterCommand                  = "callbackBeforeUserRegisterCommand" | ||||||
| 	CallbackAfterUserRegisterCommand        = "callbackAfterUserRegisterCommand" | 	CallbackAfterUserRegisterCommand                   = "callbackAfterUserRegisterCommand" | ||||||
| 	CallbackAfterTransferGroupOwnerCommand  = "callbackAfterTransferGroupOwnerCommand" | 	CallbackAfterTransferGroupOwnerCommand             = "callbackAfterTransferGroupOwnerCommand" | ||||||
| 	CallbackBeforeSetFriendRemarkCommand    = "callbackBeforeSetFriendRemarkCommand" | 	CallbackBeforeSetFriendRemarkCommand               = "callbackBeforeSetFriendRemarkCommand" | ||||||
| 	CallbackAfterSetFriendRemarkCommand     = "callbackAfterSetFriendRemarkCommand" | 	CallbackAfterSetFriendRemarkCommand                = "callbackAfterSetFriendRemarkCommand" | ||||||
| 	CallbackAfterSingleMsgReadCommand       = "callbackAfterSingleMsgReadCommand" | 	CallbackAfterSingleMsgReadCommand                  = "callbackAfterSingleMsgReadCommand" | ||||||
| 	CallbackBeforeSendSingleMsgCommand      = "callbackBeforeSendSingleMsgCommand" | 	CallbackBeforeSendSingleMsgCommand                 = "callbackBeforeSendSingleMsgCommand" | ||||||
| 	CallbackAfterSendSingleMsgCommand       = "callbackAfterSendSingleMsgCommand" | 	CallbackAfterSendSingleMsgCommand                  = "callbackAfterSendSingleMsgCommand" | ||||||
| 	CallbackBeforeSendGroupMsgCommand       = "callbackBeforeSendGroupMsgCommand" | 	CallbackBeforeSendGroupMsgCommand                  = "callbackBeforeSendGroupMsgCommand" | ||||||
| 	CallbackAfterSendGroupMsgCommand        = "callbackAfterSendGroupMsgCommand" | 	CallbackAfterSendGroupMsgCommand                   = "callbackAfterSendGroupMsgCommand" | ||||||
| 	CallbackAfterUserOnlineCommand          = "callbackAfterUserOnlineCommand" | 	CallbackAfterUserOnlineCommand                     = "callbackAfterUserOnlineCommand" | ||||||
| 	CallbackAfterUserOfflineCommand         = "callbackAfterUserOfflineCommand" | 	CallbackAfterUserOfflineCommand                    = "callbackAfterUserOfflineCommand" | ||||||
| 	CallbackAfterUserKickOffCommand         = "callbackAfterUserKickOffCommand" | 	CallbackAfterUserKickOffCommand                    = "callbackAfterUserKickOffCommand" | ||||||
| 	CallbackBeforeOfflinePushCommand        = "callbackBeforeOfflinePushCommand" | 	CallbackBeforeOfflinePushCommand                   = "callbackBeforeOfflinePushCommand" | ||||||
| 	CallbackBeforeOnlinePushCommand         = "callbackBeforeOnlinePushCommand" | 	CallbackBeforeOnlinePushCommand                    = "callbackBeforeOnlinePushCommand" | ||||||
| 	CallbackBeforeGroupOnlinePushCommand    = "callbackBeforeGroupOnlinePushCommand" | 	CallbackBeforeGroupOnlinePushCommand               = "callbackBeforeGroupOnlinePushCommand" | ||||||
| 	CallbackBeforeAddFriendCommand          = "callbackBeforeAddFriendCommand" | 	CallbackBeforeAddFriendCommand                     = "callbackBeforeAddFriendCommand" | ||||||
| 	CallbackBeforeUpdateUserInfoCommand     = "callbackBeforeUpdateUserInfoCommand" | 	CallbackBeforeUpdateUserInfoCommand                = "callbackBeforeUpdateUserInfoCommand" | ||||||
| 	CallbackBeforeCreateGroupCommand        = "callbackBeforeCreateGroupCommand" | 	CallbackBeforeCreateGroupCommand                   = "callbackBeforeCreateGroupCommand" | ||||||
| 	CallbackAfterCreateGroupCommand         = "callbackAfterCreateGroupCommand" | 	CallbackAfterCreateGroupCommand                    = "callbackAfterCreateGroupCommand" | ||||||
| 	CallbackBeforeMembersJoinGroupCommand   = "callbackBeforeMembersJoinGroupCommand" | 	CallbackBeforeMembersJoinGroupCommand              = "callbackBeforeMembersJoinGroupCommand" | ||||||
| 	CallbackBeforeSetGroupMemberInfoCommand = "callbackBeforeSetGroupMemberInfoCommand" | 	CallbackBeforeSetGroupMemberInfoCommand            = "callbackBeforeSetGroupMemberInfoCommand" | ||||||
| 	CallbackAfterSetGroupMemberInfoCommand  = "callbackAfterSetGroupMemberInfoCommand" | 	CallbackAfterSetGroupMemberInfoCommand             = "callbackAfterSetGroupMemberInfoCommand" | ||||||
|  | 	CallbackBeforeCreateSingleChatConversationsCommand = "callbackBeforeCreateSingleChatConversationsCommand" | ||||||
|  | 	CallbackAfterCreateSingleChatConversationsCommand  = "callbackAfterCreateSingleChatConversationsCommand" | ||||||
|  | 	CallbackBeforeCreateGroupChatConversationsCommand  = "callbackBeforeCreateGroupChatConversationsCommand" | ||||||
|  | 	CallbackAfterCreateGroupChatConversationsCommand   = "callbackAfterCreateGroupChatConversationsCommand" | ||||||
| ) | ) | ||||||
|  | |||||||
							
								
								
									
										91
									
								
								pkg/callbackstruct/conversation.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										91
									
								
								pkg/callbackstruct/conversation.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,91 @@ | |||||||
|  | package callbackstruct | ||||||
|  | 
 | ||||||
|  | type CallbackBeforeCreateSingleChatConversationsReq struct { | ||||||
|  | 	CallbackCommand  `json:"callbackCommand"` | ||||||
|  | 	OwnerUserID      string `json:"owner_user_id"` | ||||||
|  | 	ConversationID   string `json:"conversation_id"` | ||||||
|  | 	ConversationType int32  `json:"conversation_type"` | ||||||
|  | 	UserID           string `json:"user_id"` | ||||||
|  | 	RecvMsgOpt       int32  `json:"recv_msg_opt"` | ||||||
|  | 	IsPinned         bool   `json:"is_pinned"` | ||||||
|  | 	IsPrivateChat    bool   `json:"is_private_chat"` | ||||||
|  | 	BurnDuration     int32  `json:"burn_duration"` | ||||||
|  | 	GroupAtType      int32  `json:"group_at_type"` | ||||||
|  | 	AttachedInfo     string `json:"attached_info"` | ||||||
|  | 	Ex               string `json:"ex"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type CallbackBeforeCreateSingleChatConversationsResp struct { | ||||||
|  | 	CommonCallbackResp | ||||||
|  | 	RecvMsgOpt    *int32  `json:"recv_msg_opt"` | ||||||
|  | 	IsPinned      *bool   `json:"is_pinned"` | ||||||
|  | 	IsPrivateChat *bool   `json:"is_private_chat"` | ||||||
|  | 	BurnDuration  *int32  `json:"burn_duration"` | ||||||
|  | 	GroupAtType   *int32  `json:"group_at_type"` | ||||||
|  | 	AttachedInfo  *string `json:"attached_info"` | ||||||
|  | 	Ex            *string `json:"ex"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type CallbackAfterCreateSingleChatConversationsReq struct { | ||||||
|  | 	CallbackCommand  `json:"callbackCommand"` | ||||||
|  | 	OwnerUserID      string `json:"owner_user_id"` | ||||||
|  | 	ConversationID   string `json:"conversation_id"` | ||||||
|  | 	ConversationType int32  `json:"conversation_type"` | ||||||
|  | 	UserID           string `json:"user_id"` | ||||||
|  | 	RecvMsgOpt       int32  `json:"recv_msg_opt"` | ||||||
|  | 	IsPinned         bool   `json:"is_pinned"` | ||||||
|  | 	IsPrivateChat    bool   `json:"is_private_chat"` | ||||||
|  | 	BurnDuration     int32  `json:"burn_duration"` | ||||||
|  | 	GroupAtType      int32  `json:"group_at_type"` | ||||||
|  | 	AttachedInfo     string `json:"attached_info"` | ||||||
|  | 	Ex               string `json:"ex"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type CallbackAfterCreateSingleChatConversationsResp struct { | ||||||
|  | 	CommonCallbackResp | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type CallbackBeforeCreateGroupChatConversationsReq struct { | ||||||
|  | 	CallbackCommand  `json:"callbackCommand"` | ||||||
|  | 	OwnerUserID      string `json:"owner_user_id"` | ||||||
|  | 	ConversationID   string `json:"conversation_id"` | ||||||
|  | 	ConversationType int32  `json:"conversation_type"` | ||||||
|  | 	GroupID          string `json:"group_id"` | ||||||
|  | 	RecvMsgOpt       int32  `json:"recv_msg_opt"` | ||||||
|  | 	IsPinned         bool   `json:"is_pinned"` | ||||||
|  | 	IsPrivateChat    bool   `json:"is_private_chat"` | ||||||
|  | 	BurnDuration     int32  `json:"burn_duration"` | ||||||
|  | 	GroupAtType      int32  `json:"group_at_type"` | ||||||
|  | 	AttachedInfo     string `json:"attached_info"` | ||||||
|  | 	Ex               string `json:"ex"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type CallbackBeforeCreateGroupChatConversationsResp struct { | ||||||
|  | 	CommonCallbackResp | ||||||
|  | 	RecvMsgOpt    *int32  `json:"recv_msg_opt"` | ||||||
|  | 	IsPinned      *bool   `json:"is_pinned"` | ||||||
|  | 	IsPrivateChat *bool   `json:"is_private_chat"` | ||||||
|  | 	BurnDuration  *int32  `json:"burn_duration"` | ||||||
|  | 	GroupAtType   *int32  `json:"group_at_type"` | ||||||
|  | 	AttachedInfo  *string `json:"attached_info"` | ||||||
|  | 	Ex            *string `json:"ex"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type CallbackAfterCreateGroupChatConversationsReq struct { | ||||||
|  | 	CallbackCommand  `json:"callbackCommand"` | ||||||
|  | 	OwnerUserID      string `json:"owner_user_id"` | ||||||
|  | 	ConversationID   string `json:"conversation_id"` | ||||||
|  | 	ConversationType int32  `json:"conversation_type"` | ||||||
|  | 	GroupID          string `json:"group_id"` | ||||||
|  | 	RecvMsgOpt       int32  `json:"recv_msg_opt"` | ||||||
|  | 	IsPinned         bool   `json:"is_pinned"` | ||||||
|  | 	IsPrivateChat    bool   `json:"is_private_chat"` | ||||||
|  | 	BurnDuration     int32  `json:"burn_duration"` | ||||||
|  | 	GroupAtType      int32  `json:"group_at_type"` | ||||||
|  | 	AttachedInfo     string `json:"attached_info"` | ||||||
|  | 	Ex               string `json:"ex"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type CallbackAfterCreateGroupChatConversationsResp struct { | ||||||
|  | 	CommonCallbackResp | ||||||
|  | } | ||||||
| @ -19,6 +19,7 @@ import ( | |||||||
| 
 | 
 | ||||||
| 	"github.com/openimsdk/open-im-server/v3/internal/api" | 	"github.com/openimsdk/open-im-server/v3/internal/api" | ||||||
| 	"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/startrpc" | 	"github.com/openimsdk/open-im-server/v3/pkg/common/startrpc" | ||||||
| 	"github.com/openimsdk/open-im-server/v3/version" | 	"github.com/openimsdk/open-im-server/v3/version" | ||||||
| 	"github.com/openimsdk/tools/system/program" | 	"github.com/openimsdk/tools/system/program" | ||||||
| @ -84,7 +85,7 @@ func (a *ApiCmd) runE() error { | |||||||
| 		a.apiConfig.API.Api.ListenIP, "", | 		a.apiConfig.API.Api.ListenIP, "", | ||||||
| 		a.apiConfig.API.Prometheus.AutoSetPorts, | 		a.apiConfig.API.Prometheus.AutoSetPorts, | ||||||
| 		nil, int(a.apiConfig.Index), | 		nil, int(a.apiConfig.Index), | ||||||
| 		a.apiConfig.Discovery.RpcService.MessageGateway, | 		prommetrics.APIKeyName, | ||||||
| 		&a.apiConfig.Notification, | 		&a.apiConfig.Notification, | ||||||
| 		a.apiConfig, | 		a.apiConfig, | ||||||
| 		[]string{}, | 		[]string{}, | ||||||
|  | |||||||
| @ -41,6 +41,7 @@ func NewConversationRpcCmd() *ConversationRpcCmd { | |||||||
| 		config.MongodbConfigFileName:            &conversationConfig.MongodbConfig, | 		config.MongodbConfigFileName:            &conversationConfig.MongodbConfig, | ||||||
| 		config.ShareFileName:                    &conversationConfig.Share, | 		config.ShareFileName:                    &conversationConfig.Share, | ||||||
| 		config.NotificationFileName:             &conversationConfig.NotificationConfig, | 		config.NotificationFileName:             &conversationConfig.NotificationConfig, | ||||||
|  | 		config.WebhooksConfigFileName:           &conversationConfig.WebhooksConfig, | ||||||
| 		config.LocalCacheConfigFileName:         &conversationConfig.LocalCacheConfig, | 		config.LocalCacheConfigFileName:         &conversationConfig.LocalCacheConfig, | ||||||
| 		config.DiscoveryConfigFilename:          &conversationConfig.Discovery, | 		config.DiscoveryConfigFilename:          &conversationConfig.Discovery, | ||||||
| 	} | 	} | ||||||
| @ -67,6 +68,7 @@ func (a *ConversationRpcCmd) runE() error { | |||||||
| 			a.conversationConfig.NotificationConfig.GetConfigFileName(), | 			a.conversationConfig.NotificationConfig.GetConfigFileName(), | ||||||
| 			a.conversationConfig.Share.GetConfigFileName(), | 			a.conversationConfig.Share.GetConfigFileName(), | ||||||
| 			a.conversationConfig.LocalCacheConfig.GetConfigFileName(), | 			a.conversationConfig.LocalCacheConfig.GetConfigFileName(), | ||||||
|  | 			a.conversationConfig.WebhooksConfig.GetConfigFileName(), | ||||||
| 			a.conversationConfig.Discovery.GetConfigFileName(), | 			a.conversationConfig.Discovery.GetConfigFileName(), | ||||||
| 		}, nil, | 		}, nil, | ||||||
| 		conversation.Start) | 		conversation.Start) | ||||||
|  | |||||||
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