Merge pull request #1698 from wknet123/dev-revised

Update latest changes of UI codes for dev branch.
This commit is contained in:
Daniel Jiang 2017-03-21 15:21:33 +08:00 committed by GitHub
commit 1320bc8685
18 changed files with 1101 additions and 134 deletions

11
.gitignore vendored
View File

@ -12,13 +12,14 @@ jobservice/test
src/ui/static/dist/ src/ui/static/dist/
src/clarity-seed/coverage/ src/ui_ng/coverage/
src/clarity-seed/dist/ src/ui_ng/dist/
src/clarity-seed/html-report/ src/ui_ng/html-report/
src/clarity-seed/node_modules/ src/ui_ng/node_modules/
src/clarity-seed/typings/ src/ui_ng/typings/
**/*npm-debug.log.* **/*npm-debug.log.*
**/*yarn-error.log.* **/*yarn-error.log.*
.idea/ .idea/
.DS_Store .DS_Store

View File

@ -0,0 +1,386 @@
{
"SIGN_IN": {
"REMEMBER": "Remember me",
"INVALID_MSG": "Invalid user name or password",
"FORGOT_PWD": "Forgot password",
"HEADER_LINK": "Sign In"
},
"SIGN_UP": {
"TITLE": "Sign Up"
},
"BUTTON": {
"CANCEL": "CANCEL",
"OK": "OK",
"DELETE": "DELETE",
"LOG_IN": "LOG IN",
"SIGN_UP_LINK": "Sign up for an account",
"SIGN_UP": "SIGN UP",
"CONFIRM": "CONFIRM",
"SEND": "SEND",
"SAVE": "SAVE",
"TEST_MAIL": "TEST MAIL SERVER",
"CLOSE": "CLOSE",
"TEST_LDAP": "TEST LDAP SERVER",
"MORE_INFO": "More info...",
"YES": "YES"
},
"TOOLTIP": {
"EMAIL": "Email should be a valid email address like name@example.com",
"USER_NAME": "Can not contain \"~#$% and maxinum length should be less than 20 characters",
"FULL_NAME": "Max length should be less than 20 characters",
"COMMENT": "Length of comment should be less than 20 characters",
"CURRENT_PWD": "Current password is required",
"PASSWORD": "Password should be at least 8 characters with 1 uppercase, 1 lowercase and 1 number",
"CONFIRM_PWD": "Password should be same as the password above",
"SIGN_IN_USERNAME": "Username is required",
"SIGN_IN_PWD": "Password is required",
"SIGN_UP_MAIL": "Email is only used for resetting your password",
"SIGN_UP_REAL_NAME": "First name and last name",
"ITEM_REQUIRED": "Field is required",
"NUMBER_REQUIRED": "Field is required and should be numbers",
"PORT_REQUIRED": "Field is required and should be valid port number",
"EMAIL_EXISTING": "Email address is already existing",
"USER_EXISTING": "Username is already used"
},
"PLACEHOLDER": {
"CURRENT_PWD": "Enter current password",
"NEW_PWD": "Enter new password",
"CONFIRM_PWD": "Confirm new password",
"USER_NAME": "Enter username",
"MAIL": "Enter email address",
"FULL_NAME": "Enter full name",
"SIGN_IN_NAME": "Username",
"SIGN_IN_PWD": "Password"
},
"PROFILE": {
"TITLE": "User Profile",
"USER_NAME": "Username",
"EMAIL": "Email",
"FULL_NAME": "Full name",
"COMMENT": "Comments",
"PASSWORD": "Password",
"SAVE_SUCCESS": "User profile saved successfully"
},
"CHANGE_PWD": {
"TITLE": "Change Password",
"CURRENT_PWD": "Current Password",
"NEW_PWD": "New Password",
"CONFIRM_PWD": "Confirm Password",
"SAVE_SUCCESS": "User password changed successfully"
},
"ACCOUNT_SETTINGS": {
"PROFILE": "User Profile",
"CHANGE_PWD": "Change Password",
"ABOUT": "About",
"LOGOUT": "Log Out"
},
"GLOBAL_SEARCH": {
"PLACEHOLDER": "Search Harbor..."
},
"SIDE_NAV": {
"DASHBOARD": "Dashboard",
"PROJECTS": "Projects",
"SYSTEM_MGMT": {
"NAME": "Administration",
"USER": "Users",
"REPLICATION": "Replication",
"CONFIG": "Configuration"
},
"LOGS": "Logs"
},
"USER": {
"ADD_ACTION": "USER",
"ENABLE_ADMIN_ACTION": "Enable administrator",
"DISABLE_ADMIN_ACTION": "Disable administrator",
"DEL_ACTION": "Delete",
"FILTER_PLACEHOLDER": "Filter users",
"COLUMN_NAME": "Name",
"COLUMN_ADMIN": "Administrator",
"COLUMN_EMAIL": "Email",
"COLUMN_REG_NAME": "Registration time",
"IS_ADMIN": "Yes",
"IS_NOT_ADMIN": "No",
"ADD_USER_TITLE": "Add User",
"SAVE_SUCCESS": "New user added successfully",
"DELETION_TITLE": "Confirm user deletion",
"DELETION_SUMMARY": "Do you want to delete user {{param}}?",
"DELETE_SUCCESS": "User deleted successfully"
},
"PROJECT": {
"PROJECTS": "Projects",
"NAME": "Project Name",
"PUBLIC_OR_PRIVATE": "Public",
"REPO_COUNT": "Repositories Count",
"CREATION_TIME": "Creation Time",
"DESCRIPTION": "Description",
"PUBLIC": "Public",
"PRIVATE": "Private",
"MAKE": "Make",
"NEW_POLICY": "New Policy",
"DELETE": "Delete",
"MY_PROJECTS": "My Projects",
"PUBLIC_PROJECTS": "Public Projects",
"PROJECT": "Project",
"NEW_PROJECT": "New Project",
"NAME_IS_REQUIRED": "Project name is required.",
"NAME_MINIMUM_LENGTH": "Project name is too short, it should be greater than 2 characters.",
"NAME_ALREADY_EXISTS": "Project name already exists.",
"NAME_IS_ILLEGAL": "Project name is invalid.",
"UNKNOWN_ERROR": "An unknown error occurred while creating the project.",
"ITEMS": "item(s)",
"DELETION_TITLE": "Confirm project deletion",
"DELETION_SUMMARY": "Do you want to delete project {{param}}?",
"FILTER_PLACEHOLDER": "Filter Projects",
"REPLICATION_RULE": "Replication Rule"
},
"PROJECT_DETAIL": {
"REPOSITORIES": "Repositories",
"REPLICATION": "Replication",
"USERS": "Users",
"LOGS": "Logs",
"PROJECTS": "Projects"
},
"MEMBER": {
"NEW_MEMBER": "New Member",
"MEMBER": "Member",
"NAME": "Name",
"ROLE": "Role",
"SYS_ADMIN": "System Admin",
"PROJECT_ADMIN": "Project Admin",
"DEVELOPER": "Developer",
"GUEST": "Guest",
"DELETE": "Delete",
"ITEMS": "item(s)",
"ACTIONS": "Actions",
"USERNAME_DOES_NOT_EXISTS": "Username does not exist.",
"USERNAME_ALREADY_EXISTS": "Username already exists.",
"UNKNOWN_ERROR": "Unknown error occurred while adding member.",
"FILTER_PLACEHOLDER": "Filter Members",
"DELETION_TITLE": "Confirm project member deletion",
"DELETION_SUMMARY": "Do you want to delete project member {{param}}?"
},
"AUDIT_LOG": {
"USERNAME": "Username",
"REPOSITORY_NAME": "Repository Name",
"TAGS": "Tags",
"OPERATION": "Operation",
"OPERATIONS": "Operations",
"TIMESTAMP": "Timestamp",
"ALL_OPERATIONS": "All Operations",
"PULL": "Pull",
"PUSH": "Push",
"CREATE": "Create",
"DELETE": "Delete",
"OTHERS": "Others",
"ADVANCED": "Advanced",
"SIMPLE": "Simple",
"ITEMS": "item(s)",
"FILTER_PLACEHOLDER": "Filter Logs"
},
"REPLICATION": {
"REPLICATION_RULE": "Replication Rule",
"NEW_REPLICATION_RULE": "New Replication Rule",
"ENDPOINTS": "Endpoints",
"FILTER_POLICIES_PLACEHOLDER": "Filter Policies",
"FILTER_JOBS_PLACEHOLDER": "Filter Jobs",
"DELETION_TITLE": "Confirm Policy Deletion",
"DELETION_SUMMARY": "Do you want to delete policy {{param}}?",
"FILTER_TARGETS_PLACEHOLDER": "Filter Targets",
"DELETION_TITLE_TARGET": "Confirm Target Deletion",
"DELETION_SUMMARY_TARGET": "Do you want to delete target {{param}}?",
"ADD_POLICY": "New Replication Rule",
"EDIT_POLICY": "Edit Replication Rule",
"DELETE_POLICY": "Delete Policy",
"TEST_CONNECTION": "Test Connection",
"TESTING_CONNECTION": "Testing Connection...",
"TEST_CONNECTION_SUCCESS": "Connection tested successfully.",
"TEST_CONNECTION_FAILURE": "Failed to ping target.",
"NAME": "Name",
"PROJECT": "Project",
"NAME_IS_REQUIRED": "Name is required.",
"DESCRIPTION": "Description",
"ENABLE": "Enable",
"DISABLE": "Disable",
"DESTINATION_NAME": "Destination Name",
"DESTINATION_NAME_IS_REQUIRED": "Destination name is required.",
"NEW_DESTINATION": "New Destination",
"DESTINATION_URL": "Endpoint URL",
"DESTINATION_URL_IS_REQUIRED": "Endpoint URL is required.",
"DESTINATION_USERNAME": "Username",
"DESTINATION_PASSWORD": "Password",
"ALL_STATUS": "All Status",
"ENABLED": "Enabled",
"DISABLED": "Disabled",
"LAST_START_TIME": "Last Start Time",
"ACTIVATION": "Activation",
"REPLICATION_JOBS": "Replication Jobs",
"ALL": "All",
"PENDING": "Pending",
"RUNNING": "Running",
"ERROR": "Error",
"RETRYING": "Retrying",
"STOPPED": "Stopped",
"FINISHED": "Finished",
"CANCELED": "Canceled",
"SIMPLE": "Simple",
"ADVANCED": "Advanced",
"STATUS": "Status",
"OPERATION": "Operation",
"CREATION_TIME": "Start Time",
"END_TIME": "End Time",
"LOGS": "Logs",
"ITEMS": "item(s)"
},
"DESTINATION": {
"NEW_ENDPOINT": "New Endpoint",
"ENDPOINT": "Endpoint",
"NAME": "Destination Name",
"NAME_IS_REQUIRED": "Destination name is required.",
"URL": "Endpoint URL",
"URL_IS_REQUIRED": "Endpoint URL is required.",
"USERNAME": "Username",
"PASSWORD": "Password",
"TEST_CONNECTION": "Test Connection",
"TITLE_EDIT": "Edit Endpoint",
"TITLE_ADD": "Create Endpoint",
"DELETE": "Delete Endpoint",
"TESTING_CONNECTION": "Testing Connection...",
"TEST_CONNECTION_SUCCESS": "Connection tested successfully.",
"TEST_CONNECTION_FAILURE": "Failed to ping target.",
"CONFLICT_NAME": "Name or endpoint URL already exists.",
"INVALID_NAME": "Invalid destination name.",
"FAILED_TO_GET_TARGET": "Failed to get endpoint.",
"CREATION_TIME": "Creation Time",
"ITEMS": "item(s)"
},
"REPOSITORY": {
"COPY_ID": "Copy ID",
"COPY_PARENT_ID": "Copy Parent ID",
"DELETE": "Delete",
"NAME": "Name",
"TAGS_COUNT": "Tags",
"PULL_COUNT": "Pulls",
"PULL_COMMAND": "Pull Command",
"MY_REPOSITORY": "My Repository",
"PUBLIC_REPOSITORY": "Public Repository",
"DELETION_TITLE_REPO": "Confirm Repository Deletion",
"DELETION_SUMMARY_REPO": "Do you want to delete repository {{param}}?",
"DELETION_TITLE_TAG": "Confirm Tag Deletion",
"DELETION_SUMMARY_TAG": "Do you want to delete tag {{param}}?",
"DELETION_TITLE_TAG_DENIED": "Signed Tag can't be deleted",
"DELETION_SUMMARY_TAG_DENIED": "The tag must be removed from the Notary before it can be deleted.",
"FILTER_FOR_REPOSITORIES": "Filter for repositories",
"TAG": "Tag",
"SIGNED": "Signed",
"AUTHOR": "Author",
"CREATED": "Creation Time",
"DOCKER_VERSION": "Docker Version",
"ARCHITECTURE": "Architecture",
"OS": "OS",
"SHOW_DETAILS": "Show Details",
"REPOSITORIES": "Repositories",
"ITEMS": "item(s)",
"POP_REPOS": "Popular Repositories"
},
"ALERT": {
"FORM_CHANGE_CONFIRMATION": "Some changes are not saved yet, do you really want to cancel?"
},
"RESET_PWD": {
"TITLE": "Reset Password",
"CAPTION": "Enter your email to reset your password",
"EMAIL": "Email",
"SUCCESS": "Mail of resetting password is successfully send to your mail box",
"CAPTION2": "Enter your new password",
"RESET_OK": "Password has been successfully reset. Click OK to login with new password"
},
"RECENT_LOG": {
"SUB_TITLE": "Show recent",
"SUB_TITLE_SUFIX": "logs"
},
"CONFIG": {
"TITLE": "Configuration",
"AUTH": "Authentication",
"REPLICATION": "Replication",
"EMAIL": "Email",
"SYSTEM": "System Settings",
"CONFIRM_TITLE": "Confirm to cancel",
"CONFIRM_SUMMARY": "Some changes are not saved yet, do you really want to leave?",
"SAVE_SUCCESS": "Configurations have been successfully saved",
"MAIL_SERVER": "Email Server",
"MAIL_SERVER_PORT": "Email Server Port",
"MAIL_USERNAME": "Email Username",
"MAIL_PASSWORD": "Email Password",
"MAIL_FROM": "Email From",
"MAIL_SSL": "Email SSL",
"SSL_TOOLTIP": "Enable SSL for email server connection",
"VERIFY_REMOTE_CERT": "Verify Remote Certificate",
"TOKEN_EXPIRATION": "Token Expiration (Minutes)",
"AUTH_MODE": "Authentication",
"PRO_CREATION_RESTRICTION": "Project Creation Restriction",
"SELF_REGISTRATION": "Self Registration",
"AUTH_MODE_DB": "Database",
"AUTH_MODE_LDAP": "LDAP",
"SCOPE_BASE": "Base",
"SCOPE_ONE_LEVEL": "OneLevel",
"SCOPE_SUBTREE": "Subtree",
"PRO_CREATION_EVERYONE": "Everyone",
"PRO_CREATION_ADMIN": "Admin Only",
"TOOLTIP": {
"SELF_REGISTRATION": "Enable sign up",
"VERIFY_REMOTE_CERT": "Determine whether the image replication should verify the certificate of a remote Habor registry. Uncheck this box when the remote registry uses a self-signed or untrusted certificate.",
"AUTH_MODE": "By default the auth mode is db_auth, i.e. the credentials are stored in a local database. Set it to ldap_auth if you want to verify a user's credentials against an LDAP server.",
"LDAP_SEARCH_DN": "A user's DN who has the permission to search the LDAP/AD server. If your LDAP/AD server does not support anonymous search, you should configure this DN and ldap_search_pwd.",
"LDAP_BASE_DN": "The base DN from which to look up a user in LDAP/AD",
"LDAP_UID": "The attribute used in a search to match a user. It could be uid, cn, email, sAMAccountName or other attributes depending on your LDAP/AD",
"LDAP_SCOPE": "The scope to search for users",
"TOKEN_EXPIRATION": "The expiration time (in minutes) of a token created by the token service. Default is 30 minutes",
"PRO_CREATION_RESTRICTION": "The flag to control what users have permission to create projects. By default everyone can create a project. Set to 'adminonly' so only an admin can create a project."
},
"LDAP": {
"URL": "LDAP URL",
"SEARCH_DN": "LDAP Search DN",
"SEARCH_PWD": "LDAP Search Password",
"BASE_DN": "LDAP Base DN",
"FILTER": "LDAP Filter",
"UID": "LDAP UID",
"SCOPE": "LDAP Scope"
},
"TEST_MAIL_SUCCESS": "Connection to mail server is verified",
"TEST_LDAP_SUCCESS": "Connection to LDAP server is verified",
"LEAVING_CONFIRMATION_TITLE": "Confirm to leave",
"LEAVING_CONFIRMATION_SUMMARY": "Changes have not been saved yet, do you really want to leave currnet page?"
},
"PAGE_NOT_FOUND": {
"MAIN_TITLE": "Page not found",
"SUB_TITLE": "Redirect to Harbor page in",
"UNIT": "seconds..."
},
"ABOUT": {
"VERSION": "Version",
"BUILD": "Build",
"COPYRIGHT": "Copyright 1998-2016 VMware. Inc. All rights reserved. This product is protected by U.S. and international property laws. VMware products are covered by one or more patents listed at",
"TRADEMARK": "VMware is a registered trademark or trademark of VMware. Inc. in the United States and other jurisdictions. All other marks and names mentioned herein may be trademark of their respective companies.",
"END_USER_LICENSE": "End User License Agreement",
"OPEN_SOURCE_LICENSE": "Open Source/Third Party License"
},
"START_PAGE": {
"GETTING_START": "Project Harbor is an enterprise-class registry server that stores and distributes Docker images. Harbor extends the open source Docker Distribution by adding the functionalities usually required by an enterprise, such as security, identity and management. As an enterprise private registry, Harbor offers better performance and security.",
"GETTING_START_TITLE": "Getting Started"
},
"TOP_REPO": "Popular Repositories",
"STATISTICS": {
"TITLE": "STATISTICS",
"PRO_ITEM": "PROJECTS",
"REPO_ITEM": "REPOSITORIES",
"INDEX_MY": "MY",
"INDEX_PUB": "PUBLIC",
"INDEX_TOTAL": "TOTAL"
},
"SEARCH": {
"IN_PROGRESS": "Search...",
"BACK": "Back"
},
"UNKNOWN_ERROR": "Some unknown errors HAVE occurred. Please try again later",
"UNAUTHORIZED_ERROR": "Your session is invalid or has expired. You need to sign in to continue the operation",
"FORBIDDEN_ERROR": "You are not allowed to perform this operation"
}

View File

@ -0,0 +1,386 @@
{
"SIGN_IN": {
"REMEMBER": "记住我",
"INVALID_MSG": "用户名或者密码不正确",
"FORGOT_PWD": "忘记密码",
"HEADER_LINK": "登录"
},
"SIGN_UP": {
"TITLE": "注册"
},
"BUTTON": {
"CANCEL": "取消",
"OK": "确定",
"DELETE": "删除",
"LOG_IN": "登录",
"SIGN_UP_LINK": "注册账号",
"SIGN_UP": "注册",
"CONFIRM": "确定",
"SEND": "发送",
"SAVE": "保存",
"TEST_MAIL": "测试邮件服务器",
"CLOSE": "关闭",
"TEST_LDAP": "测试LDAP服务器",
"MORE_INFO": "更多信息...",
"YES": "确定"
},
"TOOLTIP": {
"EMAIL": "请使用正确的邮箱地址比如name@example.com",
"USER_NAME": "不能包含\"~#$%特殊字符且长度不能超过20",
"FULL_NAME": "长度不能超过20",
"COMMENT": "长度不能超过20",
"CURRENT_PWD": "当前密码必需",
"PASSWORD": "密码长度至少为8且需包含至少一个大写字符一个小写字符和一个数字",
"CONFIRM_PWD": "当前密码须与上述输入密码一致",
"SIGN_IN_USERNAME": "用户名必需",
"SIGN_IN_PWD": "密码必需",
"SIGN_UP_MAIL": "邮件地址仅用来重置您的密码",
"SIGN_UP_REAL_NAME": "全名",
"ITEM_REQUIRED": "此项必需",
"NUMBER_REQUIRED": "此项必需且为数字",
"PORT_REQUIRED": "此项必需且为合理端口号",
"EMAIL_EXISTING": "邮件地址已经存在",
"USER_EXISTING": "用户名已经存在"
},
"PLACEHOLDER": {
"CURRENT_PWD": "输入当前密码",
"NEW_PWD": "输入新密码",
"CONFIRM_PWD": "确认新密码",
"USER_NAME": "输入用户名称",
"MAIL": "输入邮箱地址",
"FULL_NAME": "输入全名",
"SIGN_IN_NAME": "用户名",
"SIGN_IN_PWD": "密码"
},
"PROFILE": {
"TITLE": "用户设置",
"USER_NAME": "用户名",
"EMAIL": "邮箱",
"FULL_NAME": "全名",
"COMMENT": "注释",
"PASSWORD": "密码",
"SAVE_SUCCESS": "成功保存用户配置"
},
"CHANGE_PWD": {
"TITLE": "修改密码",
"CURRENT_PWD": "当前密码",
"NEW_PWD": "新密码",
"CONFIRM_PWD": "确认密码",
"SAVE_SUCCESS": "更改用户密码成功"
},
"ACCOUNT_SETTINGS": {
"PROFILE": "用户设置",
"CHANGE_PWD": "修改密码",
"ABOUT": "关于",
"LOGOUT": "退出"
},
"GLOBAL_SEARCH": {
"PLACEHOLDER": "搜索 Harbor..."
},
"SIDE_NAV": {
"DASHBOARD": "仪表板",
"PROJECTS": "项目",
"SYSTEM_MGMT": {
"NAME": "系统管理",
"USER": "用户管理",
"REPLICATION": "复制管理",
"CONFIG": "配置管理"
},
"LOGS": "日志"
},
"USER": {
"ADD_ACTION": "用户",
"ENABLE_ADMIN_ACTION": "设置为管理员",
"DISABLE_ADMIN_ACTION": "取消管理员",
"DEL_ACTION": "删除",
"FILTER_PLACEHOLDER": "过滤用户",
"COLUMN_NAME": "用户名",
"COLUMN_ADMIN": "管理员",
"COLUMN_EMAIL": "邮件",
"COLUMN_REG_NAME": "注册时间",
"IS_ADMIN": "是",
"IS_NOT_ADMIN": "否",
"ADD_USER_TITLE": "添加用户",
"SAVE_SUCCESS": "添加用户成功",
"DELETION_TITLE": "删除用户确认",
"DELETION_SUMMARY": "你确认删除用户 {{param}}?",
"DELETE_SUCCESS": "删除用户成功"
},
"PROJECT": {
"PROJECTS": "项目",
"NAME": "项目名称",
"PUBLIC_OR_PRIVATE": "公开",
"REPO_COUNT": "镜像仓库数",
"CREATION_TIME": "创建时间",
"DESCRIPTION": "描述",
"PUBLIC": "公开",
"PRIVATE": "私有",
"MAKE": "设为",
"NEW_POLICY": "新建策略",
"DELETE": "删除",
"MY_PROJECTS": "我的项目",
"PUBLIC_PROJECTS": "公开项目",
"PROJECT": "项目",
"NEW_PROJECT": "新建项目",
"NAME_IS_REQUIRED": "项目名称为必填项",
"NAME_MINIMUM_LENGTH": "项目名称长度过短至少多于2个字符。",
"NAME_ALREADY_EXISTS": "项目名称已存在。",
"NAME_IS_ILLEGAL": "项目名称非法。",
"UNKNOWN_ERROR": "创建项目时发生未知错误。",
"ITEMS": "条记录",
"DELETION_TITLE": "删除项目确认",
"DELETION_SUMMARY": "你确认删除项目 {{param}}",
"FILTER_PLACEHOLDER": "过滤项目",
"REPLICATION_RULE": "复制策略"
},
"PROJECT_DETAIL": {
"REPOSITORIES": "镜像仓库",
"REPLICATION": "复制",
"USERS": "用户",
"LOGS": "日志",
"PROJECTS": "项目"
},
"MEMBER": {
"NEW_MEMBER": "新建成员",
"MEMBER": "成员",
"NAME": "姓名",
"ROLE": "角色",
"SYS_ADMIN": "系统管理员",
"PROJECT_ADMIN": "项目管理员",
"DEVELOPER": "开发人员",
"GUEST": "访客",
"DELETE": "删除",
"ITEMS": "条记录",
"ACTIONS": "操作",
"USERNAME_DOES_NOT_EXISTS": "用户名不存在",
"USERNAME_ALREADY_EXISTS": "用户名已存在",
"UNKNOWN_ERROR": "添加成员时发生未知错误。",
"FILTER_PLACEHOLDER": "过滤成员",
"DELETION_TITLE": "删除项目成员确认",
"DELETION_SUMMARY": "你确认删除项目成员 {{param}}?"
},
"AUDIT_LOG": {
"USERNAME": "用户名",
"REPOSITORY_NAME": "镜像名称",
"TAGS": "标签",
"OPERATION": "操作",
"OPERATIONS": "操作",
"TIMESTAMP": "时间戳",
"ALL_OPERATIONS": "所有操作",
"PULL": "Pull",
"PUSH": "Push",
"CREATE": "Create",
"DELETE": "Delete",
"OTHERS": "其他",
"ADVANCED": "高级检索",
"SIMPLE": "简单检索",
"ITEMS": "条记录",
"FILTER_PLACEHOLDER": "过滤日志"
},
"REPLICATION": {
"REPLICATION_RULE": "复制策略",
"NEW_REPLICATION_RULE": "新建策略",
"ENDPOINTS": "目标",
"FILTER_POLICIES_PLACEHOLDER": "过滤策略",
"FILTER_JOBS_PLACEHOLDER": "过滤任务",
"DELETION_TITLE": "删除策略确认",
"DELETION_SUMMARY": "确认删除策略 {{param}}?",
"FILTER_TARGETS_PLACEHOLDER": "过滤目标",
"DELETION_TITLE_TARGET": "删除目标确认",
"DELETION_SUMMARY_TARGET": "确认删除目标 {{param}}?",
"ADD_POLICY": "新建策略",
"EDIT_POLICY": "修改策略",
"DELETE_POLICY": "删除策略",
"TEST_CONNECTION": "测试连接",
"TESTING_CONNECTION": "正在测试连接...",
"TEST_CONNECTION_SUCCESS": "测试连接成功。",
"TEST_CONNECTION_FAILURE": "测试连接失败。",
"NAME": "名称",
"PROJECT": "项目",
"NAME_IS_REQUIRED": "名称为必填项",
"DESCRIPTION": "描述",
"ENABLE": "启用",
"DISABLE": "停用",
"DESTINATION_NAME": "目标名",
"DESTINATION_NAME_IS_REQUIRED": "目标名称为必填项。",
"NEW_DESTINATION": "创建目标",
"DESTINATION_URL": "目标URL",
"DESTINATION_URL_IS_REQUIRED": "目标URL为必填项。",
"DESTINATION_USERNAME": "用户名",
"DESTINATION_PASSWORD": "密码",
"ALL_STATUS": "所有状态",
"ENABLED": "启用",
"DISABLED": "停用",
"LAST_START_TIME": "上次起始时间",
"ACTIVATION": "活动状态",
"REPLICATION_JOBS": "复制任务",
"ALL": "全部",
"PENDING": "挂起",
"RUNNING": "运行中",
"ERROR": "错误",
"RETRYING": "重试中",
"STOPPED": "已停止",
"FINISHED": "已完成",
"CANCELED": "已取消",
"SIMPLE": "简单检索",
"ADVANCED": "高级检索",
"STATUS": "状态",
"OPERATION": "操作",
"CREATION_TIME": "创建时间",
"END_TIME": "结束时间",
"LOGS": "日志",
"ITEMS": "条记录"
},
"DESTINATION": {
"NEW_ENDPOINT": "新建目标",
"ENDPOINT": "目标",
"NAME": "目标名",
"NAME_IS_REQUIRED": "目标名为必填项。",
"URL": "目标URL",
"URL_IS_REQUIRED": "目标URL为必填项。",
"USERNAME": "用户名",
"PASSWORD": "密码",
"TEST_CONNECTION": "测试连接",
"TITLE_EDIT": "编辑目标",
"TITLE_ADD": "新建目标",
"DELETE": "删除目标",
"TESTING_CONNECTION": "正在测试连接...",
"TEST_CONNECTION_SUCCESS": "测试连接成功。",
"TEST_CONNECTION_FAILURE": "测试连接失败。",
"CONFLICT_NAME": "目标名或目标URL已存在。",
"INVALID_NAME": "无效的目标名称。",
"FAILED_TO_GET_TARGET": "获取目标失败。",
"CREATION_TIME": "创建时间",
"ITEMS": "条记录"
},
"REPOSITORY": {
"COPY_ID": "复制ID",
"COPY_PARENT_ID": "复制父级ID",
"DELETE": "删除",
"NAME": "名称",
"TAGS_COUNT": "标签数",
"PULL_COUNT": "下载数",
"PULL_COMMAND": "Pull命令",
"MY_REPOSITORY": "我的镜像",
"PUBLIC_REPOSITORY": "公共镜像",
"DELETION_TITLE_REPO": "删除镜像仓库确认",
"DELETION_SUMMARY_REPO": "确认删除镜像仓库 {{param}}?",
"DELETION_TITLE_TAG": "删除镜像标签确认",
"DELETION_SUMMARY_TAG": "确认删除镜像标签 {{param}}?",
"DELETION_TITLE_TAG_DENIED": "已签名的镜像不能被删除",
"DELETION_SUMMARY_TAG_DENIED": "要删除此镜像标签必须首先从Notary中删除。",
"FILTER_FOR_REPOSITORIES": "过滤镜像仓库",
"TAG": "标签",
"SIGNED": "已签名",
"AUTHOR": "作者",
"CREATED": "创建时间",
"DOCKER_VERSION": "Docker版本",
"ARCHITECTURE": "架构",
"OS": "操作系统",
"SHOW_DETAILS": "显示详细",
"REPOSITORIES": "镜像仓库",
"ITEMS": "条记录",
"POP_REPOS": "受欢迎的镜像库"
},
"ALERT": {
"FORM_CHANGE_CONFIRMATION": "表单内容改变,确认取消?"
},
"RESET_PWD": {
"TITLE": "重置密码",
"CAPTION": "输入用来重置密码的邮箱",
"EMAIL": "邮箱",
"SUCCESS": "重置密码邮件已成功发送",
"CAPTION2": "请输入您的新密码",
"RESET_OK": "密码重置成功,点击确定按钮前往登录页登录"
},
"RECENT_LOG": {
"SUB_TITLE": "显示最近",
"SUB_TITLE_SUFIX": "条日志记录"
},
"CONFIG": {
"TITLE": "配置",
"AUTH": "认证",
"REPLICATION": "复制",
"EMAIL": "邮箱",
"SYSTEM": "系统设置",
"CONFIRM_TITLE": "确认取消",
"CONFIRM_SUMMARY": "配置项有改动, 确定取消?",
"SAVE_SUCCESS": "变更的配置项成功保存",
"MAIL_SERVER": "邮件服务器",
"MAIL_SERVER_PORT": "邮件服务器端口",
"MAIL_USERNAME": "用户名",
"MAIL_PASSWORD": "密码",
"MAIL_FROM": "邮件来源",
"MAIL_SSL": "邮件 SSL",
"SSL_TOOLTIP": "应用SSL到邮件服务器连接",
"VERIFY_REMOTE_CERT": "验证远程证书",
"TOKEN_EXPIRATION": "令牌过期时间(分钟)",
"AUTH_MODE": "认证模式",
"PRO_CREATION_RESTRICTION": "项目创建限制",
"SELF_REGISTRATION": "自注册",
"AUTH_MODE_DB": "数据库",
"AUTH_MODE_LDAP": "LDAP",
"SCOPE_BASE": "基础",
"SCOPE_ONE_LEVEL": "单级",
"SCOPE_SUBTREE": "子树",
"PRO_CREATION_EVERYONE": "所有人",
"PRO_CREATION_ADMIN": "仅管理员",
"TOOLTIP": {
"SELF_REGISTRATION": "激活注册功能",
"VERIFY_REMOTE_CERT": "确定镜像复制是否要验证远程Harbor镜像库的证书。如果远程镜像库使用的是自签或者非信任证书不要勾选此选项。",
"AUTH_MODE": "默认认证模式为本地认证比如用户凭证存储在本地数据库。如果使用LDAP服务来认证用户则设置为LDAP服务。",
"LDAP_SEARCH_DN": "有权搜索LDAP服务器的用户的DN。如果LDAP服务器不支持匿名搜索则需要配置此DN之和搜索密码。",
"LDAP_BASE_DN": "用来在LDAP和AD中搜寻用户的基础DN。",
"LDAP_UID": "在搜索中用来匹配用户的属性可以是uid,cn,email,sAMAccountName或者其它LDAP/AD服务器支持的属性。",
"LDAP_SCOPE": "搜索用户的范围",
"TOKEN_EXPIRATION": "由令牌服务创建的令牌的过期时间分钟默认为30分钟。",
"PRO_CREATION_RESTRICTION": "用来控制那些用户有权创建项目的标志位,默认为’所有人‘,设置为’仅管理员‘则只有管理员可以创建项目。"
},
"LDAP": {
"URL": "LDAP地址",
"SEARCH_DN": "LDAP搜索专有名称DN)",
"SEARCH_PWD": "LDAP搜索密码",
"BASE_DN": "LDAP基础专有名称DN)",
"FILTER": "LDAP过滤器",
"UID": "LDAP用户标识UID)",
"SCOPE": "lDAP范围"
},
"TEST_MAIL_SUCCESS": "邮件服务器的连通正常",
"TEST_LDAP_SUCCESS": "LDAP服务器的连通正常",
"LEAVING_CONFIRMATION_TITLE": "确定离开?",
"LEAVING_CONFIRMATION_SUMMARY": "存在未保存的配置更改, 确认离开当前配置页?"
},
"PAGE_NOT_FOUND": {
"MAIN_TITLE": "页面不存在",
"SUB_TITLE": "重定向到harbor主页面在",
"UNIT": "秒后..."
},
"ABOUT": {
"VERSION": "版本",
"BUILD": "构建",
"COPYRIGHT": "Copyright 1998-2016 VMware. Inc. All rights reserved. This product is protected by U.S. and international property laws. VMware products are covered by one or more patents listed at",
"TRADEMARK": "Vmware is a registered trademark or trademark of VMware. Inc. in the United States and other jurisdictions. All other marks and names mentioned herein may be trademark of their respective companies.",
"END_USER_LICENSE": "终端用户许可协议",
"OPEN_SOURCE_LICENSE": "开源/第三方许可协议"
},
"START_PAGE": {
"GETTING_START": "Project Harbor is an enterprise-class registry server that stores and distributes Docker images. Harbor extends the open source Docker Distribution by adding the functionalities usually required by an enterprise, such as security, identity and management. As an enterprise private registry, Harbor offers better performance and security.",
"GETTING_START_TITLE": "从这开始"
},
"TOP_REPO": "受欢迎镜像库",
"STATISTICS": {
"TITLE": "统计",
"PRO_ITEM": "项目",
"REPO_ITEM": "镜像库",
"INDEX_MY": "私有的",
"INDEX_PUB": "公开的",
"INDEX_TOTAL": "总计"
},
"SEARCH": {
"IN_PROGRESS": "搜索中...",
"BACK": "返回"
},
"UNKNOWN_ERROR": "发生未知错误,请稍后再试",
"UNAUTHORIZED_ERROR": "会话无效或者已经过期, 请重新登录以继续",
"FORBIDDEN_ERROR": "当前操作被禁止,请确认你有合法的权限"
}

