Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feat(Auth): Authorization by Env Cluster #5302

Open
wants to merge 26 commits into
base: master
Choose a base branch
from

Conversation

BlackBear2003
Copy link
Member

@BlackBear2003 BlackBear2003 commented Dec 27, 2024

What's the purpose of this PR

支持按集群给用户授予管理权限(修改、发布配置)

Which issue(s) this PR fixes:

Fixes:

Related

Brief changelog

see git commits

Follow this checklist to help us incorporate your contribution quickly and easily:

  • Read the Contributing Guide before making this pull request.
  • Write a pull request description that is detailed enough to understand what the pull request does, how, and why.
  • Write necessary unit tests to verify the code.
  • Run mvn clean test to make sure this pull request doesn't break anything.
  • Update the CHANGES log.

Summary by CodeRabbit

Based on the comprehensive changes, here are the updated release notes:

  • New Features

    • Added cluster-level permission management for namespaces.
    • Introduced granular role-based access control for cluster configurations.
    • Implemented new cluster management interface.
    • Added new endpoints for initializing and checking cluster namespace permissions.
    • Introduced a new AngularJS controller for managing user roles within a cluster namespace.
  • Improvements

    • Enhanced permission validation across multiple controllers.
    • Expanded role initialization service to support cluster-specific permissions.
    • Added more detailed localization for cluster management.
    • Improved error handling and user feedback mechanisms in role assignment.
  • User Interface

    • New cluster management page.
    • Role assignment interface for cluster namespaces.
    • Improved navigation for cluster-related operations.
  • Security

    • More precise permission checks incorporating cluster context.
    • New role types for modifying and releasing namespaces within clusters.

@dosubot dosubot bot added the size:XXL This PR changes 1000+ lines, ignoring generated files. label Dec 27, 2024
Copy link
Contributor

coderabbitai bot commented Dec 27, 2024

Walkthrough

This pull request introduces comprehensive enhancements to the Apollo configuration portal's permission management system, specifically focusing on cluster-level authorization. The changes enable more granular permission controls by introducing new role types and permission checks that consider cluster and environment contexts. Key modifications span across multiple components, including permission validators, role initialization services, controllers, and frontend interfaces.

Changes

File Change Summary
PermissionValidator.java Modified permission checking methods to include cluster context, added new methods for cluster namespace permissions
PermissionType.java Added new permission constants for cluster namespace modifications and releases
RoleType.java Introduced new role types for cluster namespace permissions
ConfigsImportController.java, ItemController.java, NamespaceBranchController.java, ReleaseController.java, CommitController.java, ConfigsExportController.java, NamespaceController.java, ReleaseHistoryController.java Updated @PreAuthorize annotations and permission validation logic to include cluster name
PermissionController.java Added new endpoints for managing cluster namespace permissions
RoleInitializationService.java, DefaultRoleInitializationService.java Introduced methods for initializing cluster namespace roles
ClusterNamespaceRolesAssignedUsers.java New class for managing users assigned to cluster namespace roles
Frontend Files Added new HTML templates and JavaScript controllers for cluster role management
AppService.java, ClusterService.java Updated to include role initialization logic during app and cluster creation
i18n/en.json, i18n/zh-CN.json Enhanced localization for cluster permission management
config.html Updated UI for cluster management
namespace-panel-directive.js, PermissionService.js Modified permission handling logic for clusters
ControllableAuthorizationConfiguration.java, SkipAuthorizationConfiguration.java, ItemControllerAuthIntegrationTest.java, PermissionControllerTest.java, RoleInitializationServiceTest.java, AppServiceTest.java Added or updated test classes and methods to validate new permission logic

Assessment against linked issues

Objective Addressed Explanation
Support cluster authorization by environment Implemented new permission types and role management for cluster-level permissions
Initialize cluster roles during creation Added initClusterNamespaceRoles method in role initialization services
Provide management interfaces for cluster permissions Created new controller methods and frontend interfaces for role assignment

Suggested labels

size:XL, lgtm

Suggested reviewers

  • nobodyiam

Poem

🐰 Cluster permissions dance and sway,
In Apollo's realm, a brand new way!
Roles and rights now finely tuned,
With cluster context, access is pruned.
Hop along the authorization trail! 🔐


Thank you for using CodeRabbit. We offer it for free to the OSS community and would appreciate your support in helping us grow. If you find it useful, would you consider giving us a shout-out on your favorite social media?

❤️ Share
🪧 Tips

Chat

There are 3 ways to chat with CodeRabbit:

  • Review comments: Directly reply to a review comment made by CodeRabbit. Example:
    • I pushed a fix in commit <commit_id>, please review it.
    • Generate unit testing code for this file.
    • Open a follow-up GitHub issue for this discussion.
  • Files and specific lines of code (under the "Files changed" tab): Tag @coderabbitai in a new review comment at the desired location with your query. Examples:
    • @coderabbitai generate unit testing code for this file.
    • @coderabbitai modularize this function.
  • PR comments: Tag @coderabbitai in a new PR comment to ask questions about the PR branch. For the best results, please provide a very specific query, as very limited context is provided in this mode. Examples:
    • @coderabbitai gather interesting stats about this repository and render them as a table. Additionally, render a pie chart showing the language distribution in the codebase.
    • @coderabbitai read src/utils.ts and generate unit testing code.
    • @coderabbitai read the files in the src/scheduler package and generate a class diagram using mermaid and a README in the markdown format.
    • @coderabbitai help me debug CodeRabbit configuration file.

Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments.

CodeRabbit Commands (Invoked using PR comments)

  • @coderabbitai pause to pause the reviews on a PR.
  • @coderabbitai resume to resume the paused reviews.
  • @coderabbitai review to trigger an incremental review. This is useful when automatic reviews are disabled for the repository.
  • @coderabbitai full review to do a full review from scratch and review all the files again.
  • @coderabbitai summary to regenerate the summary of the PR.
  • @coderabbitai generate docstrings to generate docstrings for this PR. (Beta)
  • @coderabbitai resolve resolve all the CodeRabbit review comments.
  • @coderabbitai configuration to show the current CodeRabbit configuration for the repository.
  • @coderabbitai help to get help.

Other keywords and placeholders

  • Add @coderabbitai ignore anywhere in the PR description to prevent this PR from being reviewed.
  • Add @coderabbitai summary to generate the high-level summary at a specific location in the PR description.
  • Add @coderabbitai anywhere in the PR title to generate the title automatically.

CodeRabbit Configuration File (.coderabbit.yaml)

  • You can programmatically configure CodeRabbit by adding a .coderabbit.yaml file to the root of your repository.
  • Please see the configuration documentation for more information.
  • If your editor has YAML language server enabled, you can add the path at the top of this file to enable auto-completion and validation: # yaml-language-server: $schema=https://coderabbit.ai/integrations/schema.v2.json

Documentation and Community

  • Visit our Documentation for detailed information on how to use CodeRabbit.
  • Join our Discord Community to get help, request features, and share feedback.
  • Follow us on X/Twitter for updates and announcements.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 2

🧹 Nitpick comments (20)
apollo-portal/src/test/java/com/ctrip/framework/apollo/ControllableAuthorizationConfiguration.java (1)

44-45: Hard-coded username for permission checks.
Relying on "luke" as a special-cased user is fine for a single scenario, but consider parameterizing or documenting this for clarity and future extension to different test scenarios.

- when(mock.userHasPermission(eq("luke"), any(), any())).thenReturn(true);
+ // Optionally, read from a property or variable, allowing different user test scenarios
apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/entity/vo/ClusterRolesAssignedUsers.java (3)

24-28: Consider using final or a constructor for immutability
Currently, fields can be freely reassigned. Making them final and initializing them through a constructor can reinforce immutability. This is a minor suggestion that can help avoid unintended reassignments.


54-60: Getter/Setter for modifyRoleUsers
If you want to avoid accidental external modifications, consider returning and storing unmodifiable sets or defensive copies.

- public void setModifyRoleUsers(Set<UserInfo> modifyRoleUsers) {
-   this.modifyRoleUsers = modifyRoleUsers;
- }
+ public void setModifyRoleUsers(Set<UserInfo> modifyRoleUsers) {
+   this.modifyRoleUsers = modifyRoleUsers == null
+       ? Collections.emptySet()
+       : Collections.unmodifiableSet(new HashSet<>(modifyRoleUsers));
+ }

62-68: Getter/Setter for releaseRoleUsers
The same consideration for immutability applies here if that aligns with the design approach.

apollo-portal/src/main/resources/static/scripts/directive/namespace-panel-directive.js (1)

280-290: Consolidate repeated permission-checking logic

Within these lines, the permission checks for both namespace and namespace.branch use nearly identical expressions for setting hasModifyPermission. Consider extracting this repetition into a helper function or a shared utility to improve code maintainability.

apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/ItemControllerAuthIntegrationTest.java (1)

71-75: **Consider externalizing test profile activation **

Setting the active Spring profile programmatically within the test may mask potential environment inconsistencies. It's often preferable to configure the test profile via configuration files or command-line properties for clearer separation of concerns.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/PermissionController.java (2)

84-88: Validate environment for consistency.
Unlike some other methods in this controller (e.g., getClusterRoles), initClusterPermission does not explicitly validate env. Consider adding a similar check for Env.UNKNOWN or invalid environment formats for consistency and safety.


215-237: Cluster role retrieval logic is clear.
This endpoint neatly validates env and fetches release/modify role users. The approach is consistent with existing naming and usage patterns. Consider extracting the environment validation into a small helper function to reduce duplicate code blocks across multiple endpoints.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/RoleInitializationService.java (1)

36-37: Consider adding Javadoc.
The initClusterRoles method is clear in intent but lacks any Javadoc/explanatory comments describing preconditions, side effects, or error handling. Adding brief documentation can help future maintainers.

apollo-portal/src/test/java/com/ctrip/framework/apollo/SkipAuthorizationConfiguration.java (1)

68-68: Expand test coverage for negative scenarios.
Mocking hasAssignRolePermission(...) to always return true can mask potential authorization bugs. While this configuration may be intentional for integration tests, consider adding complementary tests to verify scenarios when permissions are absent or restricted.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/ClusterService.java (1)

54-56: Confirm role initialization for newly created clusters.

Calling roleInitializationService.initClusterRoles is crucial for setting up cluster-level permissions upon creation. Consider adding error handling or logging if role initialization fails, so administrators can investigate any issues.

apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/PermissionControllerTest.java (1)

52-106: Comprehensive test for cluster role lifecycle.

  1. Validation: The test checks the role assignment, retrieval, and removal thoroughly, ensuring coverage of main CRUD operations.
  2. Clarity: Each step is clearly separated, making it easy to follow the test logic.
  3. Potential Improvement: Include a negative test case where attempts to modify the cluster fail when the user lacks the needed role.
apollo-portal/src/main/resources/static/scripts/controller/role/ClusterRoleController.js (1)

36-41: Initialize permissions gracefully.

Initializing cluster permissions on controller load ensures the UI can conditionally show or hide role management functionality. The promise handlers properly show a warning if initialization fails. Consider adding debug logs for better troubleshooting.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/AppService.java (1)

172-176: Handle possible exceptions in environment iteration
When retrieving active environments via portalSettings.getActiveEnvs(), consider adding error-handling or logging if one or more target environments fail to initialize cluster roles (e.g., network or permission availability issues). This is especially important for robust production usage.

apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/spi/defaultImpl/RoleInitializationServiceTest.java (1)

45-45: Test environment should be configurable
private final String ENV = "DEV"; is hardcoded. Making this configurable (e.g., from a properties file or test parameter) would allow environment-specific test variations without code changes.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ItemController.java (2)

211-212: Loop-based validation
When syncing items to multiple namespaces, each iteration checks:

hasModifyNamespacePermission(...) || hasModifyClusterPermission(...)

If any environment requires extra constraints, consider customizing the logic or restricting the OR condition to reduce vulnerabilities.


226-227: Syntax check permission
Running syntax checks can be less restrictive than actual item modification, but is currently governed by the same permission check. If the design requires a less strict check, consider adjusting.

apollo-portal/src/main/resources/static/scripts/services/PermissionService.js (2)

26-32: Ensure consistent error-handling approach.
This new init_cluster_permission resource is consistent with the existing pattern; however, if the server response contains unexpected error details, consider logging them to help with debugging and ensuring the UI is notified properly.


101-105: Add user-notification logic if request fails.
Retrieving cluster role users is straightforward here but might benefit from more explicit error-handling or UI updates (e.g., toaster error messages) on failure.

apollo-portal/src/main/resources/static/cluster/role.html (1)

1-173: Well-structured new file for cluster-level role management.

  1. The layout is intuitive, using AngularJS forms and repeaters to manage “ModifyCluster” and “ReleaseCluster” roles.
  2. It's good to see the hasAssignUserPermission condition guard the role assignment UI. This prevents showing sensitive options to unauthorized users.
  3. Consider adding a loading indicator for slow network cases when retrieving assigned users or assigning roles.
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between 0c7c341 and d3c607a.

