FeNameFormat.java
// Licensed to the Apache Software Foundation (ASF) under one
// or more contributor license agreements. See the NOTICE file
// distributed with this work for additional information
// regarding copyright ownership. The ASF licenses this file
// to you under the Apache License, Version 2.0 (the
// "License"); you may not use this file except in compliance
// with the License. You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing,
// software distributed under the License is distributed on an
// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
// KIND, either express or implied. See the License for the
// specific language governing permissions and limitations
// under the License.
package org.apache.doris.common;
import org.apache.doris.alter.SchemaChangeHandler;
import org.apache.doris.analysis.CreateMaterializedViewStmt;
import org.apache.doris.analysis.ResourceTypeEnum;
import org.apache.doris.datasource.InternalCatalog;
import org.apache.doris.mysql.privilege.Role;
import org.apache.doris.mysql.privilege.RoleManager;
import org.apache.doris.qe.ConnectContext;
import org.apache.doris.qe.VariableMgr;
import com.google.common.base.Strings;
public class FeNameFormat {
private static final String LABEL_REGEX = "^[\\-_A-Za-z0-9:]{1," + Config.label_regex_length + "}$";
// if modify the matching length of a regular expression,
// please modify error msg when FeNameFormat.checkCommonName throw exception in CreateRoutineLoadStmt
private static final String COMMON_NAME_REGEX = "^[a-zA-Z][a-zA-Z0-9\\-_]{0,63}$";
private static final String UNDERSCORE_COMMON_NAME_REGEX = "^[_a-zA-Z][a-zA-Z0-9\\-_]{0,63}$";
private static final String TABLE_NAME_REGEX = "^[a-zA-Z][a-zA-Z0-9\\-_]*$";
private static final String USER_NAME_REGEX = "^[a-zA-Z][a-zA-Z0-9.\\-_]*$";
private static final String REPOSITORY_NAME_REGEX = "^[a-zA-Z][a-zA-Z0-9\\-_]{0,255}$";
private static final String COLUMN_NAME_REGEX = "^[.a-zA-Z0-9_+\\-/?@#$%^&*\"\\s,:]{1,256}$";
private static final String UNICODE_LABEL_REGEX = "^[\\-_A-Za-z0-9:\\p{L}]{1," + Config.label_regex_length + "}$";
private static final String UNICODE_COMMON_NAME_REGEX = "^[a-zA-Z\\p{L}][a-zA-Z0-9\\-_\\p{L}]{0,63}$";
private static final String UNICODE_UNDERSCORE_COMMON_NAME_REGEX = "^[_a-zA-Z\\p{L}][a-zA-Z0-9\\-_\\p{L}]{0,63}$";
private static final String UNICODE_TABLE_NAME_REGEX = "^[a-zA-Z\\p{L}][a-zA-Z0-9\\-_\\p{L}]*$";
private static final String UNICODE_USER_NAME_REGEX = "^[a-zA-Z\\p{L}][a-zA-Z0-9.\\-_\\p{L}]*$";
private static final String UNICODE_COLUMN_NAME_REGEX
= "^[.a-zA-Z0-9_+\\-/?@#$%^&*\"\\s,:\\p{L}]{1,256}$";
private static final String UNICODE_REPOSITORY_NAME_REGEX = "^[a-zA-Z\\p{L}][a-zA-Z0-9\\-_\\p{L}]{0,255}$";
public static final String FORBIDDEN_PARTITION_NAME = "placeholder_";
public static final String TEMPORARY_TABLE_SIGN = "_#TEMP#_";
public static void checkCatalogName(String catalogName) throws AnalysisException {
if (!InternalCatalog.INTERNAL_CATALOG_NAME.equals(catalogName) && (Strings.isNullOrEmpty(catalogName)
|| !catalogName.matches(getCommonNameRegex()))) {
ErrorReport.reportAnalysisException(ErrorCode.ERR_WRONG_CATALOG_NAME, catalogName);
}
}
public static void checkDbName(String dbName) throws AnalysisException {
if (Strings.isNullOrEmpty(dbName) || !dbName.matches(getCommonNameRegex())) {
ErrorReport.reportAnalysisException(ErrorCode.ERR_WRONG_DB_NAME, dbName);
}
}
public static void checkTableName(String tableName) throws AnalysisException {
if (Strings.isNullOrEmpty(tableName)
|| !tableName.matches(getTableNameRegex())) {
ErrorReport.reportAnalysisException(ErrorCode.ERR_WRONG_TABLE_NAME, tableName,
getTableNameRegex());
}
if (tableName.length() > Config.table_name_length_limit) {
ErrorReport.reportAnalysisException(ErrorCode.ERR_TABLE_NAME_LENGTH_LIMIT, tableName,
tableName.length(), Config.table_name_length_limit);
}
// forbid table name contains sign of temporary table
if (tableName.indexOf(FeNameFormat.TEMPORARY_TABLE_SIGN) != -1) {
ErrorReport.reportAnalysisException("Incorrect table name, table name can't contains "
+ FeNameFormat.TEMPORARY_TABLE_SIGN);
}
}
public static void checkPartitionName(String partitionName) throws AnalysisException {
if (Strings.isNullOrEmpty(partitionName) || !partitionName.matches(getCommonNameRegex())) {
ErrorReport.reportAnalysisException(ErrorCode.ERR_WRONG_PARTITION_NAME, partitionName);
}
if (partitionName.startsWith(FORBIDDEN_PARTITION_NAME)) {
ErrorReport.reportAnalysisException(ErrorCode.ERR_WRONG_PARTITION_NAME, partitionName);
}
}
public static void checkColumnName(String columnName) throws AnalysisException {
if (Strings.isNullOrEmpty(columnName) || !columnName.matches(getColumnNameRegex())) {
ErrorReport.reportAnalysisException(ErrorCode.ERR_WRONG_COLUMN_NAME,
columnName, getColumnNameRegex());
}
if (columnName.startsWith(SchemaChangeHandler.SHADOW_NAME_PREFIX)) {
ErrorReport.reportAnalysisException(ErrorCode.ERR_WRONG_COLUMN_NAME,
columnName, getColumnNameRegex());
}
if (columnName.startsWith(CreateMaterializedViewStmt.MATERIALIZED_VIEW_NAME_PREFIX)
|| columnName.startsWith(CreateMaterializedViewStmt.MATERIALIZED_VIEW_AGGREGATE_NAME_PREFIX)) {
throw new AnalysisException(
"Incorrect column name " + columnName + ", column name can't start with 'mv_'/'mva_'");
}
}
public static void checkColumnCommentLength(String comment) throws AnalysisException {
if (!Strings.isNullOrEmpty(comment) && Config.column_comment_length_limit > 0
&& comment.length() > Config.column_comment_length_limit) {
throw new AnalysisException("Column comment is too long " + comment.length() + ", max length is "
+ Config.column_comment_length_limit);
}
}
public static void checkLabel(String label) throws AnalysisException {
if (Strings.isNullOrEmpty(label) || !label.matches(getLabelRegex())) {
throw new AnalysisException("Label format error. regex: " + getLabelRegex() + ", label: " + label);
}
}
public static void checkUserName(String userName) throws AnalysisException {
if (Strings.isNullOrEmpty(userName) || !userName.matches(getUserNameRegex())) {
throw new AnalysisException("invalid user name: " + userName);
}
}
public static void checkRoleName(String role, boolean canBeAdmin, String errMsg) throws AnalysisException {
if (Strings.isNullOrEmpty(role) || !role.matches(getCommonNameRegex())) {
throw new AnalysisException("invalid role format: " + role);
}
boolean res = false;
if (CaseSensibility.ROLE.getCaseSensibility()) {
res = role.equals(Role.OPERATOR_ROLE) || (!canBeAdmin && role.equals(Role.ADMIN_ROLE));
} else {
res = role.equalsIgnoreCase(Role.OPERATOR_ROLE)
|| (!canBeAdmin && role.equalsIgnoreCase(Role.ADMIN_ROLE));
}
if (res || role.startsWith(RoleManager.DEFAULT_ROLE_PREFIX)) {
throw new AnalysisException(errMsg + ": " + role);
}
}
public static void checkResourceName(String resourceName, ResourceTypeEnum type) throws AnalysisException {
if (type == ResourceTypeEnum.GENERAL) {
checkCommonName("resource", resourceName);
} else {
checkCommonName("clusterName", resourceName);
}
}
public static void checkStorageVaultName(String vaultName) throws AnalysisException {
checkCommonName("vault", vaultName);
}
public static void checkWorkloadGroupName(String workloadGroupName) throws AnalysisException {
checkCommonName("workload group", workloadGroupName);
}
public static void checkWorkloadSchedPolicyName(String policyName) throws AnalysisException {
checkCommonName("workload schedule policy", policyName);
}
public static void checkCommonName(String type, String name) throws AnalysisException {
final String regex = getCommonNameRegex();
if (Strings.isNullOrEmpty(name) || !name.matches(regex)) {
ErrorReport.reportAnalysisException(ErrorCode.ERR_WRONG_NAME_FORMAT, type, name, regex);
}
}
public static void checkOutfileSuccessFileName(String type, String name) throws AnalysisException {
final String regex = getOutfileSuccessFileNameRegex();
if (Strings.isNullOrEmpty(name) || !name.matches(regex)) {
ErrorReport.reportAnalysisException(ErrorCode.ERR_WRONG_NAME_FORMAT, type, name, regex);
}
}
public static void checkRepositoryName(String repositoryName) throws AnalysisException {
final String regex = getRepositoryNameRegex();
if (Strings.isNullOrEmpty(repositoryName) || !repositoryName.matches(regex)) {
ErrorReport.reportAnalysisException(ErrorCode.ERR_WRONG_NAME_FORMAT, "repository", repositoryName, regex);
}
}
private static boolean isEnableUnicodeNameSupport() {
boolean unicodeSupport;
if (ConnectContext.get() != null) {
unicodeSupport = ConnectContext.get().getSessionVariable().isEnableUnicodeNameSupport();
} else {
unicodeSupport = VariableMgr.getDefaultSessionVariable().isEnableUnicodeNameSupport();
}
return unicodeSupport;
}
public static String getColumnNameRegex() {
if (FeNameFormat.isEnableUnicodeNameSupport()) {
return UNICODE_COLUMN_NAME_REGEX;
} else {
return COLUMN_NAME_REGEX;
}
}
public static String getTableNameRegex() {
if (FeNameFormat.isEnableUnicodeNameSupport()) {
return UNICODE_TABLE_NAME_REGEX;
} else {
return TABLE_NAME_REGEX;
}
}
public static String getUserNameRegex() {
if (FeNameFormat.isEnableUnicodeNameSupport()) {
return UNICODE_USER_NAME_REGEX;
} else {
return USER_NAME_REGEX;
}
}
public static String getLabelRegex() {
if (FeNameFormat.isEnableUnicodeNameSupport()) {
return UNICODE_LABEL_REGEX;
} else {
return LABEL_REGEX;
}
}
public static String getCommonNameRegex() {
if (FeNameFormat.isEnableUnicodeNameSupport()) {
return UNICODE_COMMON_NAME_REGEX;
} else {
return COMMON_NAME_REGEX;
}
}
public static String getOutfileSuccessFileNameRegex() {
if (FeNameFormat.isEnableUnicodeNameSupport()) {
return UNICODE_UNDERSCORE_COMMON_NAME_REGEX;
} else {
return UNDERSCORE_COMMON_NAME_REGEX;
}
}
public static String getRepositoryNameRegex() {
if (FeNameFormat.isEnableUnicodeNameSupport()) {
return UNICODE_REPOSITORY_NAME_REGEX;
} else {
return REPOSITORY_NAME_REGEX;
}
}
}