View File

@ -3,6 +3,10 @@ import { Headers, Http, RequestOptions } from '@angular/http';
import 'rxjs/add/operator/toPromise'; import 'rxjs/add/operator/toPromise';
import { AppConfig } from './app-config'; import { AppConfig } from './app-config';
import { CookieService } from 'angular2-cookie/core';
import { CookieKeyOfAdmiral, HarborQueryParamKey } from './shared/shared.const';
import { maintainUrlQueryParmas } from './shared/shared.utils';
export const systemInfoEndpoint = "/api/systeminfo"; export const systemInfoEndpoint = "/api/systeminfo";
/** /**
@ -24,15 +28,29 @@ export class AppConfigService {
//Store the application configuration //Store the application configuration
private configurations: AppConfig = new AppConfig(); private configurations: AppConfig = new AppConfig();
constructor(private http: Http) { } constructor(
private http: Http,
private cookie: CookieService) { }
public load(): Promise<AppConfig> { public load(): Promise<AppConfig> {
return this.http.get(systemInfoEndpoint, this.options).toPromise() return this.http.get(systemInfoEndpoint, this.options).toPromise()
.then(response => this.configurations = response.json() as AppConfig) .then(response => {
.catch(error => { this.configurations = response.json() as AppConfig;
//Catch the error
console.error("Failed to load bootstrap options with error: ", error); //Read admiral endpoint from cookie if existing
}); let admiralUrlFromCookie: string = this.cookie.get(CookieKeyOfAdmiral);
if(admiralUrlFromCookie){
//Override the endpoint from configuration file
this.configurations.admiral_endpoint = decodeURIComponent(admiralUrlFromCookie);
}
return this.configurations;
})
.catch(error => {
//Catch the error
console.error("Failed to load bootstrap options with error: ", error);
});
} }
public getConfig(): AppConfig { public getConfig(): AppConfig {
@ -41,7 +59,27 @@ export class AppConfigService {
public isIntegrationMode(): boolean { public isIntegrationMode(): boolean {
return this.configurations && return this.configurations &&
this.configurations.with_admiral && this.configurations.with_admiral &&
this.configurations.admiral_endpoint.trim() != ""; this.configurations.admiral_endpoint.trim() != "";
}
//Return the reconstructed admiral url
public getAdmiralEndpoint(currentHref: string): string {
let admiralUrl:string = this.configurations.admiral_endpoint;
if(admiralUrl.trim() === "" || currentHref.trim() === ""){
return "#";
}
return maintainUrlQueryParmas(admiralUrl, HarborQueryParamKey, encodeURIComponent(currentHref));
}
public saveAdmiralEndpoint(endpoint: string): void {
if(!(endpoint.trim())){
return;
}
//Save back to cookie
this.cookie.put(CookieKeyOfAdmiral, endpoint);
this.configurations.admiral_endpoint = endpoint;
} }
} }

View File

@ -55,13 +55,7 @@ export class NavigatorComponent implements OnInit {
} }
public get admiralLink(): string { public get admiralLink(): string {
let appConfig = this.appConfigService.getConfig(); return this.appConfigService.getAdmiralEndpoint(window.location.href);
let routeSegments = [appConfig.admiral_endpoint,
"?registry_url=",
encodeURIComponent(window.location.href)
];
return routeSegments.join("");
} }
public get isIntegrationMode(): boolean { public get isIntegrationMode(): boolean {

View File

@ -94,7 +94,7 @@ export class MessageComponent implements OnInit {
} }
signIn(): void { signIn(): void {
this.router.navigate([CommonRoutes.EMBEDDED_SIGN_IN]); this.router.navigateByUrl(CommonRoutes.EMBEDDED_SIGN_IN);
} }
onClose() { onClose() {

View File

@ -27,7 +27,7 @@
</form> </form>
</div> </div>
<div class="modal-footer"> <div class="modal-footer">
<button type="button" class="btn btn-outline" (click)="createProjectOpened = false">{{'BUTTON.CANCEL' | translate}}</button> <button type="button" class="btn btn-outline" (click)="onCancel()">{{'BUTTON.CANCEL' | translate}}</button>
<button type="button" class="btn btn-primary" [disabled]="!projectForm.form.valid" (click)="onSubmit()">{{'BUTTON.OK' | translate}}</button> <button type="button" class="btn btn-primary" [disabled]="!projectForm.form.valid" (click)="onSubmit()">{{'BUTTON.OK' | translate}}</button>
</div> </div>
</clr-modal> </clr-modal>

View File

@ -1,5 +1,7 @@
import { Component, EventEmitter, Output, ViewChild } from '@angular/core'; import { Component, EventEmitter, Output, ViewChild, AfterViewChecked } from '@angular/core';
import { Response } from '@angular/http'; import { Response } from '@angular/http';
import { NgForm } from '@angular/forms';
import { Project } from '../project'; import { Project } from '../project';
import { ProjectService } from '../project.service'; import { ProjectService } from '../project.service';
@ -17,13 +19,19 @@ import { TranslateService } from '@ngx-translate/core';
templateUrl: 'create-project.component.html', templateUrl: 'create-project.component.html',
styleUrls: [ 'create-project.css' ] styleUrls: [ 'create-project.css' ]
}) })
export class CreateProjectComponent { export class CreateProjectComponent implements AfterViewChecked {
projectForm: NgForm;
@ViewChild('projectForm')
currentForm: NgForm;
project: Project = new Project(); project: Project = new Project();
createProjectOpened: boolean; createProjectOpened: boolean;
errorMessageOpened: boolean;
errorMessage: string; hasChanged: boolean;
@Output() create = new EventEmitter<boolean>(); @Output() create = new EventEmitter<boolean>();
@ViewChild(InlineAlertComponent) @ViewChild(InlineAlertComponent)
@ -42,40 +50,65 @@ export class CreateProjectComponent {
this.createProjectOpened = false; this.createProjectOpened = false;
}, },
error=>{ error=>{
this.errorMessageOpened = true; let errorMessage: string;
if (error instanceof Response) { if (error instanceof Response) {
switch(error.status) { switch(error.status) {
case 409: case 409:
this.translateService.get('PROJECT.NAME_ALREADY_EXISTS').subscribe(res=>this.errorMessage = res); this.translateService.get('PROJECT.NAME_ALREADY_EXISTS').subscribe(res=>errorMessage = res);
break; break;
case 400: case 400:
this.translateService.get('PROJECT.NAME_IS_ILLEGAL').subscribe(res=>this.errorMessage = res); this.translateService.get('PROJECT.NAME_IS_ILLEGAL').subscribe(res=>errorMessage = res);
break; break;
default: default:
this.translateService.get('PROJECT.UNKNOWN_ERROR').subscribe(res=>{ this.translateService.get('PROJECT.UNKNOWN_ERROR').subscribe(res=>{
this.errorMessage = res; errorMessage = res;
this.messageService.announceMessage(error.status, this.errorMessage, AlertType.DANGER); this.messageService.announceMessage(error.status, errorMessage, AlertType.DANGER);
}); });
} }
this.inlineAlert.showInlineError(this.errorMessage); this.inlineAlert.showInlineError(errorMessage);
} }
}); });
} }
newProject() { onCancel() {
this.project = new Project(); if(this.hasChanged) {
this.createProjectOpened = true; this.inlineAlert.showInlineConfirmation({message: 'ALERT.FORM_CHANGE_CONFIRMATION'});
this.errorMessageOpened = false; } else {
this.errorMessage = ''; this.createProjectOpened = false;
}
} }
onErrorMessageClose(): void { ngAfterViewChecked(): void {
this.errorMessageOpened = false; this.projectForm = this.currentForm;
this.errorMessage = ''; if(this.projectForm) {
this.projectForm.valueChanges.subscribe(data=>{
for(let i in data) {
let item = data[i];
if(typeof item === 'string' && (<string>item).trim().length !== 0) {
this.hasChanged = true;
break;
} else if (typeof item === 'boolean' && (<boolean>item)) {
this.hasChanged = true;
break;
} else {
this.hasChanged = false;
this.inlineAlert.close();
break;
}
}
});
}
}
newProject() {
this.project = new Project();
this.hasChanged = false;
this.createProjectOpened = true;
} }
confirmCancel(event: boolean): void { confirmCancel(event: boolean): void {
this.errorMessageOpened = false; this.createProjectOpened = false;
this.inlineAlert.close();
} }
} }

View File

@ -1,5 +1,6 @@
<clr-modal [(clrModalOpen)]="addMemberOpened"> <clr-modal [(clrModalOpen)]="addMemberOpened">
<h3 class="modal-title">{{'MEMBER.NEW_MEMBER' | translate}}</h3> <h3 class="modal-title">{{'MEMBER.NEW_MEMBER' | translate}}</h3>
<inline-alert class="modal-title" (confirmEvt)="confirmCancel($event)"></inline-alert>
<div class="modal-body"> <div class="modal-body">
<form #memberForm="ngForm"> <form #memberForm="ngForm">
<section class="form-block"> <section class="form-block">
@ -22,15 +23,15 @@
<div class="form-group"> <div class="form-group">
<label class="col-md-4">{{'MEMBER.ROLE' | translate}}</label> <label class="col-md-4">{{'MEMBER.ROLE' | translate}}</label>
<div class="radio"> <div class="radio">
<input type="radio" name="roleRadios" id="checkrads_project_admin" (click)="member.role_id = 1" [checked]="member.role_id === 1"> <input type="radio" name="roleRadios" id="checkrads_project_admin" value="1" [(ngModel)]="member.role_id">
<label for="checkrads_project_admin">{{'MEMBER.PROJECT_ADMIN' | translate}}</label> <label for="checkrads_project_admin">{{'MEMBER.PROJECT_ADMIN' | translate}}</label>
</div> </div>
<div class="radio"> <div class="radio">
<input type="radio" name="roleRadios" id="checkrads_developer" (click)="member.role_id = 2" [checked]="member.role_id === 2"> <input type="radio" name="roleRadios" id="checkrads_developer" value="2" [(ngModel)]="member.role_id">
<label for="checkrads_developer">{{'MEMBER.DEVELOPER' | translate}}</label> <label for="checkrads_developer">{{'MEMBER.DEVELOPER' | translate}}</label>
</div> </div>
<div class="radio"> <div class="radio">
<input type="radio" name="roleRadios" id="checkrads_guest" (click)="member.role_id = 3" [checked]="member.role_id === 3"> <input type="radio" name="roleRadios" id="checkrads_guest" value="3" [(ngModel)]="member.role_id">
<label for="checkrads_guest">{{'MEMBER.GUEST' | translate}}</label> <label for="checkrads_guest">{{'MEMBER.GUEST' | translate}}</label>
</div> </div>
</div> </div>
@ -38,7 +39,7 @@
</form> </form>
</div> </div>
<div class="modal-footer"> <div class="modal-footer">
<button type="button" class="btn btn-outline" (click)="addMemberOpened = false">{{'BUTTON.CANCEL' | translate}}</button> <button type="button" class="btn btn-outline" (click)="onCancel()">{{'BUTTON.CANCEL' | translate}}</button>
<button type="button" class="btn btn-primary" (click)="onSubmit()">{{'BUTTON.OK' | translate}}</button> <button type="button" class="btn btn-primary" (click)="onSubmit()">{{'BUTTON.OK' | translate}}</button>
</div> </div>
</clr-modal> </clr-modal>

View File

@ -1,10 +1,14 @@
import { Component, Input, EventEmitter, Output } from '@angular/core'; import { Component, Input, EventEmitter, Output, ViewChild, AfterViewChecked } from '@angular/core';
import { Response } from '@angular/http'; import { Response } from '@angular/http';
import { NgForm } from '@angular/forms';
import { MemberService } from '../member.service'; import { MemberService } from '../member.service';
import { MessageService } from '../../../global-message/message.service'; import { MessageService } from '../../../global-message/message.service';
import { AlertType } from '../../../shared/shared.const'; import { AlertType } from '../../../shared/shared.const';
import { InlineAlertComponent } from '../../../shared/inline-alert/inline-alert.component';
import { TranslateService } from '@ngx-translate/core'; import { TranslateService } from '@ngx-translate/core';
import { Member } from '../member'; import { Member } from '../member';
@ -13,14 +17,20 @@ import { Member } from '../member';
selector: 'add-member', selector: 'add-member',
templateUrl: 'add-member.component.html' templateUrl: 'add-member.component.html'
}) })
export class AddMemberComponent { export class AddMemberComponent implements AfterViewChecked {
member: Member = new Member(); member: Member = new Member();
addMemberOpened: boolean; addMemberOpened: boolean;
errorMessage: string;
errorMessageOpened: boolean; memberForm: NgForm;
@ViewChild('memberForm')
currentForm: NgForm;
hasChanged: boolean;
@ViewChild(InlineAlertComponent)
inlineAlert: InlineAlertComponent;
@Input() projectId: number; @Input() projectId: number;
@Output() added = new EventEmitter<boolean>(); @Output() added = new EventEmitter<boolean>();
@ -32,7 +42,7 @@ export class AddMemberComponent {
onSubmit(): void { onSubmit(): void {
console.log('Adding member:' + JSON.stringify(this.member)); console.log('Adding member:' + JSON.stringify(this.member));
this.memberService this.memberService
.addMember(this.projectId, this.member.username, this.member.role_id) .addMember(this.projectId, this.member.username, +this.member.role_id)
.subscribe( .subscribe(
response=>{ response=>{
console.log('Added member successfully.'); console.log('Added member successfully.');
@ -40,37 +50,69 @@ export class AddMemberComponent {
this.addMemberOpened = false; this.addMemberOpened = false;
}, },
error=>{ error=>{
this.errorMessageOpened = true;
if (error instanceof Response) { if (error instanceof Response) {
let errorMessageKey: string;
switch(error.status){ switch(error.status){
case 404: case 404:
this.translateService.get('MEMBER.USERNAME_DOES_NOT_EXISTS').subscribe(res=>this.errorMessage = res); errorMessageKey = 'MEMBER.USERNAME_DOES_NOT_EXISTS';
break; break;
case 409: case 409:
this.translateService.get('MEMBER.USERNAME_ALREADY_EXISTS').subscribe(res=>this.errorMessage = res); errorMessageKey = 'MEMBER.USERNAME_ALREADY_EXISTS';
break; break;
default: default:
this.translateService.get('MEMBER.UNKNOWN_ERROR').subscribe(res=>{ errorMessageKey = 'MEMBER.UNKNOWN_ERROR';
this.errorMessage = res;
this.messageService.announceMessage(error.status, this.errorMessage, AlertType.DANGER);
});
} }
this.translateService
.get(errorMessageKey)
.subscribe(errorMessage=>this.inlineAlert.showInlineError(errorMessage));
} }
console.log('Failed to add member of project:' + this.projectId, ' with error:' + error); console.log('Failed to add member of project:' + this.projectId, ' with error:' + error);
} }
); );
} }
openAddMemberModal(): void { onCancel() {
this.errorMessageOpened = false; if(this.hasChanged) {
this.errorMessage = ''; this.inlineAlert.showInlineConfirmation({message: 'ALERT.FORM_CHANGE_CONFIRMATION'});
this.member = new Member(); } else {
this.addMemberOpened = true; this.addMemberOpened = false;
}
} }
onErrorMessageClose(): void { ngAfterViewChecked(): void {
this.errorMessageOpened = false; this.memberForm = this.currentForm;
this.errorMessage = ''; if(this.memberForm) {
this.memberForm.valueChanges.subscribe(data=>{
for(let i in data) {
let item = data[i];
if(typeof item === 'string' && (<string>item).trim().length !== 0) {
this.hasChanged = true;
break;
} else if (typeof item === 'boolean' && (<boolean>item)) {
this.hasChanged = true;
break;
} else if (typeof item === 'number' && (<number>item) !== 0) {
this.hasChanged = true;
break;
} else {
this.hasChanged = false;
this.inlineAlert.close();
break;
}
}
});
}
} }
confirmCancel(confirmed: boolean) {
this.addMemberOpened = false;
this.inlineAlert.close();
}
openAddMemberModal(): void {
this.member = new Member();
this.addMemberOpened = true;
this.hasChanged = false;
}
} }

View File

@ -30,28 +30,4 @@
<clr-dg-footer>{{ (members ? members.length : 0) }} {{'MEMBER.ITEMS' | translate}}</clr-dg-footer> <clr-dg-footer>{{ (members ? members.length : 0) }} {{'MEMBER.ITEMS' | translate}}</clr-dg-footer>
</clr-datagrid> </clr-datagrid>
</div> </div>
</div>
<div class="col-lg-12 col-md-12 datagrid-margin-top">
<clr-datagrid>
<clr-dg-column>{{'MEMBER.NAME' | translate}}</clr-dg-column>
<clr-dg-column>{{'MEMBER.ROLE' | translate}}</clr-dg-column>
<clr-dg-column style="width: 12px;" class="datagrid-column-ext"></clr-dg-column>
<clr-dg-row *ngFor="let u of members">
<clr-dg-cell>{{u.username}}</clr-dg-cell>
<clr-dg-cell>
{{roleInfo[u.role_id] | translate}}
</clr-dg-cell>
<clr-dg-cell>
<harbor-action-overflow [hidden]="u.user_id === currentUser.user_id" class="habor-action-pos">
<a href="javascript:void(0)" class="dropdown-item" (click)="changeRole(u.user_id, 1)">{{'MEMBER.PROJECT_ADMIN' | translate}}</a>
<a href="javascript:void(0)" class="dropdown-item" (click)="changeRole(u.user_id, 2)">{{'MEMBER.DEVELOPER' | translate}}</a>
<a href="javascript:void(0)" class="dropdown-item" (click)="changeRole(u.user_id, 3)">{{'MEMBER.GUEST' | translate}}</a>
<div class="dropdown-divider"></div>
<a href="javascript:void(0)" class="dropdown-item" (click)="deleteMember(u.user_id)">{{'MEMBER.DELETE' | translate}}</a>
</harbor-action-overflow>
</clr-dg-cell>
</clr-dg-row>
<clr-dg-footer>{{ (members ? members.length : 0) }} {{'MEMBER.ITEMS' | translate}}</clr-dg-footer>
</clr-datagrid>
</div>
</div> </div>

View File

@ -1,5 +1,6 @@
<clr-modal [(clrModalOpen)]="createEditDestinationOpened"> <clr-modal [(clrModalOpen)]="createEditDestinationOpened">
<h3 class="modal-title">{{modalTitle}}</h3> <h3 class="modal-title">{{modalTitle}}</h3>
<inline-alert class="modal-title" (confirmEvt)="confirmCancel($event)"></inline-alert>
<div class="modal-body"> <div class="modal-body">
<form #targetForm="ngForm"> <form #targetForm="ngForm">
<section class="form-block"> <section class="form-block">
@ -46,7 +47,7 @@
</div> </div>
<div class="modal-footer"> <div class="modal-footer">
<button type="button" class="btn btn-outline" (click)="testConnection()" [disabled]="testOngoing || targetEndpoint.errors">{{ 'DESTINATION.TEST_CONNECTION' | translate }}</button> <button type="button" class="btn btn-outline" (click)="testConnection()" [disabled]="testOngoing || targetEndpoint.errors">{{ 'DESTINATION.TEST_CONNECTION' | translate }}</button>
<button type="button" class="btn btn-outline" (click)="createEditDestinationOpened = false" [disabled]="testOngoing">{{ 'BUTTON.CANCEL' | translate }}</button> <button type="button" class="btn btn-outline" (click)="onCancel()" [disabled]="testOngoing">{{ 'BUTTON.CANCEL' | translate }}</button>
<button type="submit" class="btn btn-primary" [disabled]="!targetForm.form.valid" (click)="onSubmit()" [disabled]="testOngoing">{{ 'BUTTON.OK' | translate }}</button> <button type="submit" class="btn btn-primary" [disabled]="!targetForm.form.valid" (click)="onSubmit()" [disabled]="testOngoing">{{ 'BUTTON.OK' | translate }}</button>
</div> </div>
</clr-modal> </clr-modal>

View File

@ -1,9 +1,12 @@
import { Component, Output, EventEmitter } from '@angular/core'; import { Component, Output, EventEmitter, ViewChild, AfterViewChecked } from '@angular/core';
import { NgForm } from '@angular/forms';
import { ReplicationService } from '../replication.service'; import { ReplicationService } from '../replication.service';
import { MessageService } from '../../global-message/message.service'; import { MessageService } from '../../global-message/message.service';
import { AlertType, ActionType } from '../../shared/shared.const'; import { AlertType, ActionType } from '../../shared/shared.const';
import { InlineAlertComponent } from '../../shared/inline-alert/inline-alert.component';
import { Target } from '../target'; import { Target } from '../target';
import { TranslateService } from '@ngx-translate/core'; import { TranslateService } from '@ngx-translate/core';
@ -17,9 +20,6 @@ export class CreateEditDestinationComponent {
modalTitle: string; modalTitle: string;
createEditDestinationOpened: boolean; createEditDestinationOpened: boolean;
errorMessageOpened: boolean;
errorMessage: string;
testOngoing: boolean; testOngoing: boolean;
pingTestMessage: string; pingTestMessage: string;
pingStatus: boolean; pingStatus: boolean;
@ -28,6 +28,16 @@ export class CreateEditDestinationComponent {
target: Target = new Target(); target: Target = new Target();
targetForm: NgForm;
@ViewChild('targetForm')
currentForm: NgForm;
hasChanged: boolean;
@ViewChild(InlineAlertComponent)
inlineAlert: InlineAlertComponent;
@Output() reload = new EventEmitter<boolean>(); @Output() reload = new EventEmitter<boolean>();
constructor( constructor(
@ -40,8 +50,7 @@ export class CreateEditDestinationComponent {
this.createEditDestinationOpened = true; this.createEditDestinationOpened = true;
this.errorMessageOpened = false; this.hasChanged = false;
this.errorMessage = '';
this.pingTestMessage = ''; this.pingTestMessage = '';
this.pingStatus = true; this.pingStatus = true;
@ -84,9 +93,6 @@ export class CreateEditDestinationComponent {
} }
onSubmit() { onSubmit() {
this.errorMessage = '';
this.errorMessageOpened = false;
switch(this.actionType) { switch(this.actionType) {
case ActionType.ADD_NEW: case ActionType.ADD_NEW:
this.replicationService this.replicationService
@ -98,7 +104,6 @@ export class CreateEditDestinationComponent {
this.reload.emit(true); this.reload.emit(true);
}, },
error=>{ error=>{
this.errorMessageOpened = true;
let errorMessageKey = ''; let errorMessageKey = '';
switch(error.status) { switch(error.status) {
case 409: case 409:
@ -110,11 +115,12 @@ export class CreateEditDestinationComponent {
default: default:
errorMessageKey = 'UNKNOWN_ERROR'; errorMessageKey = 'UNKNOWN_ERROR';
} }
this.translateService this.translateService
.get(errorMessageKey) .get(errorMessageKey)
.subscribe(res=>{ .subscribe(res=>{
this.errorMessage = res;
this.messageService.announceMessage(error.status, errorMessageKey, AlertType.DANGER); this.messageService.announceMessage(error.status, errorMessageKey, AlertType.DANGER);
this.inlineAlert.showInlineError(errorMessageKey);
}); });
} }
); );
@ -129,8 +135,6 @@ export class CreateEditDestinationComponent {
this.reload.emit(true); this.reload.emit(true);
}, },
error=>{ error=>{
this.errorMessageOpened = true;
this.errorMessage = 'Failed to update target:' + error;
let errorMessageKey = ''; let errorMessageKey = '';
switch(error.status) { switch(error.status) {
case 409: case 409:
@ -145,7 +149,7 @@ export class CreateEditDestinationComponent {
this.translateService this.translateService
.get(errorMessageKey) .get(errorMessageKey)
.subscribe(res=>{ .subscribe(res=>{
this.errorMessage = res; this.inlineAlert.showInlineError(errorMessageKey);
this.messageService.announceMessage(error.status, errorMessageKey, AlertType.DANGER); this.messageService.announceMessage(error.status, errorMessageKey, AlertType.DANGER);
}); });
} }
@ -154,9 +158,39 @@ export class CreateEditDestinationComponent {
} }
} }
onErrorMessageClose(): void { onCancel() {
this.errorMessageOpened = false; if(this.hasChanged) {
this.errorMessage = ''; this.inlineAlert.showInlineConfirmation({message: 'ALERT.FORM_CHANGE_CONFIRMATION'});
} else {
this.createEditDestinationOpened = false;
}
}
confirmCancel(confirmed: boolean) {
this.createEditDestinationOpened = false;
this.inlineAlert.close();
}
ngAfterViewChecked(): void {
this.targetForm = this.currentForm;
if(this.targetForm) {
this.targetForm.valueChanges.subscribe(data=>{
for(let i in data) {
let item = data[i];
if(typeof item === 'string' && (<string>item).trim().length !== 0) {
this.hasChanged = true;
break;
} else if (typeof item === 'boolean' && (<boolean>item)) {
this.hasChanged = true;
break;
} else {
this.hasChanged = false;
this.inlineAlert.close();
break;
}
}
});
}
} }
} }

View File

@ -1,5 +1,6 @@
<clr-modal [(clrModalOpen)]="createEditPolicyOpened"> <clr-modal [(clrModalOpen)]="createEditPolicyOpened">
<h3 class="modal-title">{{modalTitle}}</h3> <h3 class="modal-title">{{modalTitle}}</h3>
<inline-alert class="modal-title" (confirmEvt)="confirmCancel($event)"></inline-alert>
<div class="modal-body"> <div class="modal-body">
<form #policyForm="ngForm"> <form #policyForm="ngForm">
<section class="form-block"> <section class="form-block">
@ -75,7 +76,7 @@
</div> </div>
<div class="modal-footer"> <div class="modal-footer">
<button type="button" class="btn btn-outline" (click)="testConnection()" [disabled]="testOngoing">{{'REPLICATION.TEST_CONNECTION' | translate}}</button> <button type="button" class="btn btn-outline" (click)="testConnection()" [disabled]="testOngoing">{{'REPLICATION.TEST_CONNECTION' | translate}}</button>
<button type="button" class="btn btn-outline" (click)="createEditPolicyOpened = false">{{'BUTTON.CANCEL' | translate }}</button> <button type="button" class="btn btn-outline" (click)="onCancel()">{{'BUTTON.CANCEL' | translate }}</button>
<button type="submit" class="btn btn-primary" [disabled]="!policyForm.form.valid" (click)="onSubmit()">{{'BUTTON.OK' | translate}}</button> <button type="submit" class="btn btn-primary" [disabled]="!policyForm.form.valid" (click)="onSubmit()">{{'BUTTON.OK' | translate}}</button>
</div> </div>
</clr-modal> </clr-modal>

View File

@ -1,4 +1,6 @@
import { Component, Input, Output, EventEmitter, OnInit, HostBinding } from '@angular/core'; import { Component, Input, Output, EventEmitter, OnInit, ViewChild, AfterViewChecked } from '@angular/core';
import { NgForm } from '@angular/forms';
import { CreateEditPolicy } from './create-edit-policy'; import { CreateEditPolicy } from './create-edit-policy';
@ -6,6 +8,8 @@ import { ReplicationService } from '../../replication/replication.service';
import { MessageService } from '../../global-message/message.service'; import { MessageService } from '../../global-message/message.service';
import { AlertType, ActionType } from '../../shared/shared.const'; import { AlertType, ActionType } from '../../shared/shared.const';
import { InlineAlertComponent } from '../../shared/inline-alert/inline-alert.component';
import { Policy } from '../../replication/policy'; import { Policy } from '../../replication/policy';
import { Target } from '../../replication/target'; import { Target } from '../../replication/target';
@ -15,7 +19,7 @@ import { TranslateService } from '@ngx-translate/core';
selector: 'create-edit-policy', selector: 'create-edit-policy',
templateUrl: 'create-edit-policy.component.html' templateUrl: 'create-edit-policy.component.html'
}) })
export class CreateEditPolicyComponent implements OnInit { export class CreateEditPolicyComponent implements OnInit, AfterViewChecked {
modalTitle: string; modalTitle: string;
createEditPolicyOpened: boolean; createEditPolicyOpened: boolean;
@ -23,9 +27,6 @@ export class CreateEditPolicyComponent implements OnInit {
actionType: ActionType; actionType: ActionType;
errorMessageOpened: boolean;
errorMessage: string;
isCreateDestination: boolean; isCreateDestination: boolean;
@Input() projectId: number; @Input() projectId: number;
@ -37,6 +38,16 @@ export class CreateEditPolicyComponent implements OnInit {
testOngoing: boolean; testOngoing: boolean;
pingStatus: boolean; pingStatus: boolean;
policyForm: NgForm;
@ViewChild('policyForm')
currentForm: NgForm;
hasChanged: boolean;
@ViewChild(InlineAlertComponent)
inlineAlert: InlineAlertComponent;
constructor( constructor(
private replicationService: ReplicationService, private replicationService: ReplicationService,
private messageService: MessageService, private messageService: MessageService,
@ -68,8 +79,8 @@ export class CreateEditPolicyComponent implements OnInit {
this.createEditPolicyOpened = true; this.createEditPolicyOpened = true;
this.createEditPolicy = new CreateEditPolicy(); this.createEditPolicy = new CreateEditPolicy();
this.isCreateDestination = false; this.isCreateDestination = false;
this.errorMessageOpened = false;
this.errorMessage = ''; this.hasChanged = false;
this.pingTestMessage = ''; this.pingTestMessage = '';
this.pingStatus = true; this.pingStatus = true;
@ -119,11 +130,6 @@ export class CreateEditPolicyComponent implements OnInit {
} }
} }
onErrorMessageClose(): void {
this.errorMessageOpened = false;
this.errorMessage = '';
}
getPolicyByForm(): Policy { getPolicyByForm(): Policy {
let policy = new Policy(); let policy = new Policy();
policy.project_id = this.projectId; policy.project_id = this.projectId;
@ -156,8 +162,7 @@ export class CreateEditPolicyComponent implements OnInit {
this.reload.emit(true); this.reload.emit(true);
}, },
error=>{ error=>{
this.errorMessageOpened = true; this.inlineAlert.showInlineError(error['_body']);
this.errorMessage = error['_body'];
console.log('Failed to create policy:' + error.status + ', error message:' + JSON.stringify(error['_body'])); console.log('Failed to create policy:' + error.status + ', error message:' + JSON.stringify(error['_body']));
}); });
} }
@ -173,8 +178,7 @@ export class CreateEditPolicyComponent implements OnInit {
this.reload.emit(true); this.reload.emit(true);
}, },
error=>{ error=>{
this.errorMessageOpened = true; this.inlineAlert.showInlineError(error['_body']);
this.errorMessage = error['_body'];
console.log('Failed to create policy and target:' + error.status + ', error message:' + JSON.stringify(error['_body'])); console.log('Failed to create policy and target:' + error.status + ', error message:' + JSON.stringify(error['_body']));
} }
); );
@ -191,8 +195,7 @@ export class CreateEditPolicyComponent implements OnInit {
this.reload.emit(true); this.reload.emit(true);
}, },
error=>{ error=>{
this.errorMessageOpened = true; this.inlineAlert.showInlineError(error['_body']);
this.errorMessage = error['_body'];
console.log('Failed to create policy and target:' + error.status + ', error message:' + JSON.stringify(error['_body'])); console.log('Failed to create policy and target:' + error.status + ', error message:' + JSON.stringify(error['_body']));
} }
); );
@ -208,11 +211,44 @@ export class CreateEditPolicyComponent implements OnInit {
this.updatePolicy(); this.updatePolicy();
} }
} }
this.errorMessageOpened = false;
this.errorMessage = '';
} }
onCancel() {
if(this.hasChanged) {
this.inlineAlert.showInlineConfirmation({message: 'ALERT.FORM_CHANGE_CONFIRMATION'});
} else {
this.createEditPolicyOpened = false;
}
}
confirmCancel(confirmed: boolean) {
this.createEditPolicyOpened = false;
this.inlineAlert.close();
}
ngAfterViewChecked(): void {
this.policyForm = this.currentForm;
if(this.policyForm) {
this.policyForm.valueChanges.subscribe(data=>{
for(let i in data) {
let item = data[i];
if(typeof item === 'string' && (<string>item).trim().length !== 0) {
this.hasChanged = true;
break;
} else if (typeof item === 'boolean' && (<boolean>item)) {
this.hasChanged = true;
break;
} else {
this.hasChanged = false;
this.inlineAlert.close();
break;
}
}
});
}
}
testConnection() { testConnection() {
this.pingStatus = true; this.pingStatus = true;
this.translateService.get('REPLICATION.TESTING_CONNECTION').subscribe(res=>this.pingTestMessage=res); this.translateService.get('REPLICATION.TESTING_CONNECTION').subscribe(res=>this.pingTestMessage=res);

View File

@ -6,9 +6,12 @@ import {
CanActivateChild, CanActivateChild,
NavigationExtras NavigationExtras
} from '@angular/router'; } from '@angular/router';
import { SessionService } from '../../shared/session.service'; import { SessionService } from '../../shared/session.service';
import { CommonRoutes, AdmiralQueryParamKey } from '../../shared/shared.const'; import { CommonRoutes, AdmiralQueryParamKey } from '../../shared/shared.const';
import { AppConfigService } from '../../app-config.service'; import { AppConfigService } from '../../app-config.service';
import { maintainUrlQueryParmas } from '../../shared/shared.utils';
@Injectable() @Injectable()
export class AuthCheckGuard implements CanActivate, CanActivateChild { export class AuthCheckGuard implements CanActivate, CanActivateChild {
@ -38,7 +41,16 @@ export class AuthCheckGuard implements CanActivate, CanActivateChild {
let queryParams = route.queryParams; let queryParams = route.queryParams;
if (queryParams) { if (queryParams) {
if (queryParams[AdmiralQueryParamKey]) { if (queryParams[AdmiralQueryParamKey]) {
console.debug(queryParams[AdmiralQueryParamKey]); this.appConfigService.saveAdmiralEndpoint(queryParams[AdmiralQueryParamKey]);
//Remove the query parameter key pair and redirect
let keyRemovedUrl = maintainUrlQueryParmas(state.url, AdmiralQueryParamKey, undefined);
if(!/[?]{1}.+/i.test(keyRemovedUrl)){
keyRemovedUrl = keyRemovedUrl.replace('?','');
}
this.router.navigateByUrl(keyRemovedUrl);
return resolve(false);
} }
} }

View File

@ -45,6 +45,8 @@ export const CommonRoutes = {
}; };
export const AdmiralQueryParamKey = "admiral_redirect_url"; export const AdmiralQueryParamKey = "admiral_redirect_url";
export const HarborQueryParamKey = "harbor_redirect_url";
export const CookieKeyOfAdmiral = "admiral.endpoint.latest";
export const enum ConfirmationState { export const enum ConfirmationState {
NA, CONFIRMED, CANCEL NA, CONFIRMED, CANCEL

View File

@ -60,3 +60,27 @@ export const accessErrorHandler = function (error: any, msgService: MessageServi
return false; return false;
} }
//Provide capability of reconstructing the query paramter
export const maintainUrlQueryParmas = function (uri: string, key: string, value: string): string {
let re: RegExp = new RegExp("([?&])" + key + "=.*?(&|#|$)", "i");
if (value === undefined) {
if (uri.match(re)) {
return uri.replace(re, '$1$2');
} else {
return uri;
}
} else {
if (uri.match(re)) {
return uri.replace(re, '$1' + key + "=" + value + '$2');
} else {
var hash = '';
if (uri.indexOf('#') !== -1) {
hash = uri.replace(/.*#/, '#');
uri = uri.replace(/#.*/, '');
}
var separator = uri.indexOf('?') !== -1 ? "&" : "?";
return uri + separator + key + "=" + value + hash;
}
}
}