📒 Files selected for processing (29)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/component/PermissionValidator.java (2 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/PermissionType.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/RoleType.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/AppController.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ClusterController.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ConfigsImportController.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ItemController.java (7 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/NamespaceBranchController.java (3 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/PermissionController.java (4 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ReleaseController.java (3 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/entity/vo/ClusterRolesAssignedUsers.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/AppService.java (6 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/ClusterService.java (2 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/RoleInitializationService.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/spi/defaultimpl/DefaultRoleInitializationService.java (2 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/util/RoleUtils.java (4 hunks)
  • apollo-portal/src/main/resources/static/cluster/role.html (1 hunks)
  • apollo-portal/src/main/resources/static/config.html (1 hunks)
  • apollo-portal/src/main/resources/static/i18n/en.json (1 hunks)
  • apollo-portal/src/main/resources/static/i18n/zh-CN.json (1 hunks)
  • apollo-portal/src/main/resources/static/scripts/controller/role/ClusterRoleController.js (1 hunks)
  • apollo-portal/src/main/resources/static/scripts/directive/namespace-panel-directive.js (2 hunks)
  • apollo-portal/src/main/resources/static/scripts/services/PermissionService.js (7 hunks)
  • apollo-portal/src/main/resources/static/styles/common-style.css (1 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/ControllableAuthorizationConfiguration.java (1 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/SkipAuthorizationConfiguration.java (1 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/ItemControllerAuthIntegrationTest.java (1 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/PermissionControllerTest.java (1 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/spi/defaultImpl/RoleInitializationServiceTest.java (2 hunks)
✅ Files skipped from review due to trivial changes (1)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ClusterController.java
🧰 Additional context used
🪛 Biome (1.9.4)
apollo-portal/src/main/resources/static/scripts/controller/role/ClusterRoleController.js

[error] 88-88: Shouldn't redeclare 'user'. Consider to delete it or rename it.

'user' is defined here:

(lint/suspicious/noRedeclare)

🔇 Additional comments (77)
apollo-portal/src/test/java/com/ctrip/framework/apollo/ControllableAuthorizationConfiguration.java (4)

1-16: License header continuity check.
The license header follows the Apache 2.0 format consistently, matching the project’s standard. No action needed.


31-33: Clear class naming and annotations.
Naming reflects the intent of providing a controllable authorization context for tests, and usage of @Configuration is appropriate.


37-38: Mocked userInfoHolder bean.
Using a @Primary bean overrides any default beans in the context. This is a good approach for test isolation. Make sure that user details are correctly set via further stubbing if needed in actual tests.


50-51: Mocked itemService bean.
Again, overriding with a primary bean mock for the test environment is consistent with the overall design. Ensure that dependent test cases provide the necessary stubbing logic for item operations if required.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/entity/vo/ClusterRolesAssignedUsers.java (8)

1-16: License header is well-structured
This file is correctly including the Apache License 2.0 header, which is consistent with project requirements.


17-17: Package declaration looks good
The package path follows logical grouping under com.ctrip.framework.apollo.portal.entity.vo, matching the project's code organization.


19-20: Imports are straightforward and minimal
You’re only importing the UserInfo class and Set, which is appropriate. No extraneous imports are present.


22-22: Class name is descriptive
The name ClusterRolesAssignedUsers clearly indicates the purpose of this class, reflecting its role in storing user-role assignments.


30-36: Getter/Setter for env
Straightforward implementation. The semantics are clear, and the naming is consistent with the field name.


38-44: Getter/Setter for appId
Same pattern: minimal, readable, and consistent.


46-52: Getter/Setter for cluster
Matches the earlier style and ensures clarity for cluster information.


70-80: toString() is concise and adequate
It includes all fields, making debugging and logging easier.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/AppController.java (1)

128-128: Verify null or empty admin list handling
If appModel.getAdmins() is empty or null, the method might silently skip assigning admins. Ensure the app creation logic handles edge cases or returns an informative error.

apollo-portal/src/main/resources/static/scripts/directive/namespace-panel-directive.js (2)

295-306: Validate fallback logic for cluster-level modify permission

This block enriches the permission model by setting namespace.hasModifyPermission (and branch) if the user has cluster-level modify permission. Ensure that permission results do not inadvertently override existing narrower-scope permissions and that future logic properly accounts for cluster-level overrides.


335-347: Ensure consistency for cluster-level release permission

This block mirrors the modify-permission calls with cluster-level release permission checks. Verify that setting hasReleasePermission does not conflict with narrower namespace permissions. Additionally, consider unifying permission checks into a single routine for improved maintainability.

apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/ItemControllerAuthIntegrationTest.java (5)

1-16: **License Header Check **

The Apache 2.0 License header is properly included, adhering to standard open-source licensing requirements.


81-101: **Robust negative test coverage **

The test method testCreateItemPermissionDenied correctly validates that a user without the required permission receives a 403 FORBIDDEN response. This improves confidence in the authorization checks for item creation.


103-120: **Positive test coverage and service interaction **

The test method testCreateItemPermissionAccessed verifies that an authorized user can create items, and the verify call ensures the ItemService is invoked with correct arguments. This is a well-structured integration test.


121-125: **Mocking user identity **

The setUserId helper method appropriately mocks UserInfoHolder to simulate different user identities, supporting thorough permission testing scenarios.


103-105: **Clean-up strategy after tests **

The usage of @Sql with the AFTER_TEST_METHOD phase is a good approach for cleaning up database changes. Confirm that this script reliably reverts data, especially if multiple tests interact with the same database state.

✅ Verification successful

Let me analyze the test class and cleanup script further to ensure we have a complete picture.


Let me check one more aspect to see if there are other tests in this class that might affect shared state.


Clean-up strategy is comprehensive and reliable

The cleanup.sql script is well-designed and thoroughly cleans up all relevant database tables after test execution:

  • It includes cleanup for all core tables: Item, Namespace, AppNamespace, Cluster, App, etc.
  • It also covers authentication-related tables: Users, Authorities, Permission, Role, etc.
  • The script is executed in the AFTER_TEST_METHOD phase, ensuring proper cleanup regardless of test outcome

The cleanup strategy is reliable for this test case as:

  1. The script removes all data from tables that could be affected by item creation tests
  2. It's comprehensive enough to handle shared state between different test methods
  3. The DELETE statements are unconditional, ensuring complete cleanup
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Check if the cleanup.sql is present and confirm references to the relevant tables or data
fd cleanup.sql --exec cat

Length of output: 2539


Script:

#!/bin/bash
# Check the test class for database interactions and setup
rg -B 5 -A 5 "testCreateItemPermissionAccessed" 

# Look for other tests in the same class that might affect shared state
ast-grep --pattern 'class ItemControllerAuthIntegrationTest {
  $$$
}'

Length of output: 1717


Script:

#!/bin/bash
# Check for all test methods and their cleanup annotations in this class
rg -l "ItemControllerAuthIntegrationTest" | xargs rg "@Test|@Sql.*cleanup"

Length of output: 182

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/PermissionController.java (4)

28-28: Import usage looks fine.
This import statement for ClusterRolesAssignedUsers indicates a new VO class for cluster-level role assignments. It's clearly used downstream for retrieving and displaying cluster roles.


124-135: Might need environment validation.
Here, too, the code retrieves cluster permissions without an explicit environment format check. Ensure consistency by validating env just as other methods (e.g., lines 218–222) do.


239-263: Role assignment usage is consistent.
Assignments follow the same pattern used throughout the controller: verifying user existence, validating env, checking roleType, etc. The logic is straightforward and consistent with established patterns for namespace roles.


265-281: Removal of cluster role is correct.
The removal approach parallels the assignment logic, providing a symmetrical API design. The environment checks, role checks, and user checks all align well with the rest of the code.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/RoleType.java (2)

27-30: New cluster role types introduced.
The new constants MODIFY_CLUSTER and RELEASE_CLUSTER follow existing naming conventions, which promotes clarity.


32-33: Updated isValidRoleType logic.
Extending the check to include cluster roles ensures consistency. The approach is straightforward and leverages existing comparisons appropriately.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/PermissionType.java (1)

48-54: Cluster-level permission constants.
Defining MODIFY_CLUSTER and RELEASE_CLUSTER aligns with the newly introduced cluster role types in RoleType.java. This extension cleanly supports the new cluster-level permissions.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/ClusterService.java (2)

35-35: Field injection is properly defined.

Declaring the RoleInitializationService as final is a good practice, preventing unintended reassignment.


37-41: Constructor injection looks consistent.

The new constructor parameter roleInitializationService follows established best practices for dependency injection. Ensure that any calling code in other parts of the application is updated accordingly to avoid NoSuchMethodError.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ConfigsImportController.java (1)

64-65: Expanded permission check is aligned with cluster-level security.

Including hasModifyClusterPermission in the @PreAuthorize expression ensures tighter access control. Verify that the logic remains understandable to maintainers and consider documenting the relationship between namespace- and cluster-level permissions.

apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/PermissionControllerTest.java (2)

44-45: Autowired service usage is correct.

Injecting RoleInitializationService is consistent. Confirm that related tests utilize mocks if needed to isolate behavior.


47-50: Set-up method ensures cluster roles are initialized before tests.

Invoking roleInitializationService.initClusterRoles prior to tests is beneficial for a clean and predictable test environment. If roles require unique states for different tests, ensure proper teardown as well.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/util/RoleUtils.java (4)

69-71: Method name aligns with purpose.

buildModifyClusterRoleName is self-explanatory. The usage of STRING_JOINER for consistent role naming is good for maintainability.


85-87: Consistent naming for release cluster roles.

buildReleaseClusterRoleName parallels the existing naming patterns, ensuring consistency. No additional concerns here.


97-99: Generalized builder for cluster role names.

The buildClusterRoleName method is a flexible addition, making it easier to handle future role types without cluttering the utility class.


113-115: Target ID generation is straightforward.

buildClusterTargetId correctly aggregates the user-specified parameters, helping maintain clarity in permission checks. Consider verifying that this target ID formatting remains consistent across all permission/role checks in usage.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/component/PermissionValidator.java (2)

63-67: Good addition for cluster modification checks.

This method effectively queries whether the current user has the new MODIFY_CLUSTER permission. The logic is consistent with the existing permission checks for namespaces and appears to align well with the cluster-level permission paradigm introduced in this PR.


81-85: Good addition for cluster release checks.

Similar to the hasModifyClusterPermission method, this method straightforwardly verifies whether the user has the RELEASE_CLUSTER permission, further expanding the portal’s fine-grained permission model.

apollo-portal/src/main/resources/static/scripts/controller/role/ClusterRoleController.js (5)

43-48: Check if user has role assignment permissions.

This call to PermissionService.has_assign_user_permission is a good practice before allowing role assignment. The result is used to toggle $scope.hasAssignUserPermission. Consider verifying if further role assignment methods are disabled accordingly in the UI when the user lacks permissions.


50-57: Handle user roles retrieval robustly.

Fetching the current cluster’s assigned users and storing them in $scope.rolesAssignedUsers is well-implemented. The error handling gracefully alerts the user if the retrieval fails. This provides clarity on assignment statuses.


58-87: Cluster role assignment logic is solid.

The assignRoleToUser function correctly handles both "ReleaseCluster" and "ModifyCluster" scenarios by calling the appropriate PermissionService methods. The use of select2 to retrieve the user ID is neat. Error handling is also well-structured, showing success or failure toasts.


88-88: 'user' variable redeclaration.

The user variable is redeclared in another branch, which may shadow or confuse references to the same identifier. This has been flagged before in a previous review.

🧰 Tools
🪛 Biome (1.9.4)

[error] 88-88: Shouldn't redeclare 'user'. Consider to delete it or rename it.

'user' is defined here:

(lint/suspicious/noRedeclare)


153-162: Clean removal from array.

The removeUserFromList utility function properly identifies and removes the user by matching userId. This is straightforward yet effective. Beware of potential off-by-one errors when splicing.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/NamespaceBranchController.java (3)

81-82: Broadened pre-authorization condition.

By combining hasModifyNamespacePermission with hasModifyClusterPermission, the code now supports either permission to create a branch. This change is consistent with the cluster-level authorization expansion, ensuring more flexibility.


101-106: Refined delete permission checks.

Splitting permissions into modification (hasModifyPermission) and release (hasReleasePermission) clarifies the conditions under which a branch may be deleted. The logic to allow deletion if there’s no release or if the user also has release rights is well-structured.


122-123: Extended pre-authorization for merge.

Allowing a merge with either release namespace permission or cluster permission is logical. This aligns nicely with the newly introduced hasReleaseClusterPermission API, preserving consistent authorization across different operations.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ReleaseController.java (3)

72-73: New pre-authorization for normal releases.

Requiring either namespace release permission or cluster release permission is consistent with the cluster-focused enhancements. It empowers more granular control over who can initiate releases.


102-103: Appropriate for gray release checks.

Similarly, broadening gray release checks to consider cluster-level permissions is consistent with the changes in the rest of the codebase. Ensures that only authorized users can initiate a gray release.


192-198: Rollback permission refinement.

Separating the checks into releaseNamespacePermission and releaseClusterPermission clearly indicates that either permission suffices to roll back a release. The subsequent AccessDeniedException is concise and appropriate.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/AppService.java (4)

28-31: Use PortalSettings with caution across modules
The newly imported PortalSettings is used to retrieve the active environments. Ensure that any future module dependencies remain consistent with portalSettings usage to avoid potential environment mismatches or stale data if caching is involved.


68-68: Consistency in final fields
Declaring portalSettings as final is a good practice that enforces immutability. Continue applying this principle for other newly introduced dependencies to preserve consistent constructor-based dependency injection.


Line range hint 82-94: Constructor updates require test updates
The constructor signature change to include PortalSettings might break code in downstream modules or tests. Ensure all test suites and integration points are updated accordingly to pass the PortalSettings parameter.


145-147: Initialize cluster roles with caution
roleInitializationService.initClusterRoles(...) depends on env.getName() and ConfigConsts.CLUSTER_NAME_DEFAULT. Make sure env is non-null and recognized by the system. Consider validating or logging in case env or cluster name is invalid.

apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/spi/defaultImpl/RoleInitializationServiceTest.java (4)

148-167: Positive path test coverage
The testInitClusterRole test method convincingly verifies a scenario where neither the modify nor release cluster role exists. This solid coverage ensures correct role creation.


168-185: Verify no extraneous calls
testInitClusterRoleHasExisted correctly checks that no new roles or permissions are created if they already exist. This covers an important idempotent scenario.


187-206: Mixed existence scenarios
testInitClusterRoleModifyClusterExisted verifies the modify cluster role pre-exists but not the release role. This partial existence scenario is well tested, ensuring graceful handling.


208-227: Release cluster pre-exists scenario
testInitClusterRoleReleaseClusterExisted ensures role creation logic is partial, addressing the final major scenario. The test suite is comprehensive.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/spi/defaultimpl/DefaultRoleInitializationService.java (2)

177-189: Transaction boundary
@Transactional on initClusterRoles ensures consistent role creation. However, if partial failures occur (e.g., cluster role creation succeeds but release role creation fails), re-check that the transaction is rolled back as expected.


246-255: Cluster role creation and permission association
createClusterRole(...) properly sets up the permission and role within a single method. This is good for maintainability. Should further permission checks be needed, keep everything within this method or consider a dedicated helper method.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ItemController.java (5)

91-92: Create item permission scope
Similar to line 76, newly created items in a namespace can be changed by anyone with cluster-level modify privileges. Confirm that this is by design.


111-112: Update item permission
Same OR-based approach applies for updating items. Confirm there are no conflicts between cluster-level roles and granular namespace-level roles that might cause unexpected modifications.


126-127: Delete item permission
Deleting an item typically requires at least the same permissions required for creation or update. The OR-based approach may be too permissive if the business logic expects stricter controls.


238-239: Revoke items permission
Revoking items also uses the OR condition. If the product intentionally merges the cluster and namespace permissions, this is validated. Otherwise, confirm the correct method of enforcement.


76-77: OR-based permission is broad
The OR logic:

@permissionValidator.hasModifyNamespacePermission(...) OR
@permissionValidator.hasModifyClusterPermission(...)

allows either cluster or namespace modify permissions. Verify this doesn't inadvertently grant excessive privileges to cluster-level operators who lack namespace-level permissions, and vice versa.

✅ Verification successful

Let me gather information about the permission validation implementation to verify the concern.


Let me try to find the actual implementation details using ripgrep for better context.


Let me search for more context about the permission types and role utils to understand the permission hierarchy.


Let me try one final search to understand the ItemController's protected operation.


Based on the code analysis so far, I can provide a final response:

OR-based permission is appropriately scoped
The permission check combines two specific, well-defined permissions:

  1. hasModifyNamespacePermission - verifies if the user has permission to modify the specific namespace
  2. hasModifyClusterPermission - verifies if the user has permission to modify the specific cluster

The implementation in PermissionValidator shows that these permissions are properly scoped:

  • Namespace permission is checked against a specific namespace target ID (appId + namespaceName)
  • Cluster permission is checked against a specific cluster target ID (appId + env + clusterName)

Each permission type is independently validated against specific resource targets, making the OR condition a valid design choice that maintains proper access control granularity.

🏁 Scripts executed

The following scripts were executed for the analysis:

Error: No shell scripts found


Script:

#!/bin/bash
# Search for the PermissionValidator class implementation
ast-grep --pattern 'class PermissionValidator {
  $$$
  hasModifyNamespacePermission($$$) {
    $$$
  }
  $$$
}'

# Also search for hasModifyClusterPermission
ast-grep --pattern 'class PermissionValidator {
  $$$
  hasModifyClusterPermission($$$) {
    $$$
  }
  $$$
}'

# Look for the ItemController class to understand the context
ast-grep --pattern 'class ItemController {
  $$$
}'

# Search for related test files
fd "PermissionValidatorTest|ItemControllerTest" --type f

Length of output: 546


Script:

#!/bin/bash
# Search for PermissionValidator implementation with context
rg -l "class PermissionValidator"

# Search for hasModifyNamespacePermission implementation
rg "hasModifyNamespacePermission.*\{" -A 10

# Search for hasModifyClusterPermission implementation
rg "hasModifyClusterPermission.*\{" -A 10

# Search for ItemController's method using this permission
rg -B 2 -A 5 "@PreAuthorize.*hasModifyNamespacePermission.*hasModifyClusterPermission"

Length of output: 4536


Script:

#!/bin/bash
# Search for PermissionType enum
rg "enum PermissionType" -A 10

# Search for RoleUtils implementation
rg "class RoleUtils" -A 20

# Search for the ItemController method using this permission to understand what operation it protects
rg -B 5 -A 10 "@PreAuthorize.*hasModifyNamespacePermission.*hasModifyClusterPermission" --type java

Length of output: 4910


Script:

#!/bin/bash
# Search for the ItemController class implementation focusing on the method with the permission
rg -B 10 -A 10 "class ItemController" --type java

# Search for PermissionType enum in a different way
find . -type f -exec grep -l "enum.*PermissionType" {} \;

# Search for the specific method in ItemController that uses this permission
rg -B 10 -A 10 "class ItemController" --type java | rg -B 10 -A 10 "@PreAuthorize.*hasModifyNamespacePermission"

Length of output: 17823

apollo-portal/src/main/resources/static/scripts/services/PermissionService.js (9)

45-48: Adequate usage alignment with existing naming.
has_cluster_permission follows a naming pattern similar to has_app_permission and has_namespace_permission, ensuring consistency throughout the service. Good job!


106-112: Role assignment is well structured.
Assigning cluster roles shares the same approach as assigning namespace roles, which aids maintainability. Good job referencing the same pattern.


113-115: Remove cluster role – usage caution.
The removal logic is sound. Double-check that front-end components have appropriate confirmation flows if these roles are business-critical.


132-145: Promise-based approach is consistent with existing code.
initClusterPermission neatly fits the existing promise pattern. Keep an eye out for potential usage of async/await in the future if the codebase transitions away from AngularJS conventions.


192-206: Maintain parallel structure with hasAppPermission.
hasClusterPermission fosters code reusability and standardization. Nicely done!


272-286: Cluster role assignment function.
This block aligns with established naming conventions. Great job on code clarity.


288-303: Symmetry between assign/remove cluster role.
It's excellent that you preserved symmetry in methods. This improves maintainability and developer onboarding.


309-311: Expose init_cluster_permission as a top-level method.
Including init_cluster_permission in the returned object is consistent with your existing approach for app/namespace permissions. Looks good!


447-476: Comprehensive methods for cluster-level checks.
Adding has_modify_cluster_permission, has_release_cluster_permission, etc., ensures your service covers all cluster operations thoroughly.

apollo-portal/src/main/resources/static/styles/common-style.css (1)

1192-1195: New .cluster-info-panel class.
Adding extra spacing around cluster information is a simple and effective visual enhancement. The choice of margin: 20px 0px aligns well with the existing layout.

apollo-portal/src/main/resources/static/config.html (1)

190-210: Improved cluster visibility with a dedicated panel.

  1. The newly added <section class="panel cluster-info-panel"> clarifies which environment and cluster the user is operating on.
  2. The “Grant” button linking to role.html is a convenient way for users to manage roles without navigating through multiple pages.
  3. Verify that permission checks are consistently enforced downstream when your backend processes these cluster-based actions.
apollo-portal/src/main/resources/static/i18n/zh-CN.json (1)

364-372: LGTM! The translations for cluster permission management are clear and consistent.

The added translations accurately reflect the cluster permission management functionality and maintain consistency with the English version.

apollo-portal/src/main/resources/static/i18n/en.json (1)

364-372: LGTM! The translations for cluster permission management are clear and consistent.

The added translations accurately reflect the cluster permission management functionality and maintain consistency with the Chinese version.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

🧹 Nitpick comments (1)
apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/service/AppServiceTest.java (1)

81-82: Consider removing unused mock

PortalSettings portalSettings is introduced as a mock but is currently unused. If you don't have immediate plans to test code using portalSettings, consider removing it to keep the test suite lean.

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between d3c607a and 92bb7c3.

📒 Files selected for processing (1)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/service/AppServiceTest.java (2 hunks)
🔇 Additional comments (1)
apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/service/AppServiceTest.java (1)

26-26: Import looks good

The new import for PortalSettings is consistent with the introduction of the mock bean below.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

🧹 Nitpick comments (1)
apollo-configservice/src/test/java/com/ctrip/framework/apollo/configservice/integration/ConfigControllerIntegrationTest.java (1)

352-352: Consider using a more robust synchronization approach instead of a fixed 2-second sleep.

Relying on a fixed delay can make tests slower than necessary and may mask possible timing-related issues. A more robust approach is to use a synchronization mechanism (e.g., CountDownLatch) or a polling approach that confirms the messages have been fully processed before proceeding.

Here is an example of how you could refactor the code to avoid an arbitrary sleep:

-    TimeUnit.MILLISECONDS.sleep(2000);
+    // Example approach using CountDownLatch:
+    // CountDownLatch latch = new CountDownLatch(1);
+    // periodicSendMessage(..., stop, latch);
+    // latch.await(5, TimeUnit.SECONDS); // Wait for message processing or timeout
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between 92bb7c3 and 7aad74b.

📒 Files selected for processing (1)
  • apollo-configservice/src/test/java/com/ctrip/framework/apollo/configservice/integration/ConfigControllerIntegrationTest.java (1 hunks)

@BlackBear2003 BlackBear2003 marked this pull request as draft December 29, 2024 07:27
@BlackBear2003 BlackBear2003 marked this pull request as ready for review January 1, 2025 12:03
Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 5

🧹 Nitpick comments (19)
apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/CommitController.java (1)

54-54: Consider returning an HTTP 403 instead of an empty list for unauthorized access.

Returning an empty list may obscure permission errors and lead to confusion. Instead, returning a 403 error (Forbidden) or some proper status code would more accurately indicate that the requested resource is disallowed.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/NamespaceController.java (1)

114-114: Consider unifying the repeated permission check.

You have a repeated pattern:

if (permissionValidator.shouldHideConfigToCurrentUser(appId, env, clusterName, ...)) {
  namespaceBO.hideItems();
}

It might be beneficial to extract this into a dedicated helper method to reduce duplication, improve clarity, and ensure consistency across the codebase.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/entity/vo/ClusterNamespaceRolesAssignedUsers.java (2)

22-22: Add class-level Javadoc for clarity.
A brief class-level Javadoc would be beneficial to explain the role of this data structure in the permission management system, enhancing understanding for future maintainers.


24-28: Consider using an enum or a well-defined constant for environment strings.
Using an enum for environments (DEV, TEST, PROD, etc.) or a known constant class eliminates potential for typos and consolidates environment-related checks.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ReleaseController.java (1)

Line range hint 72-190: Document the new cluster-based permission model.

The changes consistently implement cluster-level authorization across all release operations. To ensure proper usage and maintenance:

  1. Consider adding Javadoc explaining the cluster-based permission model
  2. Update API documentation to reflect the new authorization requirements
  3. Consider creating a migration guide for existing clients

Would you like me to help draft the documentation for the new permission model?

apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/PermissionControllerTest.java (2)

47-50: Enhance test setup reliability

The setup method could be more robust:

  1. Add verification that role initialization was successful
  2. Consider adding pre-test cleanup to ensure a clean state
  3. Consider using @Sql for both before and after test execution
 @Before
 public void setUp() {
+    // Clean up any existing roles first
+    roleInitializationService.cleanupRoles(appId, env, clusterName);
     roleInitializationService.initClusterNamespaceRoles(appId, env, clusterName, "apollo");
+    // Verify initialization
+    assertTrue(roleInitializationService.isClusterNamespaceRoleInitialized(appId, env, clusterName));
 }

56-58: Extract HTTP headers to constants

Consider extracting the content-type header to a constant or utility method for reusability and maintainability.

+    private static final String CONTENT_TYPE = "Content-Type";
+    private static final String APPLICATION_JSON = "application/json";
+
     HttpHeaders headers = new HttpHeaders();
-    headers.set("Content-Type", "application/json");
+    headers.set(CONTENT_TYPE, APPLICATION_JSON);
apollo-portal/src/main/resources/static/scripts/controller/role/ClusterNamespaceRoleController.js (3)

17-21: Ensure consistent naming convention.

The controller name ClusterNamespaceRoleController is clear. However, confirm that similar controllers across the codebase follow the same naming pattern (e.g., <Context>Controller) for consistency.


36-41: Handle initialization errors comprehensively.

Currently, the error callback only displays a warning. Consider adding a more detailed error-handling routine, such as automatically retrying, redirecting, or logging for observability.


153-162: Consider early return or short-circuit logic.

removeUserFromList iterates over the entire list to find an index. If the user is discovered early, consider returning immediately. This reduces time complexity when the user is found at earlier indices.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/RoleType.java (1)

32-33: Keep isValidRoleType updated.

Be mindful of future role additions and update isValidRoleType accordingly. Consider documenting the need to maintain this method if new role types are introduced.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/util/RoleUtils.java (1)

97-99: Generalize role name construction if needed.

buildClusterRoleName is straightforward. If further cluster-based role strings are introduced, consider centralizing them in a single place to reduce repetitive patterns.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/spi/defaultimpl/DefaultRoleInitializationService.java (2)

177-189: LGTM! Consider adding audit logging.

The implementation correctly follows the established pattern for role initialization and maintains transactional integrity. The method properly checks for existing roles before creation to avoid duplicates.

Consider adding audit logging similar to other role initialization methods to track who creates these roles and when.


246-255: Add input parameter validation.

The implementation is consistent with other role creation methods, but could benefit from explicit parameter validation.

Consider adding RequestPrecondition.checkArgumentsNotEmpty() checks for all parameters at the beginning of the method:

 private void createClusterRole(String appId, String env, String clusterName, String permissionType,
                                String roleName, String operator) {
+    RequestPrecondition.checkArgumentsNotEmpty(appId, env, clusterName, permissionType, roleName, operator);
     Permission permission =
         createPermission(RoleUtils.buildClusterTargetId(appId, env, clusterName), permissionType, operator);
     Permission createdPermission = rolePermissionService.createPermission(permission);

     Role role = createRole(roleName, operator);
     rolePermissionService
         .createRoleWithPermissions(role, Sets.newHashSet(createdPermission.getId()));
 }
apollo-portal/src/main/resources/static/scripts/services/PermissionService.js (1)

272-303: Standardize error handling messages.

The implementation follows the established patterns for role management, but could benefit from standardized error messages.

Consider creating constants for error messages to maintain consistency across the application:

const ERROR_MESSAGES = {
  ROLE_ASSIGNMENT_FAILED: 'Failed to assign role to user',
  ROLE_REMOVAL_FAILED: 'Failed to remove role from user'
};
apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/PermissionController.java (4)

83-87: Add API documentation.

The endpoint implementation is correct but lacks documentation.

Add Swagger/OpenAPI documentation to describe the endpoint, its parameters, and possible responses:

+    /**
+     * Initialize permissions for a cluster namespace.
+     *
+     * @param appId The application ID
+     * @param env The environment
+     * @param clusterName The cluster name
+     * @return ResponseEntity<Void> with status 200 on success
+     */
     @PostMapping("/apps/{appId}/envs/{env}/clusters/{clusterName}/initNsPermission")
     public ResponseEntity<Void> initClusterNamespacePermission(@PathVariable String appId, @PathVariable String env, @PathVariable String clusterName) {

123-133: Consider adding caching for permission checks.

The implementation is correct and consistent with existing patterns.

Consider adding caching to improve performance of frequent permission checks:

+    @Cacheable(value = "cluster-namespace-permissions", key = "#appId + '-' + #env + '-' + #clusterName + '-' + #permissionType")
     @GetMapping("/apps/{appId}/envs/{env}/clusters/{clusterName}/ns_permissions/{permissionType}")
     public ResponseEntity<PermissionCondition> hasClusterNamespacePermission(@PathVariable String appId, @PathVariable String env, @PathVariable String clusterName,
                                                            @PathVariable String permissionType) {

237-259: Add audit logging for role management operations.

The implementation has proper security controls and input validation.

Add audit logging annotations to track role changes:

     @PreAuthorize(value = "@permissionValidator.hasAssignRolePermission(#appId)")
     @PostMapping("/apps/{appId}/envs/{env}/clusters/{clusterName}/ns_roles/{roleType}")
+    @ApolloAuditLog(type = OpType.CREATE, name = "Auth.assignClusterNamespaceRoleToUser")
     public ResponseEntity<Void> assignClusterNamespaceRoleToUser(...)

     @PreAuthorize(value = "@permissionValidator.hasAssignRolePermission(#appId)")
     @DeleteMapping("/apps/{appId}/envs/{env}/clusters/{clusterName}/ns_roles/{roleType}")
+    @ApolloAuditLog(type = OpType.DELETE, name = "Auth.removeClusterNamespaceRoleFromUser")
     public ResponseEntity<Void> removeClusterNamespaceRoleFromUser(...)

Also applies to: 261-277


213-235: Add pagination support for role user retrieval.

The implementation correctly retrieves and validates role assignments.

Consider adding pagination support to handle large numbers of users efficiently:

     @GetMapping("/apps/{appId}/envs/{env}/clusters/{clusterName}/ns_role_users")
-    public ClusterNamespaceRolesAssignedUsers getClusterNamespaceRoles(...)
+    public ClusterNamespaceRolesAssignedUsers getClusterNamespaceRoles(
+        @PathVariable String appId,
+        @PathVariable String env,
+        @PathVariable String clusterName,
+        @RequestParam(defaultValue = "0") int page,
+        @RequestParam(defaultValue = "20") int size)
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between 7aad74b and a37dc2f.

📒 Files selected for processing (24)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/component/PermissionValidator.java (3 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/PermissionType.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/RoleType.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/CommitController.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ConfigsExportController.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ConfigsImportController.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ItemController.java (8 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/NamespaceBranchController.java (4 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/NamespaceController.java (2 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/PermissionController.java (4 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ReleaseController.java (5 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ReleaseHistoryController.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/entity/vo/ClusterNamespaceRolesAssignedUsers.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/AppService.java (6 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/ClusterService.java (2 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/RoleInitializationService.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/spi/defaultimpl/DefaultRoleInitializationService.java (2 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/util/RoleUtils.java (4 hunks)
  • apollo-portal/src/main/resources/static/cluster/role.html (1 hunks)
  • apollo-portal/src/main/resources/static/scripts/controller/role/ClusterNamespaceRoleController.js (1 hunks)
  • apollo-portal/src/main/resources/static/scripts/directive/namespace-panel-directive.js (2 hunks)
  • apollo-portal/src/main/resources/static/scripts/services/PermissionService.js (7 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/PermissionControllerTest.java (1 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/spi/defaultImpl/RoleInitializationServiceTest.java (2 hunks)
🚧 Files skipped from review as they are similar to previous changes (10)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/RoleInitializationService.java
  • apollo-portal/src/main/resources/static/scripts/directive/namespace-panel-directive.js
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/PermissionType.java
  • apollo-portal/src/main/resources/static/cluster/role.html
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ConfigsImportController.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/AppService.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/ClusterService.java
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/spi/defaultImpl/RoleInitializationServiceTest.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/NamespaceBranchController.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ItemController.java
🔇 Additional comments (23)
apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/NamespaceController.java (1)

128-128: Double-check null-safety for namespaceBO and its base info.

While you already check that namespaceBO != null, consider whether namespaceBO.getBaseInfo() might be null under any edge cases. If that's a possibility, you may need an additional check to avoid a potential NullPointerException.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ConfigsExportController.java (1)

81-81: Good addition but ensure consistent application in other permission checks

Including #clusterName in the @PreAuthorize annotation allows for more granular access control. However, please confirm that all relevant permission checks in related controllers (e.g., import or modify operations) are updated similarly to maintain consistency throughout the codebase.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ReleaseController.java (4)

101-101: Verify cluster vs branch name usage in permission check.

While the permission check is correctly updated, note that the method uses branchName as the clusterName in the model (line 108). Ensure this aligns with the intended permission model for gray releases.

Consider documenting the relationship between cluster and branch names in the method's documentation to clarify the permission model for gray releases.


149-149: LGTM! Consistent permission check implementation.

Both read methods correctly implement the new cluster-aware permission model while maintaining the secure approach of returning empty lists for unauthorized access.

Also applies to: 164-164


190-190: LGTM! Verify rollback permission checks in integration tests.

The permission check correctly uses cluster information from the release object. Since rollback is a critical operation, ensure comprehensive integration testing of the new permission model.

Run the following script to check for existing integration tests:


72-72: LGTM! Verify permission validator implementation.

The permission check now correctly includes cluster-level authorization, aligning with the feature requirements.

Run the following script to verify the permission validator implementation:

✅ Verification successful

Permission validation implementation is correctly handling cluster-level authorization

The implementation in PermissionValidator properly validates cluster-level permissions through a hierarchical check:

  1. Checks app-level namespace release permission
  2. Checks environment-specific namespace release permission
  3. Checks cluster-specific release permission using hasReleaseNamespaceInClusterPermission

The permission target IDs are correctly built using RoleUtils:

  • Namespace permission: appId:namespaceName
  • Environment-specific: appId:namespaceName:env
  • Cluster-specific: appId:env:clusterName
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Verify the implementation of hasReleaseNamespacePermission method
# to ensure it properly handles the new cluster parameter

ast-grep --pattern 'hasReleaseNamespacePermission($appId, $env, $clusterName, $namespaceName)'

Length of output: 96


Script:

#!/bin/bash
# Let's try a different approach to find the permission validator implementation
# First, let's find the validator class
fd "PermissionValidator" --type f

# Then let's search for the method implementation using ripgrep for better context
rg "hasReleaseNamespacePermission" -A 5

Length of output: 16240


Script:

#!/bin/bash
# Let's check the RoleUtils to understand how the cluster-based permission target is built
rg "buildNamespaceTargetId|buildClusterTargetId" -A 3

# Also check for any cluster-specific permission types
rg "PermissionType\." -A 1

Length of output: 23146

apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/PermissionControllerTest.java (2)

1-107: Verify test uniqueness in the codebase

Let's ensure this test implementation doesn't duplicate existing test coverage.

✅ Verification successful

Based on the search results, I can provide the final response:

Test implementation is unique and well-placed

The search results confirm this test implementation is unique in the codebase:

  • While there are other tests involving RoleInitializationService (in RoleInitializationServiceTest.java), they focus on unit testing the service itself
  • The testClusterNamespaceRoleLifeCycle method name is unique and no similar role lifecycle tests exist
  • This is the only integration test focusing on the permission controller's role management endpoints

This test provides valuable coverage of the role assignment workflow through the REST API layer, complementing the unit tests in RoleInitializationServiceTest.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for similar role lifecycle tests
rg -l "RoleInitializationService|ClusterNamespaceRoles" | xargs rg "class.*Test" -C 5

# Search for similar test method names
rg "test.*Role.*Cycle" -C 5

Length of output: 7444


35-36: Verify the necessity of skipping authorization in this test

The @ActiveProfiles("skipAuthorization") annotation seems counterintuitive for testing permission management functionality. This might bypass critical authorization checks that should be validated.

Let's check how other auth-related tests handle this:

✅ Verification successful

Let me gather more information about the authorization setup and test configuration.


Let me check the configuration for the skipAuthorization profile.


Let me check the test base class and the skipAuthorization configuration.


Let me check the actual test implementation and what permissions are being tested.


The skipAuthorization profile is correctly used in this test

The @ActiveProfiles("skipAuthorization") annotation is actually appropriate here because:

  1. The test is focused on verifying the lifecycle of cluster namespace roles (assignment, verification, and removal) through the REST API endpoints, not the authorization mechanism itself.
  2. The SkipAuthorizationConfiguration provides mock implementations that:
    • Grant superadmin access to bypass pre-authorization checks
    • Allow the test to focus on the core role management functionality
  3. The test properly validates the complete role management flow:
    • Initial state (no roles)
    • Role assignment
    • Role verification
    • Role removal

The authorization skipping is intentional and doesn't compromise the test's validity, as it's testing the role management functionality, not the authorization system itself.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for other auth-related test files and their profile configurations
rg -l "class.*Test.*Auth" | xargs rg "@ActiveProfiles" -C 5

Length of output: 6120


Script:

#!/bin/bash
# Check for skipAuthorization profile configuration
rg -l "skipAuthorization" | xargs rg "skipAuthorization|@Profile" -C 5

# Check for permission-related test configurations
rg -l "Permission.*Test" | xargs rg "@ActiveProfiles|@Test" -C 5

Length of output: 33400


Script:

#!/bin/bash
# Check the skipAuthorization configuration and its purpose
rg -l "SkipAuthorizationConfiguration" | xargs rg -C 10

# Check the actual test implementation to understand what's being tested
cat apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/PermissionControllerTest.java

Length of output: 4721


Script:

#!/bin/bash
# Check the AbstractIntegrationTest class
cat apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/AbstractIntegrationTest.java

# Check for any skipAuthorization configuration files
rg -g "*.java" -l "skipAuthorization" | xargs rg "class.*Configuration" -C 5

Length of output: 3436


Script:

#!/bin/bash
# Check the SkipAuthorizationConfiguration implementation
cat apollo-portal/src/test/java/com/ctrip/framework/apollo/SkipAuthorizationConfiguration.java

# Check what's being tested in the permission controller
rg -g "*.java" "class.*PermissionController" -A 10

Length of output: 5749

apollo-portal/src/main/resources/static/scripts/controller/role/ClusterNamespaceRoleController.js (1)

58-87: Check for null references and validate user input.

Within assignRoleToUser, ensure that roleType, env, and clusterName are non-null and valid before proceeding with the role assignment. While you do check user, you might also want to handle unexpected or undefined role types gracefully.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/RoleType.java (1)

27-30: New constants look good.

The addition of MODIFY_NAMESPACE_IN_CLUSTER and RELEASE_NAMESPACE_IN_CLUSTER aligns well with the existing naming conventions.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/util/RoleUtils.java (3)

69-71: Naming consistency.

buildModifyNamespaceInClusterRoleName follows the patterns of existing methods. This approach is consistent and clear for building role names.


85-87: Ensure release role naming alignment.

buildReleaseNamespaceInClusterRoleName is consistent with the buildModifyNamespaceInClusterRoleName approach. No issues noted.


113-115: Reconfirm delimiter usage.

buildClusterTargetId uses ConfigConsts.CLUSTER_NAMESPACE_SEPARATOR. Ensure that any external references to these role/target strings also use the same delimiter to prevent mismatches.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/component/PermissionValidator.java (9)

51-60: Visibility changes for hasModifyNamespacePermission.

Turning this method private is appropriate to limit external usage. Confirm any existing references outside this class have been updated accordingly.


63-67: Implementation fulfills new cluster namespace permission requirement.

hasModifyNamespaceInClusterPermission is a good addition. Ensure that an equivalent unit test suite checks whether these new permission checks function correctly.


69-80: Layered checks are effective.

hasModifyNamespacePermission(...) unifies three different permission checks in a short-circuit manner, which is efficient. Keep an eye on potential performance concerns if further checks are added in the future.


82-93: Visibility changes for hasReleaseNamespacePermission.

Similar to hasModifyNamespacePermission, ensure external references are updated for these newly private methods.


94-97: Consistent approach for release permissions.

hasReleaseNamespaceInClusterPermission parallels hasModifyNamespaceInClusterPermission.


100-110: Comprehensive release checks.

hasReleaseNamespacePermission uses short-circuit logic as well. Current approach is readable and maintainable.


113-114: Potential confusion in hasDeleteNamespacePermission.

hasDeleteNamespacePermission simply returns hasAssignRolePermission(appId) || isSuperAdmin(). Confirm that no dedicated "delete" role is required.


117-119: Combined operation check.

hasOperateNamespacePermission merges modify and release checks. This may help reduce duplication. It appears consistent with the new cluster-based checks.


160-161: Maintain clarity in shouldHideConfigToCurrentUser.

The extra parameters (env, clusterName) match new cluster permission logic. Confirm that all callers of shouldHideConfigToCurrentUser are updated to supply the cluster name.

Also applies to: 174-174

apollo-portal/src/main/resources/static/scripts/services/PermissionService.js (1)

26-32: LGTM! Resource configurations are well-defined.

The new resource configurations follow RESTful conventions and maintain consistency with existing patterns.

Also applies to: 45-48

@nobodyiam nobodyiam force-pushed the feat-cluster-authorize-by-env branch from a37dc2f to 071b199 Compare January 3, 2025 01:38
Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 2

🧹 Nitpick comments (13)
apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/spi/defaultImpl/RoleInitializationServiceTest.java (3)

149-167: LGTM! Comprehensive test for role initialization.

The test effectively covers the base case where both roles need to be created. Consider adding a verification for the correct permission target ID format to ensure cluster-specific permissions are created correctly.

Add verification for permission target ID:

 verify(rolePermissionService, times(2)).createPermission(any());
+verify(rolePermissionService, times(1)).createPermission(argThat(permission ->
+    permission.getTargetId().equals(RoleUtils.buildNamespaceTargetId(APP_ID, ENV, CLUSTER))
+));

209-227: LGTM! Complete test coverage matrix.

The test effectively covers the opposite partial initialization scenario. Consider adding a brief Javadoc comment to document the test matrix coverage across these four test methods.

Add documentation to clarify test coverage:

+/**
+ * Test matrix for cluster namespace role initialization:
+ * 1. testInitClusterNsRole: Both roles don't exist
+ * 2. testInitClusterNsRoleHasExisted: Both roles exist
+ * 3. testInitClusterNsRoleModifyNamespaceInClusterExisted: Only modify role exists
+ * 4. testInitClusterNsRoleReleaseNamespaceInClusterExisted: Only release role exists
+ */
 @Test
 public void testInitClusterNsRoleReleaseNamespaceInClusterExisted() {

148-227: Excellent test coverage for cluster-based authorization feature.

The test suite comprehensively covers all possible scenarios for cluster namespace role initialization, which is crucial for the new environment cluster-based authorization feature. The systematic approach to testing different role combinations ensures robust permission management.

Consider adding integration tests to verify the interaction between RoleInitializationService and PermissionController for the complete authorization flow.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/spi/defaultimpl/DefaultRoleInitializationService.java (2)

177-189: Add input validation and documentation for the new cluster roles initialization.

The method implementation looks good but could benefit from some improvements:

  1. Add JavaDoc to explain the purpose and behavior of this method
  2. Add parameter validation for appId, env, and clusterName

Consider adding the following improvements:

+ /**
+  * Initializes roles for managing namespaces within a specific cluster.
+  * Creates two roles if they don't exist:
+  * 1. Modify namespace in cluster role
+  * 2. Release namespace in cluster role
+  *
+  * @param appId The application id
+  * @param env The environment name
+  * @param clusterName The cluster name
+  * @param operator The user who performs this operation
+  * @throws IllegalArgumentException if any parameter is null or empty
+  */
  @Transactional
  @Override
  public void initClusterNamespaceRoles(String appId, String env, String clusterName, String operator) {
+   Preconditions.checkArgument(StringUtils.isNotBlank(appId), "AppId cannot be blank");
+   Preconditions.checkArgument(StringUtils.isNotBlank(env), "Env cannot be blank");
+   Preconditions.checkArgument(StringUtils.isNotBlank(clusterName), "ClusterName cannot be blank");
+   Preconditions.checkArgument(StringUtils.isNotBlank(operator), "Operator cannot be blank");
+
    String modifyNamespaceInClusterRoleName = RoleUtils.buildModifyNamespaceInClusterRoleName(appId, env, clusterName);

246-255: Enhance error handling in createClusterRole method.

The implementation follows the established pattern but could benefit from better error handling:

  1. Handle potential failures during permission creation
  2. Consider adding logging for debugging purposes

Consider enhancing the error handling:

  private void createClusterRole(String appId, String env, String clusterName, String permissionType,
                                 String roleName, String operator) {
+   try {
      Permission permission =
          createPermission(RoleUtils.buildClusterTargetId(appId, env, clusterName), permissionType, operator);
      Permission createdPermission = rolePermissionService.createPermission(permission);
+     if (createdPermission == null) {
+       throw new ServiceException("Failed to create permission for cluster role: " + roleName);
+     }

      Role role = createRole(roleName, operator);
      rolePermissionService
          .createRoleWithPermissions(role, Sets.newHashSet(createdPermission.getId()));
+   } catch (Exception e) {
+     logger.error("Failed to create cluster role: {}", roleName, e);
+     throw new ServiceException("Failed to create cluster role: " + roleName, e);
+   }
  }
apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/component/PermissionValidator.java (1)

94-110: Consider reducing code duplication in permission checks.

The modify and release permission check implementations share identical structure. Consider extracting the common logic into a reusable method.

+private boolean hasPermissionInHierarchy(String appId, String env, String clusterName, String namespaceName,
+                                        Function<String, Boolean> namespaceCheck,
+                                        Function<String, Boolean> envCheck,
+                                        Function<String, Boolean> clusterCheck) {
+    if (namespaceCheck.apply(namespaceName)) {
+        return true;
+    }
+    if (envCheck.apply(env)) {
+        return true;
+    }
+    if (clusterCheck.apply(clusterName)) {
+        return true;
+    }
+    return false;
+}
apollo-portal/src/main/resources/static/scripts/services/PermissionService.js (1)

272-303: Consider reducing promise handling duplication.

The role management functions contain repeated promise handling code. Consider extracting a utility function to reduce duplication.

+function wrapPromise(resourceCall) {
+    var d = $q.defer();
+    resourceCall(
+        function (result) {
+            d.resolve(result);
+        },
+        function (result) {
+            d.reject(result);
+        }
+    );
+    return d.promise;
+}
apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ItemController.java (2)

186-189: Consider enhancing the error message clarity

While the permission check is correct, the error message could be more specific about which permission is missing (cluster vs. namespace level).

-        diff.setExtInfo("You are not this project's administrator, nor you have edit or release permission for the namespace:" + namespace);
+        diff.setExtInfo("You do not have sufficient permissions for the namespace in cluster " + namespace.getClusterName() + ": " + namespace);

200-220: Consider optimizing permission check loop

While the permission checking logic is correct, it could be more efficient using Java streams and early return pattern.

-    NamespaceIdentifier noPermissionNamespace = null;
-    // check if user has every namespace's ModifyNamespace permission
-    boolean hasPermission = true;
-    for (NamespaceIdentifier namespaceIdentifier : model.getSyncToNamespaces()) {
-      // once user has not one of the namespace's ModifyNamespace permission, then break the loop
-      hasPermission = permissionValidator.hasModifyNamespacePermission(
-          namespaceIdentifier.getAppId(),
-          namespaceIdentifier.getEnv().getName(),
-          namespaceIdentifier.getClusterName(),
-          namespaceIdentifier.getNamespaceName()
-      );
-      if (!hasPermission) {
-        noPermissionNamespace = namespaceIdentifier;
-        break;
-      }
-    }
-    if (hasPermission) {
-      configService.syncItems(model.getSyncToNamespaces(), model.getSyncItems());
-      return ResponseEntity.status(HttpStatus.OK).build();
-    }
-    throw new AccessDeniedException(String.format("You don't have the permission to modify namespace: %s", noPermissionNamespace));
+    Optional<NamespaceIdentifier> noPermissionNamespace = model.getSyncToNamespaces().stream()
+        .filter(namespace -> !permissionValidator.hasModifyNamespacePermission(
+            namespace.getAppId(),
+            namespace.getEnv().getName(),
+            namespace.getClusterName(),
+            namespace.getNamespaceName()))
+        .findFirst();
+
+    if (noPermissionNamespace.isPresent()) {
+        throw new AccessDeniedException(
+            String.format("You don't have the permission to modify namespace: %s", noPermissionNamespace.get()));
+    }
+
+    configService.syncItems(model.getSyncToNamespaces(), model.getSyncItems());
+    return ResponseEntity.status(HttpStatus.OK).build();
apollo-portal/src/main/resources/static/cluster/role.html (4)

21-21: Consider adding additional security headers

While the character encoding is properly declared, consider adding security-related meta tags to enhance protection against common web vulnerabilities:

+  <meta http-equiv="X-Content-Type-Options" content="nosniff">
+  <meta http-equiv="X-Frame-Options" content="DENY">
+  <meta http-equiv="Content-Security-Policy" content="default-src 'self';">

77-87: Improve user list accessibility and maintainability

The user list buttons lack proper accessibility attributes and have duplicated code. Consider:

  1. Adding ARIA labels and roles
  2. Extracting the common structure into a reusable component
  3. Adding confirmation dialog for role removal

Here's a suggested improvement:

+  <div class="item-container" role="list" aria-label="{{roleTitle}} {{'Common.AssignedUsers' | translate}}">
+    <div class="btn-group item-info" role="listitem"
+         ng-repeat="user in users" 
+         aria-label="{{'Common.User' | translate}}: {{user.userId}}">
+      <button type="button" class="btn btn-default" 
+              ng-bind="user.userId"
+              aria-label="{{user.userId}}">
+      </button>
+      <button type="button" class="btn btn-default dropdown-toggle"
+              data-toggle="dropdown"
+              aria-label="{{'Common.Remove' | translate}} {{user.userId}}"
+              ng-click="confirmRemoval(roleType, user.userId)">
+        <span class="glyphicon glyphicon-remove" aria-hidden="true"></span>
+      </button>
+    </div>
+  </div>

Also applies to: 108-118


128-130: Enhance the no-permission message

The current message could be more helpful by providing guidance on how to obtain the necessary permissions.

-  <div class="panel-body text-center" ng-show="!hasAssignUserPermission">
-    <h2>{{'Cluster.Role.NoPermission' | translate }}</h2>
-  </div>
+  <div class="panel-body text-center" ng-show="!hasAssignUserPermission">
+    <h2>{{'Cluster.Role.NoPermission' | translate }}</h2>
+    <p class="text-muted">
+      {{'Cluster.Role.NoPermission.Help' | translate }}
+      <a href="{{helpDocUrl}}">{{'Common.LearnMore' | translate }}</a>
+    </p>
+  </div>

137-170: Optimize script loading for better performance

Consider the following improvements for script loading:

  1. Add integrity hashes for third-party scripts
  2. Use async/defer attributes where appropriate
  3. Consider bundling and minifying custom scripts

Example for third-party scripts:

-  <script src="../vendor/angular/angular.min.js"></script>
+  <script src="../vendor/angular/angular.min.js"
+          integrity="sha384-[hash]"
+          crossorigin="anonymous"></script>

For custom scripts:

-  <script type="application/javascript" src="../scripts/app.js"></script>
-  <script type="application/javascript" src="../scripts/services/AppService.js"></script>
+  <script type="application/javascript" src="../scripts/bundle.min.js" defer></script>
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between a37dc2f and 071b199.

📒 Files selected for processing (34)
  • apollo-configservice/src/test/java/com/ctrip/framework/apollo/configservice/integration/ConfigControllerIntegrationTest.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/component/PermissionValidator.java (3 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/PermissionType.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/RoleType.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ClusterController.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/CommitController.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ConfigsExportController.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ConfigsImportController.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ItemController.java (8 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/NamespaceBranchController.java (4 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/NamespaceController.java (2 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/PermissionController.java (4 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ReleaseController.java (5 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ReleaseHistoryController.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/entity/vo/ClusterNamespaceRolesAssignedUsers.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/AppService.java (6 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/ClusterService.java (2 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/RoleInitializationService.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/spi/defaultimpl/DefaultRoleInitializationService.java (2 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/util/RoleUtils.java (4 hunks)
  • apollo-portal/src/main/resources/static/cluster/role.html (1 hunks)
  • apollo-portal/src/main/resources/static/config.html (1 hunks)
  • apollo-portal/src/main/resources/static/i18n/en.json (1 hunks)
  • apollo-portal/src/main/resources/static/i18n/zh-CN.json (1 hunks)
  • apollo-portal/src/main/resources/static/scripts/controller/role/ClusterNamespaceRoleController.js (1 hunks)
  • apollo-portal/src/main/resources/static/scripts/directive/namespace-panel-directive.js (2 hunks)
  • apollo-portal/src/main/resources/static/scripts/services/PermissionService.js (7 hunks)
  • apollo-portal/src/main/resources/static/styles/common-style.css (1 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/ControllableAuthorizationConfiguration.java (1 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/SkipAuthorizationConfiguration.java (1 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/ItemControllerAuthIntegrationTest.java (1 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/PermissionControllerTest.java (1 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/service/AppServiceTest.java (2 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/spi/defaultImpl/RoleInitializationServiceTest.java (2 hunks)
🚧 Files skipped from review as they are similar to previous changes (28)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ClusterController.java
  • apollo-configservice/src/test/java/com/ctrip/framework/apollo/configservice/integration/ConfigControllerIntegrationTest.java
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/SkipAuthorizationConfiguration.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/RoleInitializationService.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/PermissionType.java
  • apollo-portal/src/main/resources/static/styles/common-style.css
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/service/AppServiceTest.java
  • apollo-portal/src/main/resources/static/scripts/directive/namespace-panel-directive.js
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/PermissionControllerTest.java
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/ControllableAuthorizationConfiguration.java
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/ItemControllerAuthIntegrationTest.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ConfigsImportController.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ConfigsExportController.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ReleaseHistoryController.java
  • apollo-portal/src/main/resources/static/config.html
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/CommitController.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/RoleType.java
  • apollo-portal/src/main/resources/static/scripts/controller/role/ClusterNamespaceRoleController.js
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/ClusterService.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/AppService.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/NamespaceController.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/entity/vo/ClusterNamespaceRolesAssignedUsers.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/NamespaceBranchController.java
  • apollo-portal/src/main/resources/static/i18n/en.json
  • apollo-portal/src/main/resources/static/i18n/zh-CN.json
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/util/RoleUtils.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/PermissionController.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ReleaseController.java
🔇 Additional comments (16)
apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/spi/defaultImpl/RoleInitializationServiceTest.java (3)

45-45: LGTM! Well-structured constant declaration.

The constant follows the established pattern and is effectively used across the new test methods.


170-185: LGTM! Well-structured negative test case.

The test effectively verifies that no unnecessary role or permission creation occurs when roles already exist.


188-206: LGTM! Good edge case coverage.

The test effectively verifies the partial initialization scenario where only the release role needs to be created.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/spi/defaultimpl/DefaultRoleInitializationService.java (1)

Line range hint 177-255: Verify integration with supporting components.

The new cluster-based permission model requires support from other components. Let's verify their existence:

✅ Verification successful

Integration with supporting components is properly implemented

The verification shows comprehensive integration with the required components:

  • Permission types MODIFY_NAMESPACE_IN_CLUSTER and RELEASE_NAMESPACE_IN_CLUSTER are properly defined in PermissionType.java and used consistently across the codebase
  • RoleUtils implements all required cluster-related methods:
    • buildModifyNamespaceInClusterRoleName
    • buildReleaseNamespaceInClusterRoleName
    • buildClusterTargetId
  • The functionality is properly integrated in PermissionValidator and PermissionController
  • Test coverage exists in RoleInitializationServiceTest
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Verify the existence of supporting code for cluster-based permissions

# Check for required permission types
echo "Checking for cluster permission types..."
rg "MODIFY_NAMESPACE_IN_CLUSTER|RELEASE_NAMESPACE_IN_CLUSTER" --type java

# Check for RoleUtils methods
echo "Checking for RoleUtils cluster methods..."
rg "buildModifyNamespaceInClusterRoleName|buildReleaseNamespaceInClusterRoleName|buildClusterTargetId" --type java

# Check for corresponding test coverage
echo "Checking for test coverage..."
fd -e java -e groovy -e scala test.*DefaultRoleInitializationService

Length of output: 6608

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/component/PermissionValidator.java (3)

69-80: Well-structured hierarchical permission check implementation.

The implementation follows the principle of least privilege while maintaining a clear hierarchy of permissions. The early returns optimize performance by avoiding unnecessary checks.


Line range hint 160-174: Security checks properly updated for cluster-based permissions.

The method has been correctly updated to incorporate cluster-based permissions while maintaining all existing security checks.


63-67: Implementation follows established permission check patterns.

The new cluster-level permission check is well-implemented, following the existing permission validation patterns. The method correctly uses cluster-specific permission types and target IDs.

Let's verify the consistency of permission types:

✅ Verification successful

Permission type is consistently defined and used across the codebase

The verification shows that MODIFY_NAMESPACE_IN_CLUSTER is properly defined in PermissionType.java and RoleType.java, and is consistently used across the codebase:

  • Defined as a constant in both permission and role type interfaces
  • Used in DefaultRoleInitializationService for role creation
  • Used in RoleUtils for target ID generation
  • Used in the new permission validator method

The implementation aligns perfectly with the existing permission framework.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Verify the permission type constant exists and is used consistently
rg "MODIFY_NAMESPACE_IN_CLUSTER" --type java

Length of output: 1154

apollo-portal/src/main/resources/static/scripts/services/PermissionService.js (3)

26-32: API endpoints follow RESTful design principles.

The new cluster namespace permission endpoints maintain consistency with existing patterns and properly implement RESTful principles.

Also applies to: 45-48


192-206: Permission check implementation follows established patterns.

The new cluster namespace permission check properly implements the promise-based pattern and includes appropriate error handling.


448-476: Service interface maintains consistency and clarity.

The new cluster namespace permission methods follow established naming conventions and properly map to their implementations.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ItemController.java (6)

90-91: LGTM - Consistent authorization pattern

The authorization check follows the same pattern as other methods, maintaining consistency in the permission model.


109-110: LGTM - Authorization check properly implemented

The permission check is consistent with the new model and properly validates cluster-level access.


Line range hint 123-134: Well-implemented authorization with additional security validation

The method properly implements cluster-level permission checks and includes an important security validation to prevent potential attack scenarios through namespace mismatch.


145-145: LGTM - Visibility check enhanced with cluster context

The visibility check has been properly updated to consider cluster-level permissions.


Line range hint 223-239: LGTM - Consistent permission checks

The permission checks in syntaxCheckText and revokeItems methods are properly implemented and consistent with the new cluster-based permission model.


76-77: Authorization check properly enhanced with cluster-level permission

The updated permission check now correctly includes cluster-level authorization, aligning with the new security model.

Let's verify the permission validator implementation:

✅ Verification successful

Authorization check implementation verified as correct and complete

The verification confirms that the permission check is properly implemented:

  1. The PermissionValidator class correctly implements the hasModifyNamespacePermission(appId, env, clusterName, namespaceName) method with a hierarchical permission check:

    • Checks app-level namespace permission
    • Checks environment-specific namespace permission
    • Checks cluster-level permission
  2. The permission check is consistently used across all item modification endpoints in both:

    • Regular web controllers (ItemController, NamespaceBranchController, ConfigsImportController)
    • OpenAPI controllers with their own consumer-specific validator
  3. The implementation aligns with the security model by enforcing proper authorization at all required levels (app, environment, cluster, namespace).

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Verify the hasModifyNamespacePermission implementation
ast-grep --pattern 'hasModifyNamespacePermission($appId, $env, $clusterName, $namespaceName)'

Length of output: 95


Script:

#!/bin/bash
# Let's try a different approach to find the permission validator implementation
rg -l "class.*PermissionValidator" 

Length of output: 234


Script:

#!/bin/bash
# Search for hasModifyNamespacePermission method definition
rg "hasModifyNamespacePermission.*\(" -A 5

Length of output: 25847


Script:

#!/bin/bash
# Look for other similar permission checks in the codebase
rg "@PreAuthorize.*hasModifyNamespacePermission" -A 1

Length of output: 6974

apollo-portal/src/main/resources/static/cluster/role.html Outdated Show resolved Hide resolved
apollo-portal/src/main/resources/static/cluster/role.html Outdated Show resolved Hide resolved
Copy link
Member

@nobodyiam nobodyiam left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Great Job! Please find some comments below.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 4

🔭 Outside diff range comments (1)
apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/spi/defaultImpl/RoleInitializationServiceTest.java (1)

Line range hint 261-266: Update mockPermission helper method for cluster permissions

The current mockPermission helper method only supports namespace-level permissions. Consider updating it to support cluster-level permissions:

   private Permission mockPermission(){
     Permission permission = new Permission();
-    permission.setPermissionType(PermissionType.MODIFY_NAMESPACE);
-    permission.setTargetId(RoleUtils.buildNamespaceTargetId(APP_ID, NAMESPACE));
+    permission.setPermissionType(PermissionType.MODIFY_NAMESPACE_IN_CLUSTER);
+    permission.setTargetId(RoleUtils.buildClusterTargetId(APP_ID, ENV, CLUSTER));
     return permission;
   }

Or make it more flexible:

-  private Permission mockPermission(){
+  private Permission mockPermission(PermissionType type, String targetId){
     Permission permission = new Permission();
-    permission.setPermissionType(PermissionType.MODIFY_NAMESPACE);
-    permission.setTargetId(RoleUtils.buildNamespaceTargetId(APP_ID, NAMESPACE));
+    permission.setPermissionType(type);
+    permission.setTargetId(targetId);
     return permission;
   }
🧹 Nitpick comments (19)
apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/spi/defaultImpl/RoleInitializationServiceTest.java (4)

45-45: Consider parameterizing environment tests

The hardcoded "DEV" environment constant might not cover all scenarios. Consider adding test cases for other environments (e.g., FAT) to ensure the role initialization works correctly across different environments.


148-167: Enhance test coverage with specific verifications

While the test verifies the number of method calls, it would be more robust to also verify:

  1. The exact role names being created
  2. The specific permission types being assigned
  3. The target IDs of the created permissions

Add these verifications:

       verify(rolePermissionService, times(2)).findRoleByRoleName(anyString());
       verify(rolePermissionService, times(2)).createPermission(any());
       verify(rolePermissionService, times(2)).createRoleWithPermissions(any(), anySet());
+      verify(rolePermissionService).createPermission(argThat(permission -> 
+          permission.getPermissionType() == PermissionType.MODIFY_NAMESPACE_IN_CLUSTER &&
+          permission.getTargetId().equals(RoleUtils.buildClusterTargetId(APP_ID, ENV, CLUSTER))
+      ));
+      verify(rolePermissionService).createPermission(argThat(permission ->
+          permission.getPermissionType() == PermissionType.RELEASE_NAMESPACE_IN_CLUSTER &&
+          permission.getTargetId().equals(RoleUtils.buildClusterTargetId(APP_ID, ENV, CLUSTER))
+      ));

187-206: Add specific verifications for release permission

Consider adding explicit verifications for the release permission creation:

       verify(rolePermissionService, times(2)).findRoleByRoleName(anyString());
       verify(rolePermissionService, times(1)).createPermission(any());
       verify(rolePermissionService, times(1)).createRoleWithPermissions(any(), anySet());
+      verify(rolePermissionService).createPermission(argThat(permission ->
+          permission.getPermissionType() == PermissionType.RELEASE_NAMESPACE_IN_CLUSTER &&
+          permission.getTargetId().equals(RoleUtils.buildClusterTargetId(APP_ID, ENV, CLUSTER))
+      ));

208-227: Add specific verifications for modify permission

Consider adding explicit verifications for the modify permission creation:

       verify(rolePermissionService, times(2)).findRoleByRoleName(anyString());
       verify(rolePermissionService, times(1)).createPermission(any());
       verify(rolePermissionService, times(1)).createRoleWithPermissions(any(), anySet());
+      verify(rolePermissionService).createPermission(argThat(permission ->
+          permission.getPermissionType() == PermissionType.MODIFY_NAMESPACE_IN_CLUSTER &&
+          permission.getTargetId().equals(RoleUtils.buildClusterTargetId(APP_ID, ENV, CLUSTER))
+      ));
apollo-portal/src/main/resources/static/cluster/ns_role.html (5)

20-30: Add responsive design and SEO meta tags

The document head is missing important meta tags for proper mobile support and SEO.

Add these meta tags:

  <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
+ <meta name="viewport" content="width=device-width, initial-scale=1">
+ <meta name="description" content="Apollo Configuration Center - Cluster Namespace Role Management">
  <link rel="icon" href="../img/config.png">

69-75: Standardize button disable variable naming

There's an inconsistency in the naming convention of button disable variables:

  • modifyRoleSubmitBtnDisabled
  • ReleaseRoleSubmitBtnDisabled (incorrect Pascal case)
- ng-disabled="ReleaseRoleSubmitBtnDisabled"
+ ng-disabled="releaseRoleSubmitBtnDisabled"

Also applies to: 100-106


63-123: Reduce code duplication in role assignment forms

The modify and release role forms share almost identical structure. Consider creating a reusable directive or component.

Create a new directive like:

<role-assignment-form
  role-type="ModifyNamespacesInCluster"
  title="{{ 'Cluster.Role.GrantModifyTo' | translate }}"
  subtitle="{{ 'Cluster.Role.GrantModifyTo2' | translate }}"
  assigned-users="rolesAssignedUsers.modifyRoleUsers"
  on-assign="assignRoleToUser"
  on-remove="removeUserRole">
</role-assignment-form>

62-62: Consider granular permission handling

The current implementation uses a single hasAssignUserPermission check. Consider implementing separate permissions for modify and release role management.

This would allow for more flexible access control where users might have permission to manage modify roles but not release roles, or vice versa.

Also applies to: 128-130


137-170: Optimize script loading and add integrity checks

The current script loading can be optimized for better page load performance and security:

  1. Consider using async/defer attributes for non-critical scripts
  2. Add integrity checks for external resources

Example implementation:

- <script src="../vendor/jquery.min.js" type="text/javascript"></script>
+ <script src="../vendor/jquery.min.js" 
+         integrity="sha384-[hash]" 
+         crossorigin="anonymous" 
+         defer></script>

Also consider bundling and minifying the application scripts to reduce HTTP requests.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ItemController.java (1)

200-220: Consider enhancing the permission check implementation.

While the current implementation is functional, here are some suggestions for improvement:

  1. Consider collecting all namespaces that the user lacks permission for, rather than breaking at the first one. This would help users fix all permission issues at once.
  2. The variable name hasPermission could be more specific, e.g., hasModifyPermissionForAllNamespaces.
  3. The comment on line 204 is redundant with the code logic.

Here's a suggested implementation:

-    NamespaceIdentifier noPermissionNamespace = null;
-    // check if user has every namespace's ModifyNamespace permission
-    boolean hasPermission = true;
-    for (NamespaceIdentifier namespaceIdentifier : model.getSyncToNamespaces()) {
-      // once user has not one of the namespace's ModifyNamespace permission, then break the loop
-      hasPermission = permissionValidator.hasModifyNamespacePermission(
-          namespaceIdentifier.getAppId(),
-          namespaceIdentifier.getEnv().getName(),
-          namespaceIdentifier.getClusterName(),
-          namespaceIdentifier.getNamespaceName()
-      );
-      if (!hasPermission) {
-        noPermissionNamespace = namespaceIdentifier;
-        break;
-      }
-    }
-    if (hasPermission) {
+    List<NamespaceIdentifier> noPermissionNamespaces = model.getSyncToNamespaces().stream()
+        .filter(namespace -> !permissionValidator.hasModifyNamespacePermission(
+            namespace.getAppId(),
+            namespace.getEnv().getName(),
+            namespace.getClusterName(),
+            namespace.getNamespaceName()))
+        .collect(Collectors.toList());
+    
+    if (noPermissionNamespaces.isEmpty()) {
       configService.syncItems(model.getSyncToNamespaces(), model.getSyncItems());
       return ResponseEntity.status(HttpStatus.OK).build();
     }
-    throw new AccessDeniedException(String.format("You don't have the permission to modify namespace: %s", noPermissionNamespace));
+    throw new AccessDeniedException(String.format("You don't have permission to modify the following namespaces: %s",
+        noPermissionNamespaces.stream()
+            .map(NamespaceIdentifier::toString)
+            .collect(Collectors.joining(", "))));
apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/spi/defaultimpl/DefaultRoleInitializationService.java (3)

177-189: Add input validation and documentation for the new method.

The method implementation looks good but could benefit from some improvements:

  1. Add input validation for parameters (appId, env, clusterName, operator)
  2. Add JavaDoc to document the method's purpose and parameters
  3. Consider adding debug logging for role creation events

Add parameter validation and documentation:

+ /**
+  * Initializes roles for managing namespaces within a specific cluster.
+  * Creates roles for modifying and releasing namespaces if they don't exist.
+  *
+  * @param appId The application ID
+  * @param env The environment name
+  * @param clusterName The cluster name
+  * @param operator The user performing the operation
+  * @throws IllegalArgumentException if any parameter is null or empty
+  */
  @Transactional
  @Override
  public void initClusterNamespaceRoles(String appId, String env, String clusterName, String operator) {
+   Preconditions.checkArgument(StringUtils.isNotBlank(appId), "AppId must not be blank");
+   Preconditions.checkArgument(StringUtils.isNotBlank(env), "Env must not be blank");
+   Preconditions.checkArgument(StringUtils.isNotBlank(clusterName), "ClusterName must not be blank");
+   Preconditions.checkArgument(StringUtils.isNotBlank(operator), "Operator must not be blank");
+
+   logger.debug("Initializing cluster namespace roles for app: {}, env: {}, cluster: {}", 
+                appId, env, clusterName);

246-255: Consider refactoring role creation methods to reduce duplication.

The implementation of createClusterRole follows a similar pattern to createNamespaceRole and createNamespaceEnvRole. Consider extracting the common logic into a more generic method.

Example refactoring:

+ private void createRoleWithPermission(String targetId, String permissionType,
+                                      String roleName, String operator) {
+   Permission permission = createPermission(targetId, permissionType, operator);
+   Permission createdPermission = rolePermissionService.createPermission(permission);
+
+   Role role = createRole(roleName, operator);
+   rolePermissionService
+       .createRoleWithPermissions(role, Sets.newHashSet(createdPermission.getId()));
+ }

  private void createClusterRole(String appId, String env, String clusterName, 
                               String permissionType, String roleName, String operator) {
-   Permission permission =
-       createPermission(RoleUtils.buildClusterTargetId(appId, env, clusterName), 
-                       permissionType, operator);
-   Permission createdPermission = rolePermissionService.createPermission(permission);
-
-   Role role = createRole(roleName, operator);
-   rolePermissionService
-       .createRoleWithPermissions(role, Sets.newHashSet(createdPermission.getId()));
+   createRoleWithPermission(RoleUtils.buildClusterTargetId(appId, env, clusterName),
+                           permissionType, roleName, operator);
  }

This refactoring can be applied to createNamespaceRole and createNamespaceEnvRole as well.


177-189: Document the new cluster-level permission model.

The implementation introduces a new level of permission granularity at the cluster level. This architectural change should be properly documented to help users and developers understand:

  1. The hierarchy of permissions (app > cluster > namespace)
  2. When to use cluster-level permissions vs namespace-level permissions
  3. Migration guide for existing applications

Consider adding this information to:

  1. Architecture documentation
  2. User guide
  3. Developer guide
apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/util/RoleUtils.java (1)

97-99: Refactor specific role builders to use this generic method.

This generic method could be used by both buildModifyNamespacesInClusterRoleName and buildReleaseNamespacesInClusterRoleName to reduce code duplication. Also needs input validation.

 public static String buildClusterRoleName(String appId, String env, String clusterName, String roleType) {
+    if (appId == null || env == null || clusterName == null || roleType == null) {
+        throw new IllegalArgumentException("All parameters must not be null");
+    }
+    if (appId.isEmpty() || env.isEmpty() || clusterName.isEmpty() || roleType.isEmpty()) {
+        throw new IllegalArgumentException("All parameters must not be empty");
+    }
     return STRING_JOINER.join(roleType, appId, env, clusterName);
 }

 public static String buildModifyNamespacesInClusterRoleName(String appId, String env, String clusterName) {
-    return STRING_JOINER.join(RoleType.MODIFY_NAMESPACES_IN_CLUSTER, appId, env, clusterName);
+    return buildClusterRoleName(appId, env, clusterName, RoleType.MODIFY_NAMESPACES_IN_CLUSTER);
 }

 public static String buildReleaseNamespacesInClusterRoleName(String appId, String env, String clusterName) {
-    return STRING_JOINER.join(RoleType.RELEASE_NAMESPACES_IN_CLUSTER, appId, env, clusterName);
+    return buildClusterRoleName(appId, env, clusterName, RoleType.RELEASE_NAMESPACES_IN_CLUSTER);
 }
apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/component/PermissionValidator.java (2)

69-80: Add JavaDoc to explain the permission hierarchy.

These methods implement a hierarchical permission check system, but the logic flow isn't immediately obvious. Consider adding documentation to explain:

  • The order of permission checks
  • The implications of each level of permission
  • Why the checks are short-circuited with return true
+/**
+ * Checks if the current user has permission to modify the namespace.
+ * The check follows this hierarchy:
+ * 1. Global namespace permission
+ * 2. Environment-specific namespace permission
+ * 3. Cluster-level permission
+ *
+ * @param appId The application ID
+ * @param env The environment
+ * @param clusterName The cluster name
+ * @param namespaceName The namespace name
+ * @return true if the user has any of the required permissions
+ */
 public boolean hasModifyNamespacePermission(String appId, String env, String clusterName, String namespaceName) {

Also applies to: 100-110


Line range hint 160-174: LGTM! Well-documented and secure implementation.

The method implements a secure-by-default approach with clear step-by-step validation. Consider extracting the validation steps into private methods for better testability.

 public boolean shouldHideConfigToCurrentUser(String appId, String env, String clusterName,
     String namespaceName) {
+    if (!isEnvironmentConfigViewRestricted(env)) {
+        return false;
+    }
+    if (isPublicNamespace(appId, namespaceName)) {
+        return false;
+    }
+    return !hasRequiredPermissions(appId, env, clusterName, namespaceName);
 }
+
+private boolean isEnvironmentConfigViewRestricted(String env) {
+    return portalConfig.isConfigViewMemberOnly(env);
+}
+
+private boolean isPublicNamespace(String appId, String namespaceName) {
+    AppNamespace appNamespace = appNamespaceService.findByAppIdAndName(appId, namespaceName);
+    return appNamespace != null && appNamespace.isPublic();
+}
+
+private boolean hasRequiredPermissions(String appId, String env, String clusterName, String namespaceName) {
+    return isAppAdmin(appId) || hasOperateNamespacePermission(appId, env, clusterName, namespaceName);
+}
apollo-portal/src/main/resources/static/scripts/services/PermissionService.js (3)

272-303: Add input validation for role management functions.

Consider adding input validation to ensure required parameters are not null/undefined before making the API calls. This would prevent unnecessary server calls and provide better error messages.

 function assignClusterNsRoleToUser(appId, env, clusterName, roleType, user) {
     var d = $q.defer();
+    if (!appId || !env || !clusterName || !roleType || !user) {
+        d.reject('Missing required parameters');
+        return d.promise;
+    }
     permission_resource.assign_cluster_ns_role_to_user({

448-476: Consider standardizing error handling across permission checks.

While the implementation is correct, consider standardizing error handling across all permission check methods. This would ensure consistent error reporting and make it easier to handle permission-related errors in the UI.

 has_modify_cluster_ns_permission: function (appId, env, clusterName) {
-    return hasClusterNsPermission(appId, env, clusterName, 'ModifyNamespacesInCluster');
+    return hasClusterNsPermission(appId, env, clusterName, 'ModifyNamespacesInCluster')
+        .catch(function(error) {
+            console.error('Failed to check modify cluster namespace permission:', error);
+            return false;
+        });
 },

Line range hint 26-476: Consider adding JSDoc documentation for new permission types.

The implementation is solid, but adding JSDoc documentation for the new cluster namespace permission types and their usage would improve maintainability and help other developers understand when to use cluster namespace permissions vs. regular namespace permissions.

Example documentation:

/**
 * Checks if the current user has permission to modify namespaces within a cluster.
 * This differs from namespace-level permissions as it grants access to all namespaces
 * within the specified cluster.
 * 
 * @param {string} appId - The application ID
 * @param {string} env - The environment name
 * @param {string} clusterName - The cluster name
 * @returns {Promise<boolean>} - Resolves to true if user has permission
 */
has_modify_cluster_ns_permission: function (appId, env, clusterName) {
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between 071b199 and a2acb3b.

📒 Files selected for processing (15)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/component/PermissionValidator.java (3 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/PermissionType.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/RoleType.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ItemController.java (8 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/PermissionController.java (4 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/entity/vo/ClusterNamespaceRolesAssignedUsers.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/spi/defaultimpl/DefaultRoleInitializationService.java (2 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/util/RoleUtils.java (4 hunks)
  • apollo-portal/src/main/resources/static/cluster/ns_role.html (1 hunks)
  • apollo-portal/src/main/resources/static/config.html (1 hunks)
  • apollo-portal/src/main/resources/static/scripts/controller/role/ClusterNamespaceRoleController.js (1 hunks)
  • apollo-portal/src/main/resources/static/scripts/directive/namespace-panel-directive.js (2 hunks)
  • apollo-portal/src/main/resources/static/scripts/services/PermissionService.js (7 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/PermissionControllerTest.java (1 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/spi/defaultImpl/RoleInitializationServiceTest.java (2 hunks)
🚧 Files skipped from review as they are similar to previous changes (8)
  • apollo-portal/src/main/resources/static/config.html
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/RoleType.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/PermissionType.java
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/PermissionControllerTest.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/entity/vo/ClusterNamespaceRolesAssignedUsers.java
  • apollo-portal/src/main/resources/static/scripts/directive/namespace-panel-directive.js
  • apollo-portal/src/main/resources/static/scripts/controller/role/ClusterNamespaceRoleController.js
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/PermissionController.java
🔇 Additional comments (9)
apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/spi/defaultImpl/RoleInitializationServiceTest.java (1)

169-185: LGTM!

The test effectively verifies that no unnecessary role or permission creations occur when both roles already exist.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ItemController.java (2)

145-145: LGTM! Improved permission checks and error messaging.

The permission checks have been properly updated to include cluster-level verification, and the error message has been enhanced to provide clearer context about the access denial.

The error message improvement aligns with the previous suggestion from reviewer nobodyiam.

Also applies to: 186-189


76-76: LGTM! Consistent implementation of cluster-level authorization.

The @PreAuthorize annotations have been consistently updated across all methods to include cluster-level permission checks. This aligns well with the PR objectives.

Let's verify that all callers have been updated to handle the new permission checks:

Also applies to: 90-90, 109-109, 123-123, 223-223, 234-234

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/spi/defaultimpl/DefaultRoleInitializationService.java (1)

Line range hint 177-255: Verify cluster role initialization flow.

The implementation looks solid, but we should verify:

  1. When and where cluster roles are initialized in the application lifecycle
  2. Integration with existing permission checks

Let's verify the initialization flow:

✅ Verification successful

Let's gather more information about the initialization flow and permission checks.


Let's get one more piece of information about the permission checks and role utils.


Cluster role initialization is properly integrated and secure

Based on the verification results, the cluster role initialization implementation is well-integrated and secure:

  1. Initialization is triggered at appropriate lifecycle points:

    • During cluster creation via ClusterService
    • During app creation via AppService for default cluster
    • Protected by @PreAuthorize permissions in controllers
  2. Permission model integration is complete:

    • Proper role naming conventions in RoleUtils
    • Permission checks implemented in PermissionValidator
    • OpenAPI permission validation via ConsumerPermissionValidator
    • Comprehensive test coverage in RoleInitializationServiceTest
  3. The implementation follows the established pattern:

    • Consistent role and permission creation
    • Proper transaction handling
    • Proper permission type enforcement
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Check where initClusterNamespaceRoles is called
rg -l "initClusterNamespaceRoles"

# Check for any existing cluster permission checks
ast-grep --pattern 'class $_ implements $_ {
  $$$
  buildClusterTargetId($_, $_, $_)
  $$$
}'

Length of output: 838


Script:

#!/bin/bash
# Check cluster role initialization in ClusterService
rg -A 10 "initClusterNamespaceRoles" apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/ClusterService.java

# Check cluster role initialization in AppService
rg -A 10 "initClusterNamespaceRoles" apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/AppService.java

# Check RoleUtils for cluster permission handling
ast-grep --pattern 'class RoleUtils {
  $$$
  buildClusterTargetId($_, $_, $_) {
    $$$
  }
  $$$
}'

# Check permission checks in PermissionController
rg -A 10 "hasCreateClusterPermission|hasModifyClusterPermission" apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/PermissionController.java

Length of output: 1702


Script:

#!/bin/bash
# Check RoleUtils implementation
rg -A 5 "buildClusterTargetId|buildModifyNamespacesInClusterRoleName|buildReleaseNamespacesInClusterRoleName" apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/util/RoleUtils.java

# Check permission verification in tests
rg -A 10 "initClusterNamespaceRoles" apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/spi/defaultImpl/RoleInitializationServiceTest.java

# Check for cluster permission checks in the codebase
rg "hasCreateClusterPermission|hasModifyClusterPermission|hasReleaseNamespacesPermission" -A 5

Length of output: 8930

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/component/PermissionValidator.java (2)

63-67: LGTM! Well-structured permission checks.

The cluster permission check methods are well-implemented with:

  • Consistent pattern across modify and release permissions
  • Proper encapsulation (private methods)
  • Clear separation of concerns

Also applies to: 94-97


117-119: LGTM! Clean implementation of operation permission check.

The method provides a clear and concise way to check both modify and release permissions, following the single responsibility principle.

apollo-portal/src/main/resources/static/scripts/services/PermissionService.js (3)

26-32: LGTM! Well-structured resource definitions.

The new resource definitions follow consistent patterns and REST best practices:

  • Proper HTTP methods (GET/POST/DELETE) for respective operations
  • Consistent URL structure with existing endpoints
  • Appropriate Content-Type headers for POST requests

Also applies to: 45-48, 102-115


192-206: LGTM! Consistent implementation of permission check.

The hasClusterNsPermission function follows the established patterns:

  • Proper promise handling
  • Consistent parameter passing
  • Error handling aligned with other functions

132-145: Verify if empty request body is intentional.

Unlike initAppNamespacePermission which passes a namespace parameter, initClusterNsPermission passes an empty object {} as the request body. Please verify if this is intentional or if cluster-specific data should be passed.

✅ Verification successful

Let me try a different approach to verify this by checking the permission initialization patterns in the codebase.

These scripts will help us understand:

  1. How other permission initialization methods are implemented
  2. The permission_resource configuration
  3. The backend API endpoint implementation to verify the expected request body

Let me try one more search to find the backend controller and related files.


Let me try one final search to find the complete permission resource definition.


Empty request body is correct for cluster namespace permission initialization

Based on the complete service definition, the empty request body {} in initClusterNsPermission is intentional and correct. Here's why:

  1. The endpoint /apps/:appId/envs/:env/clusters/:clusterName/initNsPermission already receives all necessary information through URL parameters (appId, env, clusterName)
  2. The function follows the same pattern as other permission initialization methods where the actual data needed for initialization is passed through URL parameters
  3. The Content-Type header is set to text/plain;charset=UTF-8, indicating that the endpoint is not expecting a JSON payload
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Check if other initialization endpoints accept empty bodies
rg -A 5 "init.*Permission.*\{\}" --type js

Length of output: 44


Script:

#!/bin/bash
# Search for permission initialization patterns
rg "permission_resource\.init" --type js -A 5

Length of output: 1408


Script:

#!/bin/bash
# Look for the permission_resource definition
ast-grep --pattern 'permission_resource = $rest_resource($url, $config)'

Length of output: 74


Script:

#!/bin/bash
# Check the API endpoint definition
fd -e java -e kt -x rg "initClusterNsPermission" {} \;

Length of output: 57


Script:

#!/bin/bash
# Search for cluster permission related endpoints in backend
rg "cluster.*permission" --type java -A 5

# Also check for the REST resource definition
rg "permission_resource\s*=" --type js -A 5

Length of output: 4308


Script:

#!/bin/bash
# Get the complete permission_resource definition with all endpoints
cat apollo-portal/src/main/resources/static/scripts/services/PermissionService.js

Length of output: 20846

@nobodyiam nobodyiam requested a review from Copilot January 5, 2025 01:35

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Copilot reviewed 17 out of 32 changed files in this pull request and generated no comments.

Files not reviewed (15)
  • apollo-portal/src/main/resources/static/cluster/ns_role.html: Language not supported
  • apollo-portal/src/main/resources/static/config.html: Language not supported
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/CommitController.java: Evaluated as low risk
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ReleaseHistoryController.java: Evaluated as low risk
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ConfigsExportController.java: Evaluated as low risk
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ConfigsImportController.java: Evaluated as low risk
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/NamespaceBranchController.java: Evaluated as low risk
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/NamespaceController.java: Evaluated as low risk
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ReleaseController.java: Evaluated as low risk
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/util/RoleUtils.java: Evaluated as low risk
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ItemController.java: Evaluated as low risk
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/RoleInitializationService.java: Evaluated as low risk
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/spi/defaultimpl/DefaultRoleInitializationService.java: Evaluated as low risk
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/component/PermissionValidator.java: Evaluated as low risk
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/RoleType.java: Evaluated as low risk
Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

🧹 Nitpick comments (1)
apollo-portal/src/main/resources/static/i18n/zh-CN.json (1)

364-372: LGTM! The translations are clear and consistent.

The new translations for cluster-level permission management follow the same pattern as existing namespace permission translations, making the UI consistent across different permission management screens.

Consider adding a period at the end of Cluster.Role.NoPermission for consistency with other similar messages in the file:

-  "Cluster.Role.NoPermission": "您没有权限哟!",
+  "Cluster.Role.NoPermission": "您没有权限哟!",
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between a2acb3b and e4c60c3.

📒 Files selected for processing (1)
  • apollo-portal/src/main/resources/static/i18n/zh-CN.json (1 hunks)

@BlackBear2003 BlackBear2003 force-pushed the feat-cluster-authorize-by-env branch from 61d6648 to 38e8989 Compare January 8, 2025 17:09
Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Caution

Inline review comments failed to post. This is likely due to GitHub's limits when posting large numbers of comments.

Actionable comments posted: 3

🧹 Nitpick comments (4)
apollo-portal/src/main/resources/static/i18n/en.json (1)

364-371: Consider adding more descriptive translations for better UX.

Consider adding these additional translation keys to enhance the user experience:

 {
   "Cluster.Role.Title": "Cluster Permission Management",
+  "Cluster.Role.Description": "Manage permissions for modifying and publishing configurations at the cluster level",
+  "Cluster.Role.ValidationError": "Please select at least one permission type",
+  "Cluster.Role.SuccessMessage": "Cluster permissions updated successfully",
+  "Cluster.Role.ErrorMessage": "Failed to update cluster permissions"
 }
apollo-portal/src/main/resources/static/scripts/controller/ClusterManageController.js (1)

17-46: Add JSDoc documentation for the controller.

The controller lacks documentation explaining its purpose, dependencies, and functionality.

Add this documentation at the start of the controller:

+/**
+ * Controller for managing cluster permissions across different environments.
+ * @param {Object} $scope - Angular scope object
+ * @param {Object} $location - Angular location service
+ * @param {Object} $window - Angular window service
+ * @param {Object} $translate - Angular translate service
+ * @param {Object} toastr - Toastr notification service
+ * @param {Object} AppService - Service for app-related operations
+ * @param {Object} EnvService - Service for environment-related operations
+ * @param {Object} ClusterService - Service for cluster-related operations
+ * @param {Object} AppUtil - Utility service for app-related operations
+ */
 cluster_manage_module.controller('ClusterManageController',
     ['$scope', '$location', '$window', '$translate', 'toastr', 'AppService', 'EnvService', 'ClusterService',
apollo-portal/src/main/resources/static/app/cluster_manage.html (1)

81-85: Enhance accessibility for cluster name display.

Add ARIA attributes to improve accessibility for screen readers.

 <div class="col-md-6 col-sm-6 header-namespace">
-  <b class="namespace-name" data-tooltip="tooltip" data-placement="bottom">
+  <b class="namespace-name" data-tooltip="tooltip" data-placement="bottom" 
+     role="heading" aria-level="3" aria-label="Cluster name">
     {{'Common.Cluster' | translate }}:{{cluster.name}}
   </b>
 </div>
apollo-portal/src/main/resources/static/i18n/zh-CN.json (1)

364-372: Add missing translation for cluster role management error.

Add a translation key for error scenarios in cluster role management.

   "Cluster.Role.Add": "添加",
   "Cluster.Role.NoPermission": "您没有权限哟!",
+  "Cluster.Role.AddError": "添加角色失败",
+  "Cluster.Role.DeleteError": "删除角色失败",
🛑 Comments failed to post (3)
apollo-portal/src/main/resources/static/scripts/controller/ClusterManageController.js (1)

28-40: 🛠️ Refactor suggestion

Enhance error handling in loadClusters function.

The error handling could be improved in the following ways:

  1. Add error callback for the promise chain
  2. Add specific error message for different failure scenarios
  3. Remove console.log in production code
 function loadClusters() {
   AppService.load_nav_tree($scope.appId).then(function (result) {
     var nodes = AppUtil.collectData(result);
     if (!nodes || nodes.length == 0) {
       toastr.error($translate.instant('Config.SystemError'));
       return;
     }
     nodes.forEach(function (node) {
       $scope.envs.push({ name: node.env, clusters: node.clusters });
     });
-    console.log($scope.envs);
+  }, function (error) {
+    toastr.error($translate.instant('Config.LoadingAllNamespaceError'));
   });
 }
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

        function loadClusters() {
          AppService.load_nav_tree($scope.appId).then(function (result) {
            var nodes = AppUtil.collectData(result);
            if (!nodes || nodes.length == 0) {
              toastr.error($translate.instant('Config.SystemError'));
              return;
            }
            nodes.forEach(function (node) {
              $scope.envs.push({ name: node.env, clusters: node.clusters });
            });
          }, function (error) {
            toastr.error($translate.instant('Config.LoadingAllNamespaceError'));
          });
        }
apollo-portal/src/main/resources/static/app/cluster_manage.html (2)

56-71: 🛠️ Refactor suggestion

Remove commented-out code.

Remove the commented-out section that appears to be unused tips/documentation. If this content is needed later, it should be tracked in a separate issue.


90-90: 🛠️ Refactor suggestion

Fix the image path for the assign icon.

The image path should use the prefixPath filter for consistency with other resources.

-    <img src="/img/assign.png">
+    <img src="{{ '/img/assign.png' | prefixPath }}">
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

                    <img src="{{ '/img/assign.png' | prefixPath }}">

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

♻️ Duplicate comments (1)
apollo-portal/src/main/resources/static/cluster/ns_role.html (1)

56-58: ⚠️ Potential issue

Secure the return URL construction

The URL construction using string concatenation could be vulnerable to XSS attacks. Consider using Angular's $sce service for URL sanitization.

- href="{{ '/config.html' | prefixPath }}?#appid={{pageContext.appId}}"
+ href="{{ '/config.html' | prefixPath }}?#appid={{pageContext.appId | encodeURIComponent}}"
🧹 Nitpick comments (12)
apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/spi/defaultImpl/RoleInitializationServiceTest.java (5)

148-167: LGTM! Consider adding a descriptive Javadoc.

The test effectively verifies the base case scenario where both modify and release roles need to be created. Consider adding a Javadoc comment to describe the test's purpose and expectations.

/**
 * Tests the initialization of cluster namespace roles when neither the modify
 * nor release roles exist. Verifies that both roles and their permissions
 * are created correctly.
 */

169-185: LGTM! Consider adding a descriptive Javadoc.

The test effectively verifies that no unnecessary role or permission creation occurs when roles already exist. Consider adding a Javadoc comment to describe the test's purpose.

/**
 * Tests the initialization of cluster namespace roles when both modify and
 * release roles already exist. Verifies that no new roles or permissions
 * are created in this case.
 */

187-206: LGTM! Consider adding a descriptive Javadoc.

The test effectively verifies the partial existence case where only the modify role exists. Consider adding a Javadoc comment to describe the test's purpose.

/**
 * Tests the initialization of cluster namespace roles when only the modify
 * role exists. Verifies that only the release role and its permissions
 * are created while the existing modify role remains unchanged.
 */

208-227: LGTM! Consider adding a descriptive Javadoc.

The test effectively verifies the partial existence case where only the release role exists. Consider adding a Javadoc comment to describe the test's purpose.

/**
 * Tests the initialization of cluster namespace roles when only the release
 * role exists. Verifies that only the modify role and its permissions
 * are created while the existing release role remains unchanged.
 */

148-227: Consider adding error scenario tests.

The current test suite thoroughly covers the success scenarios for cluster namespace role initialization. Consider adding tests for error scenarios, such as:

  • Role creation failures
  • Permission creation failures
  • Invalid input parameters (null/empty cluster name, invalid environment)

This would ensure robust error handling in the implementation.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ItemController.java (1)

200-220: Consider refactoring permission check loop for better readability

While the implementation is functionally correct, the code structure could be improved for better maintainability.

Consider extracting the permission check loop into a separate method:

+  private NamespaceIdentifier validateNamespacePermissions(List<NamespaceIdentifier> namespaces) {
+    for (NamespaceIdentifier namespace : namespaces) {
+      boolean hasPermission = permissionValidator.hasModifyNamespacePermission(
+          namespace.getAppId(),
+          namespace.getEnv().getName(),
+          namespace.getClusterName(),
+          namespace.getNamespaceName()
+      );
+      if (!hasPermission) {
+        return namespace;
+      }
+    }
+    return null;
+  }

   @PutMapping(value = "/apps/{appId}/namespaces/{namespaceName}/items", consumes = {"application/json"})
   public ResponseEntity<Void> update(@PathVariable String appId, @PathVariable String namespaceName,
                                    @RequestBody NamespaceSyncModel model) {
     checkModel(!model.isInvalid() && model.syncToNamespacesValid(appId, namespaceName));
-    NamespaceIdentifier noPermissionNamespace = null;
-    boolean hasPermission = true;
-    for (NamespaceIdentifier namespaceIdentifier : model.getSyncToNamespaces()) {
-      hasPermission = permissionValidator.hasModifyNamespacePermission(
-          namespaceIdentifier.getAppId(),
-          namespaceIdentifier.getEnv().getName(),
-          namespaceIdentifier.getClusterName(),
-          namespaceIdentifier.getNamespaceName()
-      );
-      if (!hasPermission) {
-        noPermissionNamespace = namespaceIdentifier;
-        break;
-      }
-    }
-    if (hasPermission) {
+    NamespaceIdentifier noPermissionNamespace = validateNamespacePermissions(model.getSyncToNamespaces());
+    if (noPermissionNamespace == null) {
       configService.syncItems(model.getSyncToNamespaces(), model.getSyncItems());
       return ResponseEntity.status(HttpStatus.OK).build();
     }
     throw new AccessDeniedException(String.format("You don't have the permission to modify namespace: %s", noPermissionNamespace));
   }
apollo-portal/src/main/resources/static/config.html (1)

143-145: Consider using a distinct icon for cluster management.

The implementation aligns well with the agreed approach from previous discussions. However, consider using a distinct icon instead of reusing 'project-manage' to better differentiate this feature.

-                        apollo-img-src="'project-manage'"
+                        apollo-img-src="'cluster-manage'"
apollo-portal/src/main/resources/static/scripts/services/PermissionService.js (2)

102-115: Consider adding error response handling documentation.

While the role management resources are well-defined, it would be helpful to document the expected error responses for better API consumption.

Add JSDoc comments to describe error scenarios:

+        /**
+         * Get users with roles in a cluster namespace
+         * @throws {Error} 403 if the current user lacks permission
+         * @throws {Error} 404 if the cluster or namespace doesn't exist
+         */
         get_cluster_ns_role_users: {

272-303: Consider consolidating duplicate promise handling code.

The role management functions share similar promise handling patterns. Consider extracting the common promise handling logic into a utility function.

+ function handlePromise(resourceCall) {
+   var d = $q.defer();
+   resourceCall(
+     function (result) { d.resolve(result); },
+     function (result) { d.reject(result); }
+   );
+   return d.promise;
+ }

 function assignClusterNsRoleToUser(appId, env, clusterName, roleType, user) {
-  var d = $q.defer();
-  permission_resource.assign_cluster_ns_role_to_user({
+  return handlePromise(() => permission_resource.assign_cluster_ns_role_to_user({
       appId: appId,
       env: env,
       clusterName: clusterName,
       roleType: roleType
-    }, user,
-    function (result) {
-      d.resolve(result);
-    }, function (result) {
-      d.reject(result);
-    });
-  return d.promise;
+    }, user));
 }
apollo-portal/src/main/resources/static/cluster/ns_role.html (2)

74-74: Fix inconsistent variable naming

The button disable state variables have inconsistent naming conventions:

  • modifyRoleSubmitBtnDisabled (camelCase)
  • ReleaseRoleSubmitBtnDisabled (PascalCase)

Maintain consistent naming conventions across the codebase.

- ng-disabled="ReleaseRoleSubmitBtnDisabled"
+ ng-disabled="releaseRoleSubmitBtnDisabled"

Also applies to: 105-105


138-155: Consider implementing Subresource Integrity (SRI) for external scripts

Add integrity hashes to script tags to prevent unauthorized modifications of externally loaded resources.

Example:

- <script src="../vendor/jquery.min.js" type="text/javascript"></script>
+ <script src="../vendor/jquery.min.js" 
+         integrity="sha384-[calculated-hash]" 
+         crossorigin="anonymous" 
+         type="text/javascript"></script>
apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/PermissionController.java (1)

213-235: LGTM! New endpoint for retrieving cluster namespace roles.

The endpoint correctly retrieves both modify and release role users for a cluster.

However, consider adding @PreAuthorize annotation for consistency with other role retrieval endpoints.

+  @PreAuthorize(value = "@permissionValidator.hasAssignRolePermission(#appId)")
   @GetMapping("/apps/{appId}/envs/{env}/clusters/{clusterName}/ns_role_users")
   public ClusterNamespaceRolesAssignedUsers getClusterNamespaceRoles(...)
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between 61d6648 and 38e8989.

📒 Files selected for processing (35)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/component/PermissionValidator.java (3 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/PermissionType.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/RoleType.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/CommitController.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ConfigsExportController.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ConfigsImportController.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ItemController.java (8 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/NamespaceBranchController.java (4 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/NamespaceController.java (2 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/PermissionController.java (4 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ReleaseController.java (5 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ReleaseHistoryController.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/entity/vo/ClusterNamespaceRolesAssignedUsers.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/AppService.java (6 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/ClusterService.java (2 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/RoleInitializationService.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/spi/defaultimpl/DefaultRoleInitializationService.java (2 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/util/RoleUtils.java (4 hunks)
  • apollo-portal/src/main/resources/static/app/cluster_manage.html (1 hunks)
  • apollo-portal/src/main/resources/static/cluster/ns_role.html (1 hunks)
  • apollo-portal/src/main/resources/static/config.html (1 hunks)
  • apollo-portal/src/main/resources/static/i18n/en.json (2 hunks)
  • apollo-portal/src/main/resources/static/i18n/zh-CN.json (2 hunks)
  • apollo-portal/src/main/resources/static/scripts/app.js (1 hunks)
  • apollo-portal/src/main/resources/static/scripts/controller/ClusterManageController.js (1 hunks)
  • apollo-portal/src/main/resources/static/scripts/controller/role/ClusterNamespaceRoleController.js (1 hunks)
  • apollo-portal/src/main/resources/static/scripts/directive/namespace-panel-directive.js (2 hunks)
  • apollo-portal/src/main/resources/static/scripts/services/PermissionService.js (7 hunks)
  • apollo-portal/src/main/resources/static/styles/common-style.css (1 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/ControllableAuthorizationConfiguration.java (1 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/SkipAuthorizationConfiguration.java (1 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/ItemControllerAuthIntegrationTest.java (1 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/PermissionControllerTest.java (1 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/service/AppServiceTest.java (2 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/spi/defaultImpl/RoleInitializationServiceTest.java (2 hunks)
🚧 Files skipped from review as they are similar to previous changes (25)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/SkipAuthorizationConfiguration.java
  • apollo-portal/src/main/resources/static/scripts/app.js
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/service/AppServiceTest.java
  • apollo-portal/src/main/resources/static/styles/common-style.css
  • apollo-portal/src/main/resources/static/scripts/directive/namespace-panel-directive.js
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ConfigsExportController.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/PermissionType.java
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/PermissionControllerTest.java
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/ControllableAuthorizationConfiguration.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/AppService.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ReleaseHistoryController.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/NamespaceController.java
  • apollo-portal/src/main/resources/static/scripts/controller/role/ClusterNamespaceRoleController.js
  • apollo-portal/src/main/resources/static/scripts/controller/ClusterManageController.js
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/CommitController.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/ClusterService.java
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/ItemControllerAuthIntegrationTest.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/entity/vo/ClusterNamespaceRolesAssignedUsers.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/RoleInitializationService.java
  • apollo-portal/src/main/resources/static/app/cluster_manage.html
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ConfigsImportController.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/NamespaceBranchController.java
  • apollo-portal/src/main/resources/static/i18n/zh-CN.json
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ReleaseController.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/util/RoleUtils.java
⏰ Context from checks skipped due to timeout of 90000ms (1)
  • GitHub Check: build (8)
🔇 Additional comments (29)
apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/spi/defaultImpl/RoleInitializationServiceTest.java (1)

45-45: LGTM! Clear and consistent constant declaration.

The ENV constant follows the established pattern and is appropriately used across the new test methods.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ItemController.java (3)

76-76: LGTM! Consistent implementation of cluster-level authorization

The @PreAuthorize annotations have been properly updated to include cluster-level permission checks, which aligns well with the new authorization model.

Also applies to: 90-90, 109-109, 123-123, 223-223, 234-234


145-145: LGTM! Proper implementation of cluster-aware visibility check

The visibility check has been correctly updated to include cluster-level permissions.


186-189: Consider using previously suggested error message format

The error message format was previously reviewed and a suggestion was made to improve its clarity.

-        diff.setExtInfo("You are not this project's administrator, nor you have edit or release permission for the namespace: " + namespace);
+        diff.setExtInfo("You are not this project's administrator, nor you have edit or release permission for the namespace: " + namespace.getNamespaceName());
apollo-portal/src/main/resources/static/config.html (1)

143-145: Verify environment/cluster context visibility after header removal.

The removal of the header element that displayed the current environment and cluster name might impact user awareness of their operational context. Please verify that this information remains clearly visible elsewhere in the UI to prevent accidental modifications in the wrong environment.

✅ Verification successful

Environment and cluster context remains clearly visible after header removal

The environment and cluster information remains prominently displayed through:

  • Alert/info boxes with cluster-specific context
  • Clear labeling in related views (history, diff, sync)
  • Consistent context display in all operational modals
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for other UI elements that might display environment/cluster context
rg -l "pageContext\.(env|clusterName)" --type html

Length of output: 339


Script:

#!/bin/bash
# Check how environment/cluster context is used in these files
rg "pageContext\.(env|clusterName)" --type html -A 2 -B 2

Length of output: 9472

apollo-portal/src/main/resources/static/i18n/en.json (1)

364-372: LGTM! Well-structured translation keys.

The new translation keys for cluster permission management:

  • Follow consistent naming patterns
  • Provide clear and descriptive messages
  • Maintain proper formatting
apollo-portal/src/main/resources/static/scripts/services/PermissionService.js (5)

26-32: LGTM! Resource definition follows established patterns.

The new init_cluster_ns_permission resource is well-structured and consistent with existing patterns in the codebase.


45-48: LGTM! Permission check endpoint is properly defined.

The has_cluster_ns_permission resource follows the RESTful URL pattern and matches the backend controller's endpoint structure.


192-206: LGTM! Permission check implementation is consistent.

The hasClusterNsPermission function follows the established pattern for permission checks and properly handles promises.


448-476: LGTM! Service method exports are well-organized.

The new cluster namespace permission methods are properly exposed and follow the established naming conventions.


132-145: Verify error handling in initClusterNsPermission.

The function should handle specific error cases that might occur during permission initialization.

apollo-portal/src/main/resources/static/cluster/ns_role.html (3)

112-116: Review dropdown implementation

Similar to the modify permission section, the button has unused dropdown-toggle attributes.

- <button type="button" class="btn btn-default dropdown-toggle"
-         data-toggle="dropdown" aria-haspopup="true" aria-expanded="false"
+ <button type="button" class="btn btn-default"
          ng-click="removeUserRole('ReleaseNamespacesInCluster', user.userId)">

17-173: Overall implementation review

The implementation follows Angular best practices with proper:

  • Separation of concerns
  • Internationalization
  • Permission-based access control
  • Component reuse (apollouserselector)
  • Dependency management

The identified issues are:

  1. URL sanitization in the return link
  2. Inconsistent variable naming conventions
  3. Unused dropdown attributes
  4. Missing SRI for external scripts

Despite these issues, the core functionality appears solid.


81-85: Review dropdown implementation

The button has dropdown-toggle attributes but no corresponding dropdown menu implementation. Either:

  1. Remove the dropdown-related attributes if not needed, or
  2. Implement the missing dropdown menu
- <button type="button" class="btn btn-default dropdown-toggle"
-         data-toggle="dropdown" aria-haspopup="true" aria-expanded="false"
+ <button type="button" class="btn btn-default"
          ng-click="removeUserRole('ModifyNamespacesInCluster', user.userId)">
✅ Verification successful

Remove unnecessary dropdown attributes

The button is a simple remove action button and doesn't need dropdown functionality. The dropdown-related attributes should be removed as suggested in the review.

  • Remove dropdown-toggle, data-toggle="dropdown", aria-haspopup="true", and aria-expanded="false" from both remove buttons in the file.
🏁 Scripts executed

The following scripts were executed for the analysis:

Error: No shell scripts found


Script:

#!/bin/bash
# Get the full content of the file to see surrounding context
cat "apollo-portal/src/main/resources/static/cluster/ns_role.html"

# Search for similar button patterns with dropdown-toggle
rg "btn-default dropdown-toggle" --type html

Length of output: 9532

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/RoleType.java (3)

27-29: LGTM! New role type for cluster-level modification permission.

The constant follows the established naming convention and aligns with the PR objective of implementing cluster-level authorization.


29-30: LGTM! New role type for cluster-level release permission.

The constant follows the established naming convention and aligns with the PR objective of implementing cluster-level authorization.


32-33: LGTM! Updated role type validation.

The method correctly validates all role types, including the newly added cluster-level permissions.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/component/PermissionValidator.java (6)

51-60: LGTM! Improved encapsulation of namespace permission methods.

Changed visibility from public to private enhances encapsulation, as these methods are implementation details.


63-67: LGTM! New method for cluster-level modify permission check.

The method correctly checks permissions using the cluster target ID.


69-80: LGTM! Comprehensive permission check for namespace modification.

The method follows a hierarchical check:

  1. Global namespace permission
  2. Environment-specific namespace permission
  3. Cluster-level permission

82-92: LGTM! Improved encapsulation of release permission methods.

Changed visibility from public to private enhances encapsulation, as these methods are implementation details.


94-97: LGTM! New method for cluster-level release permission check.

The method correctly checks permissions using the cluster target ID.


100-110: LGTM! Comprehensive permission check for namespace release.

The method follows the same hierarchical check pattern as the modify permission check.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/spi/defaultimpl/DefaultRoleInitializationService.java (2)

177-189: LGTM! New method for initializing cluster namespace roles.

The method follows the established pattern of role initialization and correctly handles both modify and release permissions for cluster namespaces.


246-255: LGTM! New method for creating cluster roles.

The method follows the same pattern as other role creation methods in the class, ensuring consistency.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/PermissionController.java (4)

83-87: LGTM! New endpoint for initializing cluster namespace permissions.

The endpoint correctly initializes roles and follows REST conventions.


123-133: LGTM! New endpoint for checking cluster namespace permissions.

The endpoint correctly checks permissions and follows REST conventions.


237-259: LGTM! New endpoint for assigning cluster namespace roles.

The endpoint includes proper authorization, input validation, and error handling.


261-277: LGTM! New endpoint for removing cluster namespace roles.

The endpoint includes proper authorization, input validation, and error handling.

Copy link
Member

@nobodyiam nobodyiam left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This PR is nearly ready to merge. Please see the comments below.

@BlackBear2003
Copy link
Member Author

Fixed all changes requested, great thanks for code reviewing!🫡

@nobodyiam
Copy link
Member

Should we add usage instructions for this feature to the apollo-user-guide?

@BlackBear2003 BlackBear2003 force-pushed the feat-cluster-authorize-by-env branch from ef94b09 to 88ce290 Compare January 12, 2025 06:19
Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 2

🧹 Nitpick comments (3)
apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ItemController.java (1)

200-220: Consider enhancing the permission validation implementation.

While the implementation is functional, here are some suggestions for improvement:

  1. Consider collecting all namespaces without permission to provide complete feedback.
  2. The variable name 'hasPermission' could be more specific, e.g., 'hasRequiredPermissions'.
  3. Include environment and cluster information in the error message for better context.

Here's a suggested implementation:

-    NamespaceIdentifier noPermissionNamespace = null;
-    // check if user has every namespace's ModifyNamespace permission
-    boolean hasPermission = true;
-    for (NamespaceIdentifier namespaceIdentifier : model.getSyncToNamespaces()) {
-      // once user has not one of the namespace's ModifyNamespace permission, then break the loop
-      hasPermission = permissionValidator.hasModifyNamespacePermission(
-          namespaceIdentifier.getAppId(),
-          namespaceIdentifier.getEnv().getName(),
-          namespaceIdentifier.getClusterName(),
-          namespaceIdentifier.getNamespaceName()
-      );
-      if (!hasPermission) {
-        noPermissionNamespace = namespaceIdentifier;
-        break;
-      }
-    }
-    if (hasPermission) {
+    List<NamespaceIdentifier> noPermissionNamespaces = model.getSyncToNamespaces().stream()
+        .filter(namespace -> !permissionValidator.hasModifyNamespacePermission(
+            namespace.getAppId(),
+            namespace.getEnv().getName(),
+            namespace.getClusterName(),
+            namespace.getNamespaceName()))
+        .collect(Collectors.toList());
+    
+    if (noPermissionNamespaces.isEmpty()) {
       configService.syncItems(model.getSyncToNamespaces(), model.getSyncItems());
       return ResponseEntity.status(HttpStatus.OK).build();
     }
-    throw new AccessDeniedException(String.format("You don't have the permission to modify namespace: %s", noPermissionNamespace));
+    NamespaceIdentifier firstNoPermission = noPermissionNamespaces.get(0);
+    throw new AccessDeniedException(
+        String.format("You don't have the permission to modify namespace: %s in env: %s, cluster: %s",
+            firstNoPermission.getNamespaceName(),
+            firstNoPermission.getEnv().getName(),
+            firstNoPermission.getClusterName()));
apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/spi/defaultimpl/DefaultRoleInitializationService.java (1)

177-189: LGTM! Consider adding method documentation.

The implementation correctly initializes cluster namespace roles following the established pattern. The method properly checks for existing roles before creation and is wrapped in a transaction.

Add Javadoc to document:

  • Purpose of the method
  • Parameter descriptions
  • Transaction behavior
apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/spi/defaultImpl/RoleInitializationServiceTest.java (1)

148-167: Enhance test assertions for better coverage.

While the test verifies the number of method calls correctly, consider adding assertions to verify:

  • The actual role names being created
  • The permission types being assigned
  • The target IDs being generated
 @Test
 public void testInitClusterNsRole() {
     String modifyNamespacesInClusterRoleName = RoleUtils.buildModifyNamespacesInClusterRoleName(APP_ID, ENV, CLUSTER);
     when(rolePermissionService.findRoleByRoleName(modifyNamespacesInClusterRoleName)).
         thenReturn(null);

     String releaseNamespacesInClusterRoleName = RoleUtils.buildReleaseNamespacesInClusterRoleName(APP_ID, ENV, CLUSTER);
     when(rolePermissionService.findRoleByRoleName(releaseNamespacesInClusterRoleName)).
         thenReturn(null);

     when(userInfoHolder.getUser()).thenReturn(mockUser());
     when(rolePermissionService.createPermission(any())).thenReturn(mockPermission());

     roleInitializationService.initClusterNamespaceRoles(APP_ID, ENV, CLUSTER, CURRENT_USER);

     verify(rolePermissionService, times(2)).findRoleByRoleName(anyString());
     verify(rolePermissionService, times(2)).createPermission(any());
     verify(rolePermissionService, times(2)).createRoleWithPermissions(any(), anySet());
+    
+    // Verify correct role names
+    verify(rolePermissionService).findRoleByRoleName(modifyNamespacesInClusterRoleName);
+    verify(rolePermissionService).findRoleByRoleName(releaseNamespacesInClusterRoleName);
+    
+    // Verify permission creation with correct types
+    ArgumentCaptor<Permission> permissionCaptor = ArgumentCaptor.forClass(Permission.class);
+    verify(rolePermissionService, times(2)).createPermission(permissionCaptor.capture());
+    List<Permission> createdPermissions = permissionCaptor.getAllValues();
+    assertEquals(PermissionType.MODIFY_NAMESPACES_IN_CLUSTER, createdPermissions.get(0).getPermissionType());
+    assertEquals(PermissionType.RELEASE_NAMESPACES_IN_CLUSTER, createdPermissions.get(1).getPermissionType());
+    
+    // Verify target IDs
+    String expectedTargetId = RoleUtils.buildClusterTargetId(APP_ID, ENV, CLUSTER);
+    assertEquals(expectedTargetId, createdPermissions.get(0).getTargetId());
+    assertEquals(expectedTargetId, createdPermissions.get(1).getTargetId());
 }
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between ef94b09 and 88ce290.

📒 Files selected for processing (35)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/component/PermissionValidator.java (3 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/PermissionType.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/RoleType.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/CommitController.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ConfigsExportController.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ConfigsImportController.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ItemController.java (8 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/NamespaceBranchController.java (4 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/NamespaceController.java (2 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/PermissionController.java (4 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ReleaseController.java (5 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ReleaseHistoryController.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/entity/vo/ClusterNamespaceRolesAssignedUsers.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/AppService.java (6 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/ClusterService.java (2 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/RoleInitializationService.java (1 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/spi/defaultimpl/DefaultRoleInitializationService.java (2 hunks)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/util/RoleUtils.java (4 hunks)
  • apollo-portal/src/main/resources/static/app/cluster_manage.html (1 hunks)
  • apollo-portal/src/main/resources/static/cluster/ns_role.html (1 hunks)
  • apollo-portal/src/main/resources/static/config.html (1 hunks)
  • apollo-portal/src/main/resources/static/i18n/en.json (2 hunks)
  • apollo-portal/src/main/resources/static/i18n/zh-CN.json (3 hunks)
  • apollo-portal/src/main/resources/static/scripts/app.js (1 hunks)
  • apollo-portal/src/main/resources/static/scripts/controller/ClusterManageController.js (1 hunks)
  • apollo-portal/src/main/resources/static/scripts/controller/role/ClusterNamespaceRoleController.js (1 hunks)
  • apollo-portal/src/main/resources/static/scripts/directive/namespace-panel-directive.js (2 hunks)
  • apollo-portal/src/main/resources/static/scripts/services/PermissionService.js (7 hunks)
  • apollo-portal/src/main/resources/static/styles/common-style.css (1 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/ControllableAuthorizationConfiguration.java (1 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/SkipAuthorizationConfiguration.java (1 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/ItemControllerAuthIntegrationTest.java (1 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/PermissionControllerTest.java (1 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/service/AppServiceTest.java (2 hunks)
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/spi/defaultImpl/RoleInitializationServiceTest.java (2 hunks)
🚧 Files skipped from review as they are similar to previous changes (30)
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/RoleInitializationService.java
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/SkipAuthorizationConfiguration.java
  • apollo-portal/src/main/resources/static/scripts/app.js
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/service/AppServiceTest.java
  • apollo-portal/src/main/resources/static/styles/common-style.css
  • apollo-portal/src/main/resources/static/scripts/directive/namespace-panel-directive.js
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/RoleType.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/constant/PermissionType.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ConfigsExportController.java
  • apollo-portal/src/main/resources/static/scripts/controller/ClusterManageController.js
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ConfigsImportController.java
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/PermissionControllerTest.java
  • apollo-portal/src/main/resources/static/app/cluster_manage.html
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/ClusterService.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/CommitController.java
  • apollo-portal/src/main/resources/static/scripts/controller/role/ClusterNamespaceRoleController.js
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ReleaseHistoryController.java
  • apollo-portal/src/main/resources/static/config.html
  • apollo-portal/src/main/resources/static/cluster/ns_role.html
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/service/AppService.java
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/ControllableAuthorizationConfiguration.java
  • apollo-portal/src/main/resources/static/scripts/services/PermissionService.js
  • apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/controller/ItemControllerAuthIntegrationTest.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/NamespaceController.java
  • apollo-portal/src/main/resources/static/i18n/en.json
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/entity/vo/ClusterNamespaceRolesAssignedUsers.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/NamespaceBranchController.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/util/RoleUtils.java
  • apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ReleaseController.java
  • apollo-portal/src/main/resources/static/i18n/zh-CN.json
🔇 Additional comments (11)
apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/ItemController.java (3)

76-76: LGTM! Consistent implementation of cluster-level permission checks.

The @PreAuthorize annotations have been systematically updated to include cluster-level authorization checks, maintaining consistency across all modification operations.

Also applies to: 90-90, 109-109, 123-123, 223-223, 234-234


145-145: LGTM! Secure implementation of visibility check.

The visibility check correctly includes cluster-level permissions and safely returns an empty list when access is denied.


186-189: LGTM! Improved error message clarity.

The permission check has been updated to include cluster-level authorization, and the error message has been improved as previously suggested.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/component/PermissionValidator.java (3)

51-60: Well-structured encapsulation of permission checking methods!

Good decision to make these permission checking methods private, as they are implementation details used by the public methods. This change follows the principle of information hiding and reduces the public API surface.


63-80: Well-implemented cluster permission checks!

The new cluster permission methods are well-structured and follow a consistent pattern:

  1. Clear method naming that reflects the purpose
  2. Proper delegation to rolePermissionService
  3. Logical combination of different permission levels (namespace, env, cluster)

Line range hint 160-174: Well-organized permission validation logic!

The updated shouldHideConfigToCurrentUser method follows good practices:

  1. Efficient check ordering (environment config first, then public namespace, finally permissions)
  2. Clear separation of concerns
  3. Proper integration of cluster-level permissions
apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/spi/defaultimpl/DefaultRoleInitializationService.java (1)

246-255: LGTM! Well-structured implementation.

The method follows the established pattern for role creation, properly separating permission and role creation while reusing existing helper methods.

apollo-portal/src/test/java/com/ctrip/framework/apollo/portal/spi/defaultImpl/RoleInitializationServiceTest.java (1)

169-227: LGTM! Comprehensive test coverage.

The test suite thoroughly covers all role existence combinations:

  • Both roles existing
  • Only modify role existing
  • Only release role existing

This ensures robust validation of the role initialization logic.

apollo-portal/src/main/java/com/ctrip/framework/apollo/portal/controller/PermissionController.java (3)

213-235: LGTM! Well-implemented role retrieval.

The implementation includes proper validation and follows the established pattern for role retrieval endpoints.


237-259: LGTM! Comprehensive implementation with proper validation.

The endpoint includes all necessary security controls and validations:

  • Access control via @PreAuthorize
  • User existence check
  • Role type validation
  • Environment validation
  • Duplicate assignment handling

261-277: LGTM! Consistent with role assignment implementation.

The endpoint properly mirrors the assignment endpoint's security controls and validations.

@BlackBear2003
Copy link
Member Author

BlackBear2003 commented Jan 12, 2025

Should we add usage instructions for this feature to the apollo-user-guide?

https://www.apolloconfig.com/#/zh/portal/apollo-user-guide?id=_122-%e9%85%8d%e7%bd%ae%e7%bc%96%e8%be%91%e3%80%81%e5%8f%91%e5%b8%83%e6%9d%83%e9%99%90

I suggest adding usage instructions below this label to describe 3 models we have now and its usage instructions.

I Write

image

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 5

🧹 Nitpick comments (5)
apollo-portal/src/main/resources/static/scripts/controller/ManageClusterController.js (1)

17-47: Enhance controller to fully support cluster-based authorization

The current implementation focuses only on loading clusters but lacks essential features needed for the cluster-based authorization system described in the PR objectives:

  1. Methods for managing cluster permissions
  2. Integration with the NamespaceInCluster permission model
  3. UI handlers for modifying and publishing configurations based on cluster permissions

Consider extending the controller with these features:

  1. Add methods to check and manage cluster-specific permissions
  2. Implement handlers for cluster configuration operations
  3. Add support for the new NamespaceInCluster permission model
  4. Include proper error handling for unauthorized cluster operations

Would you like me to provide a detailed implementation plan for these enhancements?

apollo-portal/src/main/resources/static/app/manage_cluster.html (4)

32-34: Enhance accessibility with ARIA landmarks

Add ARIA landmarks to improve accessibility for screen readers:

-<body>
+<body role="application">
 
-<apollonav></apollonav>
+<apollonav role="navigation"></apollonav>

36-51: Improve semantic structure and accessibility

The header section could benefit from semantic HTML elements and better accessibility:

-<section class="col-md-10 col-md-offset-1 panel">
+<main class="col-md-10 col-md-offset-1 panel" role="main">
   <header class="panel-heading">
     <div class="row">
       <div class="col-md-9">
-        <h4 class="modal-title">
+        <h1 class="modal-title h4">
           {{'Config.ManageCluster' | translate }} (
-          {{'Common.AppId' | translate }}: <label ng-bind="appId"></label> )
+          {{'Common.AppId' | translate }}: <span ng-bind="appId"></span> )
-        </h4>
+        </h1>
       </div>

74-74: Use absolute path for image resources

The image src uses a relative path which could cause issues if the base URL changes:

-    <img src="/img/assign.png">
+    <img src="{{ '/img/assign.png' | prefixPath }}" alt="{{ 'Cluster.Grant' | translate }}">

91-128: Optimize script loading for better performance

Consider these performance improvements:

  1. Add defer attribute to non-critical scripts
  2. Consider bundling vendor scripts
  3. Move scripts to the end of body (already done correctly)
-<script src="../vendor/jquery.min.js" type="text/javascript"></script>
+<script src="../vendor/jquery.min.js" type="text/javascript" defer></script>

-<script src="../vendor/angular/angular.min.js"></script>
+<script src="../vendor/angular/angular.min.js" defer></script>

Also, consider adding version hashes to script URLs for cache busting:

<script src="../vendor/jquery.min.js?v={{version}}" type="text/javascript" defer></script>
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between 88ce290 and 4c90c18.

📒 Files selected for processing (7)
  • apollo-portal/src/main/resources/static/app/manage_cluster.html (1 hunks)
  • apollo-portal/src/main/resources/static/cluster/ns_role.html (1 hunks)
  • apollo-portal/src/main/resources/static/config.html (1 hunks)
  • apollo-portal/src/main/resources/static/i18n/en.json (3 hunks)
  • apollo-portal/src/main/resources/static/i18n/zh-CN.json (3 hunks)
  • apollo-portal/src/main/resources/static/scripts/app.js (1 hunks)
  • apollo-portal/src/main/resources/static/scripts/controller/ManageClusterController.js (1 hunks)
🚧 Files skipped from review as they are similar to previous changes (5)
  • apollo-portal/src/main/resources/static/scripts/app.js
  • apollo-portal/src/main/resources/static/config.html
  • apollo-portal/src/main/resources/static/cluster/ns_role.html
  • apollo-portal/src/main/resources/static/i18n/en.json
  • apollo-portal/src/main/resources/static/i18n/zh-CN.json
⏰ Context from checks skipped due to timeout of 90000ms (1)
  • GitHub Check: build (8)
🔇 Additional comments (3)
apollo-portal/src/main/resources/static/scripts/controller/ManageClusterController.js (1)

17-21: ⚠️ Potential issue

Fix dependency injection array to match function parameters

The dependency injection array doesn't match the function parameters:

  1. AppUtil is missing in the dependency array
  2. The order of dependencies should match exactly

Apply this diff to fix the dependency injection:

 manage_cluster_module.controller('ManageClusterController',
     ['$scope', '$location', '$window', '$translate', 'toastr', 'AppService', 'EnvService', 'ClusterService',
-      'AppUtil',
-      function ($scope, $location, $window, $translate, toastr, AppService, EnvService, ClusterService,
-          AppUtil) {
+      function ($scope, $location, $window, $translate, toastr, AppService, EnvService, ClusterService) {

Likely invalid or redundant comment.

apollo-portal/src/main/resources/static/app/manage_cluster.html (2)

1-2: Verify the copyright year

The copyright notice shows 2024. Since this is a new file, please verify if this is intentional or if it should match the project's standard copyright year format.

✅ Verification successful

Copyright year is consistent with project standards

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Check the copyright year format in other files
rg -l "Copyright \d{4}" | head -n 5 | xargs cat | rg "Copyright \d{4}"

Length of output: 237


127-127: Verify controller integration

Verify that ManageClusterController.js implements all required functionality:

✅ Verification successful

Script reference is correct and properly integrated

The ManageClusterController.js file exists at the correct location and is properly integrated with the Angular module system. The relative path in the HTML file is accurate.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Check if the controller file exists and implements required methods
rg -A 10 "ManageClusterController\s*=\s*function" ../scripts/controller/ManageClusterController.js

# Check for environment and cluster service usage
rg -A 5 "EnvService|ClusterService" ../scripts/controller/ManageClusterController.js

Length of output: 365


Script:

#!/bin/bash
# Find ManageClusterController.js anywhere in the repository
fd -t f "ManageClusterController.js"

# Search for any references to ManageClusterController
rg "ManageClusterController" -t html -t js

Length of output: 645

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

🧹 Nitpick comments (2)
docs/zh/portal/apollo-user-guide.md (2)

81-89: Consider adding implementation timeline information.

The permission model table clearly shows which models are implemented. Consider adding:

  1. Expected timeline for unimplemented models
  2. Links to relevant issues/roadmap for tracking implementation progress

90-100: Comprehensive permission model explanations!

The table effectively explains each permission model with its technical details. Consider adding:

  1. Example use cases for each model
  2. Common scenarios where each model would be most appropriate
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between 4c90c18 and e0f97fe.

⛔ Files ignored due to path filters (6)
  • doc/images/manage_cluster_entry.png is excluded by !**/*.png
  • doc/images/ns-permission-app-allns-entry.png is excluded by !**/*.png
  • doc/images/ns-permission-app-allns-select.png is excluded by !**/*.png
  • doc/images/ns-permission-app-env-cluster-edit.png is excluded by !**/*.png
  • doc/images/ns-permission-app-env-cluster-entry.png is excluded by !**/*.png
  • doc/images/ns-permission-app-env-ns-select.png is excluded by !**/*.png
📒 Files selected for processing (1)
  • docs/zh/portal/apollo-user-guide.md (1 hunks)
⏰ Context from checks skipped due to timeout of 90000ms (1)
  • GitHub Check: build (8)
🔇 Additional comments (2)
docs/zh/portal/apollo-user-guide.md (2)

73-80: Well-structured introduction to permission management dimensions!

The explanation of permission hierarchy from Namespace to App is clear and helps users understand the foundation of Apollo's permission system.


101-139: Excellent step-by-step guides with visual aids!

The guides are comprehensive and well-documented with:

  1. Clear step-by-step instructions
  2. Relevant screenshots
  3. Proper coverage of all three permission models

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

🧹 Nitpick comments (3)
docs/en/portal/apollo-user-guide.md (3)

86-94: Consider adding implementation status column descriptions.

The table effectively presents different permission models, but the "Impl" column's values (yes/no) could benefit from a brief explanation of what they indicate.

Add a note below the table explaining what "yes" and "no" mean in the implementation status column. For example:

| App | Env | Cluster | Namespace | Model | Impl |
| --- | --- | --- | --- | --- |------|
...existing rows...
+
+> Note: In the "Impl" column:
+> - "yes": Currently implemented and available for use
+> - "no": Planned for future implementation

97-105: Enhance model explanation table with examples.

The table provides clear explanations of different permission models, but adding practical examples would make it more user-friendly.

Consider adding an "Example" column with concrete use cases:

| Model | Target | PermissionType (e.g. Modify) | TargetId |
| --- | --- | --- | --- |
+| Model | Target | PermissionType (e.g. Modify) | TargetId | Example |
...existing rows...
+| App → Namespace | All namespaces with specified names under App | ModifyNamespace | App+Namespace | Granting modify permissions for the 'logging' namespace across all environments |

117-117: Fix grammatical issues in section headings.

Minor grammatical inconsistencies in privilege assignment headings.

Apply these corrections:

-4. Assign publish privileges
+4. Assign publishing privileges

Also applies to: 131-131

🧰 Tools
🪛 LanguageTool

[grammar] ~117-~117: There seems to be a noun/verb agreement error. Did you mean “publishes” or “published”?
Context: ...mespace-permission-edit.png) 4. Assign publish privileges * ![namespace-publish-per...

(SINGULAR_NOUN_VERB_AGREEMENT)

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between e0f97fe and 14691d8.

📒 Files selected for processing (1)
  • docs/en/portal/apollo-user-guide.md (1 hunks)
🧰 Additional context used
🪛 LanguageTool
docs/en/portal/apollo-user-guide.md

[grammar] ~117-~117: There seems to be a noun/verb agreement error. Did you mean “publishes” or “published”?
Context: ...mespace-permission-edit.png) 4. Assign publish privileges * ![namespace-publish-per...

(SINGULAR_NOUN_VERB_AGREEMENT)


[grammar] ~131-~131: There seems to be a noun/verb agreement error. Did you mean “publishes” or “published”?
Context: ...mespace-permission-edit.png) 4. Assign publish privileges * ![namespace-publish-per...

(SINGULAR_NOUN_VERB_AGREEMENT)

⏰ Context from checks skipped due to timeout of 90000ms (1)
  • GitHub Check: build (8)
🔇 Additional comments (2)
docs/en/portal/apollo-user-guide.md (2)

79-85: Well-structured introduction to permission dimensions!

The introduction clearly explains Apollo's flexible permission management system and its hierarchical structure from App (largest unit) to Namespace (smallest unit).


106-144: LGTM! Clear step-by-step instructions with screenshots.

The configuration steps for different permission models are well-documented with clear instructions and supporting screenshots, making it easy for users to follow.

🧰 Tools
🪛 LanguageTool

[grammar] ~117-~117: There seems to be a noun/verb agreement error. Did you mean “publishes” or “published”?
Context: ...mespace-permission-edit.png) 4. Assign publish privileges * ![namespace-publish-per...

(SINGULAR_NOUN_VERB_AGREEMENT)


[grammar] ~131-~131: There seems to be a noun/verb agreement error. Did you mean “publishes” or “published”?
Context: ...mespace-permission-edit.png) 4. Assign publish privileges * ![namespace-publish-per...

(SINGULAR_NOUN_VERB_AGREEMENT)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
size:XXL This PR changes 1000+ lines, ignoring generated files.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants