From 157bc57cd1d7f980408b8c7fcfcd925d63eba64b Mon Sep 17 00:00:00 2001 From: jason Date: Tue, 16 Aug 2022 10:14:13 +0800 Subject: [PATCH] feature: impl shadow match value rule --- conf/config.yaml | 8 +- integration_test/config/db_tbl/config.yaml | 19 + integration_test/config/shadow/config.yaml | 129 ++++ integration_test/config/shadow/data.yaml | 30 + integration_test/config/shadow/expected.yaml | 30 + .../scene/shadow/integration_test.go | 96 +++ integration_test/scripts/shadow/init.sql | 115 +++ integration_test/scripts/shadow/sequence.sql | 31 + integration_test/scripts/shadow/sharding.sql | 668 ++++++++++++++++++ integration_test/testcase/casetest.yaml | 8 + pkg/boot/boot.go | 16 +- pkg/boot/discovery.go | 54 +- pkg/config/api.go | 1 + pkg/config/config.go | 7 + pkg/constants/const.go | 11 + pkg/proto/hint/hint.go | 2 + pkg/proto/rule/shadow.go | 175 +++++ pkg/runtime/namespace/command.go | 11 + pkg/runtime/namespace/namespace.go | 13 +- pkg/runtime/optimize/dml/select.go | 28 +- pkg/runtime/optimize/hints.go | 13 + pkg/runtime/optimize/optimizer.go | 20 +- pkg/runtime/optimize/optimizer_test.go | 16 +- pkg/runtime/optimize/sharder_test.go | 8 + pkg/runtime/runtime.go | 6 +- scripts/sharding.sql | 511 ++++++++++++++ test/integration_test.go | 32 + 27 files changed, 2028 insertions(+), 30 deletions(-) create mode 100644 integration_test/config/shadow/config.yaml create mode 100644 integration_test/config/shadow/data.yaml create mode 100644 integration_test/config/shadow/expected.yaml create mode 100644 integration_test/scene/shadow/integration_test.go create mode 100644 integration_test/scripts/shadow/init.sql create mode 100644 integration_test/scripts/shadow/sequence.sql create mode 100644 integration_test/scripts/shadow/sharding.sql create mode 100644 pkg/proto/rule/shadow.go diff --git a/conf/config.yaml b/conf/config.yaml index f5b016fe6..898009896 100644 --- a/conf/config.yaml +++ b/conf/config.yaml @@ -87,7 +87,7 @@ data: password: "123456" database: employees_0003 weight: r10w10 - - name: employees_shadow + - name: employees_show nodes: - name: node_shadow host: arana-mysql @@ -120,9 +120,9 @@ data: shadow_rule: tables: - - name: student + - name: employees.student enable: false - group_node: employees_shadow + group_node: employees_show match_rules: - operation: [insert,update] match_type: value @@ -137,4 +137,4 @@ data: - operation: [select] match_type: hint attributes: - - shadow: true + - value: "shadow" diff --git a/integration_test/config/db_tbl/config.yaml b/integration_test/config/db_tbl/config.yaml index b3d9702a2..82d5f88c8 100644 --- a/integration_test/config/db_tbl/config.yaml +++ b/integration_test/config/db_tbl/config.yaml @@ -84,6 +84,15 @@ data: password: "123456" database: employees_0003 weight: r10w10 + - name: employees_show + nodes: + - name: node_shadow + host: arana-mysql + port: 3306 + username: root + password: "123456" + database: employees_show + weight: r10w10 sharding_rule: tables: @@ -105,3 +114,13 @@ data: tbl_pattern: student_${0000..0031} attributes: sqlMaxLimit: -1 + shadow_rule: + tables: + - name: employees.student + enable: true + group_node: employees_show + match_rules: + - operation: [select] + match_type: hint + attributes: + - value: "shadow" diff --git a/integration_test/config/shadow/config.yaml b/integration_test/config/shadow/config.yaml new file mode 100644 index 000000000..53e53712f --- /dev/null +++ b/integration_test/config/shadow/config.yaml @@ -0,0 +1,129 @@ +# +# 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. +# + +kind: ConfigMap +apiVersion: "1.0" +metadata: + name: arana-config +data: + listeners: + - protocol_type: mysql + server_version: 5.7.0 + socket_address: + address: 0.0.0.0 + port: 13306 + + tenants: + - name: arana + users: + - username: root + password: "123456" + - username: arana + password: "123456" + + clusters: + - name: employees + type: mysql + sql_max_limit: -1 + tenant: arana + parameters: + max_allowed_packet: 256M + groups: + - name: employees_0000 + nodes: + - name: node0 + host: arana-mysql + port: 3306 + username: root + password: "123456" + database: employees_0000 + weight: r10w10 + parameters: + - name: node0_r_0 + host: arana-mysql + port: 3306 + username: root + password: "123456" + database: employees_0000_r + weight: r0w0 + - name: employees_0001 + nodes: + - name: node1 + host: arana-mysql + port: 3306 + username: root + password: "123456" + database: employees_0001 + weight: r10w10 + - name: employees_0002 + nodes: + - name: node2 + host: arana-mysql + port: 3306 + username: root + password: "123456" + database: employees_0002 + weight: r10w10 + - name: employees_0003 + nodes: + - name: node3 + host: arana-mysql + port: 3306 + username: root + password: "123456" + database: employees_0003 + weight: r10w10 + - name: employees_show + nodes: + - name: node_shadow + host: arana-mysql + port: 3306 + username: root + password: "123456" + database: employees_show + weight: r10w10 + sharding_rule: + tables: + - name: employees.student + allow_full_scan: true + sequence: + type: snowflake + option: + db_rules: + - column: uid + type: scriptExpr + expr: parseInt($value % 32 / 8) + tbl_rules: + - column: uid + type: scriptExpr + expr: $value % 32 + step: 32 + topology: + db_pattern: employees_${0000..0003} + tbl_pattern: student_${0000..0031} + attributes: + sqlMaxLimit: -1 + shadow_rule: + tables: + - name: employees.student + enable: true + group_node: employees_show + match_rules: + - operation: [select] + match_type: hint + attributes: + - value: "shadow" diff --git a/integration_test/config/shadow/data.yaml b/integration_test/config/shadow/data.yaml new file mode 100644 index 000000000..112cb3c18 --- /dev/null +++ b/integration_test/config/shadow/data.yaml @@ -0,0 +1,30 @@ +# +# 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. +# + +kind: DataSet +metadata: + tables: + - name: "order" + columns: + - name: "name" + type: "string" + - name: "value" + type: "string" +data: + - name: "order" + value: + - ["test", "test1"] diff --git a/integration_test/config/shadow/expected.yaml b/integration_test/config/shadow/expected.yaml new file mode 100644 index 000000000..542d37822 --- /dev/null +++ b/integration_test/config/shadow/expected.yaml @@ -0,0 +1,30 @@ +# +# 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. +# + +kind: excepted +metadata: + tables: + - name: "sequence" + columns: + - name: "name" + type: "string" + - name: "value" + type: "string" +data: + - name: "sequence" + value: + - ["1", "2"] diff --git a/integration_test/scene/shadow/integration_test.go b/integration_test/scene/shadow/integration_test.go new file mode 100644 index 000000000..a17c74723 --- /dev/null +++ b/integration_test/scene/shadow/integration_test.go @@ -0,0 +1,96 @@ +/* + * 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 test + +import ( + "strings" + "testing" +) + +import ( + _ "github.com/go-sql-driver/mysql" // register mysql + + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/suite" +) + +import ( + "github.com/arana-db/arana/test" +) + +type IntegrationSuite struct { + *test.MySuite +} + +func TestSuite(t *testing.T) { + su := test.NewMySuite( + test.WithMySQLServerAuth("root", "123456"), + test.WithMySQLDatabase("employees"), + test.WithConfig("../integration_test/config/shadow/config.yaml"), + test.WithScriptPath("../integration_test/scripts/shadow"), + test.WithTestCasePath("../../testcase/casetest.yaml"), + // WithDevMode(), // NOTICE: UNCOMMENT IF YOU WANT TO DEBUG LOCAL ARANA SERVER!!! + ) + suite.Run(t, &IntegrationSuite{su}) +} + +func (s *IntegrationSuite) TestShadowScene() { + var ( + db = s.DB() + t = s.T() + ) + tx, err := db.Begin() + assert.NoError(t, err, "should begin a new tx") + + cases := s.TestCases() + for _, sqlCase := range cases.ExecCases { + for _, sense := range sqlCase.Sense { + if strings.Compare(strings.TrimSpace(sense), "shadow") == 1 { + params := strings.Split(sqlCase.Parameters, ",") + args := make([]interface{}, 0, len(params)) + for _, param := range params { + k, _ := test.GetValueByType(param) + args = append(args, k) + } + + // Execute sql + result, err := tx.Exec(sqlCase.SQL, args...) + assert.NoError(t, err, "exec not right") + err = sqlCase.ExpectedResult.CompareRow(result) + assert.NoError(t, err, err) + } + } + } + + for _, sqlCase := range cases.QueryRowCases { + for _, sense := range sqlCase.Sense { + if strings.Compare(strings.TrimSpace(sense), "shadow") == 1 { + params := strings.Split(sqlCase.Parameters, ",") + args := make([]interface{}, 0, len(params)) + for _, param := range params { + k, _ := test.GetValueByType(param) + args = append(args, k) + } + + result := tx.QueryRow(sqlCase.SQL, args...) + err = sqlCase.ExpectedResult.CompareRow(result) + assert.NoError(t, err, err) + } + } + } +} diff --git a/integration_test/scripts/shadow/init.sql b/integration_test/scripts/shadow/init.sql new file mode 100644 index 000000000..d40ed6991 --- /dev/null +++ b/integration_test/scripts/shadow/init.sql @@ -0,0 +1,115 @@ +-- +-- 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. +-- + +-- Sample employee database +-- See changelog table for details +-- Copyright (C) 2007,2008, MySQL AB +-- +-- Original data created by Fusheng Wang and Carlo Zaniolo +-- http://www.cs.aau.dk/TimeCenter/software.htm +-- http://www.cs.aau.dk/TimeCenter/Data/employeeTemporalDataSet.zip +-- +-- Current schema by Giuseppe Maxia +-- Data conversion from XML to relational by Patrick Crews +-- +-- This work is licensed under the +-- Creative Commons Attribution-Share Alike 3.0 Unported License. +-- To view a copy of this license, visit +-- http://creativecommons.org/licenses/by-sa/3.0/ or send a letter to +-- Creative Commons, 171 Second Street, Suite 300, San Francisco, +-- California, 94105, USA. +-- +-- DISCLAIMER +-- To the best of our knowledge, this data is fabricated, and +-- it does not correspond to real people. +-- Any similarity to existing people is purely coincidental. +-- + +CREATE DATABASE IF NOT EXISTS employees_0000 CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci; +CREATE DATABASE IF NOT EXISTS employees_show CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci; + +USE employees_0000; + +SELECT 'CREATING DATABASE STRUCTURE' as 'INFO'; + +DROP TABLE IF EXISTS dept_emp, + dept_manager, + titles, + salaries, + employees, + departments; + +/*!50503 set default_storage_engine = InnoDB */; +/*!50503 select CONCAT('storage engine: ', @@default_storage_engine) as INFO */; + +CREATE TABLE employees ( + emp_no INT NOT NULL, + birth_date DATE NOT NULL, + first_name VARCHAR(14) NOT NULL, + last_name VARCHAR(16) NOT NULL, + gender ENUM ('M','F') NOT NULL, + hire_date DATE NOT NULL, + PRIMARY KEY (emp_no) +); + +CREATE TABLE departments ( + dept_no CHAR(4) NOT NULL, + dept_name VARCHAR(40) NOT NULL, + PRIMARY KEY (dept_no), + UNIQUE KEY (dept_name) +); + +CREATE TABLE dept_manager ( + emp_no INT NOT NULL, + dept_no CHAR(4) NOT NULL, + from_date DATE NOT NULL, + to_date DATE NOT NULL, + FOREIGN KEY (emp_no) REFERENCES employees (emp_no) ON DELETE CASCADE, + FOREIGN KEY (dept_no) REFERENCES departments (dept_no) ON DELETE CASCADE, + PRIMARY KEY (emp_no,dept_no) +); + +CREATE TABLE dept_emp ( + emp_no INT NOT NULL, + dept_no CHAR(4) NOT NULL, + from_date DATE NOT NULL, + to_date DATE NOT NULL, + FOREIGN KEY (emp_no) REFERENCES employees (emp_no) ON DELETE CASCADE, + FOREIGN KEY (dept_no) REFERENCES departments (dept_no) ON DELETE CASCADE, + PRIMARY KEY (emp_no,dept_no) +); + +CREATE TABLE titles ( + emp_no INT NOT NULL, + title VARCHAR(50) NOT NULL, + from_date DATE NOT NULL, + to_date DATE, + FOREIGN KEY (emp_no) REFERENCES employees (emp_no) ON DELETE CASCADE, + PRIMARY KEY (emp_no,title, from_date) +) +; + +CREATE TABLE salaries ( + emp_no INT NOT NULL, + salary INT NOT NULL, + from_date DATE NOT NULL, + to_date DATE NOT NULL, + FOREIGN KEY (emp_no) REFERENCES employees (emp_no) ON DELETE CASCADE, + PRIMARY KEY (emp_no, from_date), + KEY `from_date` (`from_date`) +) +; diff --git a/integration_test/scripts/shadow/sequence.sql b/integration_test/scripts/shadow/sequence.sql new file mode 100644 index 000000000..0df0add1c --- /dev/null +++ b/integration_test/scripts/shadow/sequence.sql @@ -0,0 +1,31 @@ +-- +-- 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. +-- + +CREATE DATABASE IF NOT EXISTS employees_0000 CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci; + +CREATE TABLE IF NOT EXISTS `employees_0000`.`sequence` +( + `id` BIGINT UNSIGNED NOT NULL AUTO_INCREMENT, + `name` VARCHAR(64) NOT NULL, + `value` BIGINT NOT NULL, + `step` INT NOT NULL DEFAULT 10000, + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_name` (`name`) +) ENGINE = InnoDB + DEFAULT CHARSET = utf8mb4; diff --git a/integration_test/scripts/shadow/sharding.sql b/integration_test/scripts/shadow/sharding.sql new file mode 100644 index 000000000..8a3c12cab --- /dev/null +++ b/integration_test/scripts/shadow/sharding.sql @@ -0,0 +1,668 @@ +-- +-- 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. +-- + +CREATE DATABASE IF NOT EXISTS employees_0000 CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci; +CREATE DATABASE IF NOT EXISTS employees_0001 CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci; +CREATE DATABASE IF NOT EXISTS employees_0002 CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci; +CREATE DATABASE IF NOT EXISTS employees_0003 CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci; +CREATE DATABASE IF NOT EXISTS employees_show CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci; + +CREATE DATABASE IF NOT EXISTS employees_0000_r CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci; + +CREATE TABLE IF NOT EXISTS `employees_0000`.`student_0000` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0000`.`student_0001` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0000`.`student_0002` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0000`.`student_0003` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0000`.`student_0004` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0000`.`student_0005` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0000`.`student_0006` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0000`.`student_0007` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0001`.`student_0008` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0001`.`student_0009` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0001`.`student_0010` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0001`.`student_0011` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0001`.`student_0012` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0001`.`student_0013` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0001`.`student_0014` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0001`.`student_0015` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0002`.`student_0016` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0002`.`student_0017` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0002`.`student_0018` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0002`.`student_0019` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0002`.`student_0020` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0002`.`student_0021` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0002`.`student_0022` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0002`.`student_0023` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0003`.`student_0024` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0003`.`student_0025` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0003`.`student_0026` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0003`.`student_0027` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0003`.`student_0028` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0003`.`student_0029` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0003`.`student_0030` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0003`.`student_0031` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + + +CREATE TABLE IF NOT EXISTS `employees_0000_r`.`student_0000` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0000_r`.`student_0001` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0000_r`.`student_0002` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0000_r`.`student_0003` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0000_r`.`student_0004` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0000_r`.`student_0005` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0000_r`.`student_0006` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_0000_r`.`student_0007` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + + +INSERT INTO employees_0000.student_0001(id,uid,name,score,nickname,gender,birth_year,created_at,modified_at) VALUES (1, 1, 'arana', 95, 'Awesome Arana', 0, 2021, NOW(), NOW()); diff --git a/integration_test/testcase/casetest.yaml b/integration_test/testcase/casetest.yaml index 0324d827a..d91621364 100644 --- a/integration_test/testcase/casetest.yaml +++ b/integration_test/testcase/casetest.yaml @@ -25,6 +25,14 @@ query_rows_cases: expected: type: "file" value: "../../config/db/expected.yaml" + - sql: "/*A! shadow(shadow) */ SELECT name FROM student WHERE name= ?" + parameters: "1:string" + sense: + - shadow + expected: + type: "file" + value: "../../config/db/expected.yaml" + query_row_cases: - sql: "SELECT COUNT(1) FROM sequence WHERE name=?" parameters: "1:string" diff --git a/pkg/boot/boot.go b/pkg/boot/boot.go index 683518f55..f07d7a6f6 100644 --- a/pkg/boot/boot.go +++ b/pkg/boot/boot.go @@ -145,7 +145,21 @@ func buildNamespace(ctx context.Context, provider Discovery, clusterName string) } ru.SetVTable(table, vt) } - initCmds = append(initCmds, namespace.UpdateRule(&ru)) + + var su = rule.NewShadowRule() + for _, table := range tables { + var ruleManager rule.ShadowRuleManager + if ruleManager, err = provider.GetShadowOperation(ctx, clusterName, table); err != nil { + return nil, err + } + if ruleManager == nil { + log.Warnf("no such shadow rule %s", table) + continue + } + su.SetRuleManager(table, ruleManager) + } + + initCmds = append(initCmds, namespace.UpdateRule(&ru), namespace.UpdateShadowRule(su)) return namespace.New(clusterName, initCmds...) } diff --git a/pkg/boot/discovery.go b/pkg/boot/discovery.go index a3a0e9d08..ef5956320 100644 --- a/pkg/boot/discovery.go +++ b/pkg/boot/discovery.go @@ -105,6 +105,8 @@ type Discovery interface { ListTables(ctx context.Context, cluster string) ([]string, error) // GetTable returns the table info. GetTable(ctx context.Context, cluster, table string) (*rule.VTable, error) + // GetShadowOperation return the shadow info. + GetShadowOperation(ctx context.Context, cluster, table string) (rule.ShadowRuleManager, error) // GetConfigCenter GetConfigCenter() *config.Center @@ -116,6 +118,45 @@ type discovery struct { c *config.Center } +func (fp *discovery) GetShadowOperation(ctx context.Context, cluster, table string) (rule.ShadowRuleManager, error) { + cfg, err := fp.c.Load() + if err != nil { + return nil, err + } + + if cfg.Data.ShadowRule == nil { + return nil, nil + } + + for _, tbl := range cfg.Data.ShadowRule.ShadowTables { + db, tb, err := parseTable(tbl.Name) + if err != nil { + log.Warnf("skip parsing table rule: %v", err) + continue + } + if db != cluster { + continue + } + if tb != table { + continue + } + + operations := make(map[string][]*rule.Attribute) + for _, matchRule := range tbl.MatchRules { + attributes := make([]*rule.Attribute, 0, len(matchRule.Attributes)) + for _, attr := range matchRule.Attributes { + attributes = append(attributes, rule.NewAttribute(attr.Column, attr.Value, matchRule.MatchType)) + } + for _, operation := range matchRule.Operation { + operations[operation] = attributes + } + } + return rule.NewRuleManager(operations, tbl.Enable, tbl.GroupNode), nil + } + + return nil, nil +} + func (fp *discovery) Init(ctx context.Context) error { if err := fp.loadBootOptions(); err != nil { return err @@ -285,12 +326,21 @@ func (fp *discovery) ListTables(ctx context.Context, cluster string) ([]string, return tables, nil } -func (fp *discovery) GetNode(ctx context.Context, cluster, group, node string) (*config.Node, error) { +func (fp *discovery) GetNodes(ctx context.Context, cluster, group string) ([]*config.Node, error) { bingo, ok := fp.loadGroup(cluster, group) if !ok { return nil, nil } - for _, it := range bingo.Nodes { + + return bingo.Nodes, nil +} + +func (fp *discovery) GetNode(ctx context.Context, cluster, group, node string) (*config.Node, error) { + nodes, err := fp.GetNodes(ctx, cluster, group) + if err != nil { + return nil, err + } + for _, it := range nodes { if it.Name == node { return it, nil } diff --git a/pkg/config/api.go b/pkg/config/api.go index 08105ac0f..dd73b47a0 100644 --- a/pkg/config/api.go +++ b/pkg/config/api.go @@ -39,6 +39,7 @@ const ( DefaultConfigDataSourceClustersPath PathKey = "/arana-db/config/data/dataSourceClusters" DefaultConfigDataShardingRulePath PathKey = "/arana-db/config/data/shardingRule" DefaultConfigDataTenantsPath PathKey = "/arana-db/config/data/tenants" + DefaultConfigDataShadowRulePath PathKey = "/arana-db/config/data/shadowRule" ) const ( diff --git a/pkg/config/config.go b/pkg/config/config.go index e1343975b..8aa228c4e 100644 --- a/pkg/config/config.go +++ b/pkg/config/config.go @@ -48,6 +48,7 @@ var ( DefaultConfigDataListenersPath: "data.listeners", DefaultConfigDataSourceClustersPath: "data.clusters", DefaultConfigDataShardingRulePath: "data.sharding_rule", + DefaultConfigDataShadowRulePath: "data.shadow_rule", } _configValSupplier map[PathKey]func(cfg *Configuration) interface{} = map[PathKey]func(cfg *Configuration) interface{}{ @@ -69,6 +70,9 @@ var ( DefaultConfigDataShardingRulePath: func(cfg *Configuration) interface{} { return &cfg.Data.ShardingRule }, + DefaultConfigDataShadowRulePath: func(cfg *Configuration) interface{} { + return &cfg.Data.ShadowRule + }, } ) @@ -163,6 +167,9 @@ func (c *Center) loadFromStore(ctx context.Context) (*Configuration, error) { Tenants: make([]*Tenant, 0), DataSourceClusters: make([]*DataSourceCluster, 0), ShardingRule: &ShardingRule{}, + ShadowRule: &ShadowRule{ + ShadowTables: make([]*ShadowTable, 0), + }, }, } diff --git a/pkg/constants/const.go b/pkg/constants/const.go index caeb5029e..e4695f1b8 100644 --- a/pkg/constants/const.go +++ b/pkg/constants/const.go @@ -27,3 +27,14 @@ const ( VariableNameMaxAllowedPacket = "max_allowed_packet" ) + +const ( + ShadowMatchRegex = "regex" + ShadowMatchValue = "value" + ShadowMatchHint = "hint" + + ShadowInsert = "insert" + ShadowSelect = "select" + ShadowUpdate = "update" + ShadowDelete = "delete" +) diff --git a/pkg/proto/hint/hint.go b/pkg/proto/hint/hint.go index 2b6cb905a..d15ddfdf3 100644 --- a/pkg/proto/hint/hint.go +++ b/pkg/proto/hint/hint.go @@ -38,6 +38,7 @@ const ( TypeRoute // custom route TypeFullScan // enable full-scan TypeDirect // direct route + TypeShadow // shadow route ) var _hintTypes = [...]string{ @@ -46,6 +47,7 @@ var _hintTypes = [...]string{ TypeRoute: "ROUTE", TypeFullScan: "FULLSCAN", TypeDirect: "DIRECT", + TypeShadow: "SHADOW", } // KeyValue represents a pair of key and value. diff --git a/pkg/proto/rule/shadow.go b/pkg/proto/rule/shadow.go new file mode 100644 index 000000000..37eb7fd10 --- /dev/null +++ b/pkg/proto/rule/shadow.go @@ -0,0 +1,175 @@ +/* + * 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 rule + +import ( + "sync" +) + +import ( + "github.com/arana-db/arana/pkg/constants" +) + +type ( + // ShadowRule represents the shadow of databases and tables. + ShadowRule struct { + mu sync.RWMutex + rules map[string]ShadowRuleManager // map[table]rule + } + + Operation struct { + enable bool + database string + actions map[string][]*Attribute // map[action][]*Attribute, action in (select, update, delete, update) + } + + Attribute struct { + column string + value string + typ string // regex, value, hint + } +) + +func (o *Operation) GetDatabase() string { + return o.database +} + +type ShadowRuleManager interface { + MatchValueBy(action, column, value string) bool + MatchHintBy(action, hint string) bool + MatchRegexBy(action, column, value string) bool + GetDatabase() string +} + +func (s *ShadowRule) MatchValueBy(tableName, action, column, value string) bool { + s.mu.RLock() + defer s.mu.RUnlock() + rule, ok := s.rules[tableName] + if !ok { + return false + } + + return rule.MatchValueBy(action, column, value) +} + +func (s *ShadowRule) MatchHintBy(tableName, action, hint string) bool { + s.mu.RLock() + defer s.mu.RUnlock() + rule, ok := s.rules[tableName] + if !ok { + return false + } + return rule.MatchHintBy(action, hint) +} + +func (s *ShadowRule) MatchRegexBy(tableName, action, column, value string) bool { + s.mu.RLock() + defer s.mu.RUnlock() + + rule, ok := s.rules[tableName] + if !ok { + return false + } + return rule.MatchRegexBy(action, column, value) +} + +func (s *ShadowRule) GetDatabase(tableName string) string { + s.mu.RLock() + defer s.mu.RUnlock() + + return s.rules[tableName].GetDatabase() +} + +func (s *ShadowRule) SetRuleManager(tableName string, ruleManager ShadowRuleManager) { + s.mu.Lock() + if s.rules == nil { + s.rules = make(map[string]ShadowRuleManager, 10) + } + s.rules[tableName] = ruleManager + s.mu.Unlock() +} + +func NewRuleManager(actions map[string][]*Attribute, enable bool, database string) ShadowRuleManager { + return &Operation{ + actions: actions, + database: database, + enable: enable, + } +} + +func (o *Operation) MatchValueBy(action, column, value string) bool { + if !o.enable { + return false + } + attrs, ok := o.actions[action] + if !ok { + return false + } + + for _, attr := range attrs { + if attr.typ == constants.ShadowMatchValue { + if attr.column == column && attr.value == value { + return true + } + } + } + return false +} + +func (o *Operation) MatchHintBy(action, hint string) bool { + if !o.enable { + return false + } + attrs, ok := o.actions[action] + if !ok { + return false + } + + for _, attr := range attrs { + if attr.typ == constants.ShadowMatchHint { + return attr.value == hint + } + } + return false +} + +// MatchRegexBy . +// TODO impl match regex rule +func (o *Operation) MatchRegexBy(action, column, value string) bool { + if !o.enable { + return false + } + _, ok := o.actions[action] + if !ok { + return false + } + + return false +} + +func NewAttribute(col, val, typ string) *Attribute { + return &Attribute{ + column: col, + typ: typ, + value: val, + } +} + +func NewShadowRule() *ShadowRule { + return &ShadowRule{rules: make(map[string]ShadowRuleManager, 0)} +} diff --git a/pkg/runtime/namespace/command.go b/pkg/runtime/namespace/command.go index 54a9209b4..98795fe58 100644 --- a/pkg/runtime/namespace/command.go +++ b/pkg/runtime/namespace/command.go @@ -154,3 +154,14 @@ func UpdateRule(rule *rule.Rule) Command { return nil } } + +// UpdateShadowRule updates the shadow rule. +func UpdateShadowRule(rule *rule.ShadowRule) Command { + return func(ns *Namespace) error { + ns.Lock() + defer ns.Unlock() + ns.shadowRule.Store(rule) + + return nil + } +} diff --git a/pkg/runtime/namespace/namespace.go b/pkg/runtime/namespace/namespace.go index 799fcaeb2..e5a769944 100644 --- a/pkg/runtime/namespace/namespace.go +++ b/pkg/runtime/namespace/namespace.go @@ -76,7 +76,8 @@ type ( name string // the name of Namespace - rule atomic.Value // *rule.Rule + rule atomic.Value // *rule.Rule + shadowRule atomic.Value // *rule.ShadowRule // datasource map, eg: employee_0001 -> [mysql-a,mysql-b,mysql-c], ... employee_0007 -> [mysql-x,mysql-y,mysql-z] dss atomic.Value // map[string][]proto.DB @@ -98,6 +99,7 @@ func New(name string, commands ...Command) (*Namespace, error) { } ns.dss.Store(make(map[string][]proto.DB)) // init empty map ns.rule.Store(&rule.Rule{}) // init empty rule + ns.shadowRule.Store(&rule.ShadowRule{}) // init empty shadowRule for _, cmd := range commands { if err := cmd(ns); err != nil { @@ -224,6 +226,15 @@ func (ns *Namespace) Rule() *rule.Rule { return ru } +// ShadowRule returns the shadow rule. +func (ns *Namespace) ShadowRule() *rule.ShadowRule { + su, ok := ns.shadowRule.Load().(*rule.ShadowRule) + if !ok { + return nil + } + return su +} + // EnqueueCommand enqueues the next command, it will be executed async. func (ns *Namespace) EnqueueCommand(cmd Command) error { if ns.closed.Load() { diff --git a/pkg/runtime/optimize/dml/select.go b/pkg/runtime/optimize/dml/select.go index 17757a80c..18f657e17 100644 --- a/pkg/runtime/optimize/dml/select.go +++ b/pkg/runtime/optimize/dml/select.go @@ -19,6 +19,7 @@ package dml import ( "context" + "github.com/arana-db/arana/pkg/constants" "strings" ) @@ -78,16 +79,24 @@ func optimizeSelect(ctx context.Context, o *optimize.Optimizer) (proto.Plan, err // --- SIMPLE QUERY BEGIN --- var ( - shards rule.DatabaseTables - fullScan bool - err error - vt = o.Rule.MustVTable(stmt.From[0].TableName().Suffix()) - tableName = stmt.From[0].TableName() + shards rule.DatabaseTables + fullScan bool + matchShadow bool + err error + vt = o.Rule.MustVTable(stmt.From[0].TableName().Suffix()) + tableName = stmt.From[0].TableName() ) + if len(o.Hints) > 0 { if shards, err = optimize.Hints(tableName, o.Hints, o.Rule); err != nil { return nil, errors.Wrap(err, "calculate hints failed") } + + for _, shadowHint := range optimize.ShadowHints(o.Hints) { + if !matchShadow { + matchShadow = o.ShadowRule.MatchHintBy(tableName.Suffix(), constants.ShadowSelect, shadowHint) + } + } } if shards == nil { @@ -129,6 +138,9 @@ func optimizeSelect(ctx context.Context, o *optimize.Optimizer) (proto.Plan, err return nil, errors.Errorf("cannot compute minimal topology from '%s'", stmt.From[0].TableName().Suffix()) } + if matchShadow { + db0 = o.ShadowRule.GetDatabase(tableName.Suffix()) + } return toSingle(db0, tbl0) } @@ -137,6 +149,9 @@ func optimizeSelect(ctx context.Context, o *optimize.Optimizer) (proto.Plan, err var db, tbl string for k, v := range shards { db = k + if matchShadow { + db = o.ShadowRule.GetDatabase(tableName.Suffix()) + } tbl = v[0] } return toSingle(db, tbl) @@ -164,6 +179,9 @@ func optimizeSelect(ctx context.Context, o *optimize.Optimizer) (proto.Plan, err plans := make([]proto.Plan, 0, len(shards)) for k, v := range shards { + if matchShadow { + k = o.ShadowRule.GetDatabase(tableName.Suffix()) + } next := &dml.SimpleQueryPlan{ Database: k, Tables: v, diff --git a/pkg/runtime/optimize/hints.go b/pkg/runtime/optimize/hints.go index f24c4ca6c..96c90e297 100644 --- a/pkg/runtime/optimize/hints.go +++ b/pkg/runtime/optimize/hints.go @@ -100,6 +100,19 @@ func Hints(tableName ast.TableName, hints []*hint.Hint, rule *rule.Rule) (hintTa return } +func ShadowHints(hints []*hint.Hint) []string { + res := make([]string, 0) + for _, h := range hints { + if h.Type != hint.TypeShadow { + continue + } + for _, i := range h.Inputs { + res = append(res, i.V) + } + } + return res +} + // Direct force forward to db[0] type Direct struct{} diff --git a/pkg/runtime/optimize/optimizer.go b/pkg/runtime/optimize/optimizer.go index 5147eea7a..64140e734 100644 --- a/pkg/runtime/optimize/optimizer.go +++ b/pkg/runtime/optimize/optimizer.go @@ -74,13 +74,14 @@ func Register(t rast.SQLType, h Processor) { type Processor = func(ctx context.Context, o *Optimizer) (proto.Plan, error) type Optimizer struct { - Rule *rule.Rule - Hints []*hint.Hint - Stmt rast.Statement - Args []interface{} + Rule *rule.Rule + ShadowRule *rule.ShadowRule + Hints []*hint.Hint + Stmt rast.Statement + Args []interface{} } -func NewOptimizer(rule *rule.Rule, hints []*hint.Hint, stmt ast.StmtNode, args []interface{}) (proto.Optimizer, error) { +func NewOptimizer(rule *rule.Rule, su *rule.ShadowRule, hints []*hint.Hint, stmt ast.StmtNode, args []interface{}) (proto.Optimizer, error) { var ( rstmt rast.Statement err error @@ -90,10 +91,11 @@ func NewOptimizer(rule *rule.Rule, hints []*hint.Hint, stmt ast.StmtNode, args [ } return &Optimizer{ - Rule: rule, - Hints: hints, - Stmt: rstmt, - Args: args, + Rule: rule, + ShadowRule: su, + Hints: hints, + Stmt: rstmt, + Args: args, }, nil } diff --git a/pkg/runtime/optimize/optimizer_test.go b/pkg/runtime/optimize/optimizer_test.go index 2dd254707..5e11dc4ae 100644 --- a/pkg/runtime/optimize/optimizer_test.go +++ b/pkg/runtime/optimize/optimizer_test.go @@ -65,11 +65,12 @@ func TestOptimizer_OptimizeSelect(t *testing.T) { sql = "select id, uid from student where uid in (?,?,?)" ctx = context.Background() ru = makeFakeRule(ctrl, 8) + su = makeFakeShadowRule(ctrl) ) p := parser.New() stmt, _ := p.ParseOneStmt(sql, "", "") - opt, err := NewOptimizer(ru, nil, stmt, []interface{}{1, 2, 3}) + opt, err := NewOptimizer(ru, su, nil, stmt, []interface{}{1, 2, 3}) assert.NoError(t, err) plan, err := opt.Optimize(ctx) assert.NoError(t, err) @@ -140,6 +141,7 @@ func TestOptimizer_OptimizeInsert(t *testing.T) { var ( ctx = context.Background() ru = makeFakeRule(ctrl, 8) + su = makeFakeShadowRule(ctrl) ) t.Run("sharding", func(t *testing.T) { @@ -148,7 +150,7 @@ func TestOptimizer_OptimizeInsert(t *testing.T) { p := parser.New() stmt, _ := p.ParseOneStmt(sql, "", "") - opt, err := NewOptimizer(ru, nil, stmt, []interface{}{8, 9, 16}) + opt, err := NewOptimizer(ru, su, nil, stmt, []interface{}{8, 9, 16}) assert.NoError(t, err) plan, err := opt.Optimize(ctx) // 8,16 -> fake_db_0000, 9 -> fake_db_0001 @@ -169,7 +171,7 @@ func TestOptimizer_OptimizeInsert(t *testing.T) { p := parser.New() stmt, _ := p.ParseOneStmt(sql, "", "") - opt, err := NewOptimizer(ru, nil, stmt, []interface{}{1}) + opt, err := NewOptimizer(ru, su, nil, stmt, []interface{}{1}) assert.NoError(t, err) plan, err := opt.Optimize(ctx) @@ -200,6 +202,7 @@ func TestOptimizer_OptimizeAlterTable(t *testing.T) { var ( ctx = context.Background() ru rule.Rule + su rule.ShadowRule tab rule.VTable topology rule.Topology ) @@ -224,7 +227,7 @@ func TestOptimizer_OptimizeAlterTable(t *testing.T) { p := parser.New() stmt, _ := p.ParseOneStmt(sql, "", "") - opt, err := NewOptimizer(&ru, nil, stmt, nil) + opt, err := NewOptimizer(&ru, &su, nil, stmt, nil) assert.NoError(t, err) plan, err := opt.Optimize(ctx) @@ -240,7 +243,7 @@ func TestOptimizer_OptimizeAlterTable(t *testing.T) { p := parser.New() stmt, _ := p.ParseOneStmt(sql, "", "") - opt, err := NewOptimizer(&ru, nil, stmt, nil) + opt, err := NewOptimizer(&ru, &su, nil, stmt, nil) assert.NoError(t, err) plan, err := opt.Optimize(ctx) @@ -273,6 +276,7 @@ func TestOptimizer_OptimizeInsertSelect(t *testing.T) { var ( ctx = context.Background() ru rule.Rule + su rule.ShadowRule ) ru.SetVTable("student", nil) @@ -283,7 +287,7 @@ func TestOptimizer_OptimizeInsertSelect(t *testing.T) { p := parser.New() stmt, _ := p.ParseOneStmt(sql, "", "") - opt, err := NewOptimizer(&ru, nil, stmt, []interface{}{1}) + opt, err := NewOptimizer(&ru, &su, nil, stmt, []interface{}{1}) assert.NoError(t, err) plan, err := opt.Optimize(ctx) diff --git a/pkg/runtime/optimize/sharder_test.go b/pkg/runtime/optimize/sharder_test.go index 5cac09bda..28219b660 100644 --- a/pkg/runtime/optimize/sharder_test.go +++ b/pkg/runtime/optimize/sharder_test.go @@ -124,3 +124,11 @@ func makeFakeRule(c *gomock.Controller, mod int) *rule.Rule { ru.SetVTable("student", &tab) return &ru } + +func makeFakeShadowRule(c *gomock.Controller) *rule.ShadowRule { + var ( + ru rule.ShadowRule + ) + + return &ru +} diff --git a/pkg/runtime/runtime.go b/pkg/runtime/runtime.go index 44b0f6926..815919c25 100644 --- a/pkg/runtime/runtime.go +++ b/pkg/runtime/runtime.go @@ -218,6 +218,7 @@ func (tx *compositeTx) Execute(ctx *proto.Context) (res proto.Result, warn uint1 var ( ru = tx.rt.Namespace().Rule() + su = tx.rt.Namespace().ShadowRule() plan proto.Plan c = ctx.Context ) @@ -226,7 +227,7 @@ func (tx *compositeTx) Execute(ctx *proto.Context) (res proto.Result, warn uint1 c = rcontext.WithHints(c, ctx.Stmt.Hints) var opt proto.Optimizer - if opt, err = optimize.NewOptimizer(ru, ctx.Stmt.Hints, ctx.Stmt.StmtNode, args); err != nil { + if opt, err = optimize.NewOptimizer(ru, su, ctx.Stmt.Hints, ctx.Stmt.StmtNode, args); err != nil { err = perrors.WithStack(err) return } @@ -628,6 +629,7 @@ func (pi *defaultRuntime) Execute(ctx *proto.Context) (res proto.Result, warn ui var ( ru = pi.Namespace().Rule() + su = pi.Namespace().ShadowRule() plan proto.Plan c = ctx.Context ) @@ -640,7 +642,7 @@ func (pi *defaultRuntime) Execute(ctx *proto.Context) (res proto.Result, warn ui start := time.Now() var opt proto.Optimizer - if opt, err = optimize.NewOptimizer(ru, ctx.Stmt.Hints, ctx.Stmt.StmtNode, args); err != nil { + if opt, err = optimize.NewOptimizer(ru, su, ctx.Stmt.Hints, ctx.Stmt.StmtNode, args); err != nil { err = perrors.WithStack(err) return } diff --git a/scripts/sharding.sql b/scripts/sharding.sql index d36edda0d..93c11604f 100644 --- a/scripts/sharding.sql +++ b/scripts/sharding.sql @@ -663,5 +663,516 @@ CREATE TABLE IF NOT EXISTS `employees_0000_r`.`student_0007` KEY `nickname` (`nickname`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0000` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0001` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0002` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0003` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0004` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0005` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0006` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0007` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0008` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0009` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0010` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0011` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0012` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0013` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0014` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0015` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0016` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0017` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0018` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0019` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0020` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0021` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0022` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0023` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0024` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0025` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0026` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0027` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0028` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0029` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0030` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; + +CREATE TABLE IF NOT EXISTS `employees_show`.`student_0031` +( + `id` BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, + `uid` BIGINT(20) UNSIGNED NOT NULL, + `name` VARCHAR(255) NOT NULL, + `score` DECIMAL(6,2) DEFAULT '0', + `nickname` VARCHAR(255) DEFAULT NULL, + `gender` TINYINT(4) NULL, + `birth_year` SMALLINT(5) UNSIGNED DEFAULT '0', + `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, + `modified_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, + PRIMARY KEY (`id`), + UNIQUE KEY `uk_uid` (`uid`), + KEY `nickname` (`nickname`) + ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; INSERT INTO employees_0000.student_0001(id,uid,name,score,nickname,gender,birth_year,created_at,modified_at) VALUES (1, 1, 'arana', 95, 'Awesome Arana', 0, 2021, NOW(), NOW()); diff --git a/test/integration_test.go b/test/integration_test.go index dc0b3e9aa..504c33177 100644 --- a/test/integration_test.go +++ b/test/integration_test.go @@ -911,3 +911,35 @@ func (s *IntegrationSuite) TestShowTableStatus() { }) } } + +func (s *IntegrationSuite) TestHintsShadow() { + var ( + db = s.DB() + t = s.T() + ) + type tt struct { + sqlHint string + sql string + same bool + } + + for _, it := range []tt{ + { + "/*A! shadow(shadow) */ SELECT * FROM student where uid=1", + "SELECT * FROM student where uid=1", false, + }, + } { + t.Run(it.sql, func(t *testing.T) { + rows, err := db.Query(it.sqlHint) + assert.NoError(t, err, "should query from shadow table successfully") + defer rows.Close() + data, _ := utils.PrintTable(rows) + + rows, err = db.Query(it.sql) + assert.NoError(t, err, "should query from table successfully") + defer rows.Close() + data2, _ := utils.PrintTable(rows) + assert.Equal(t, it.same, len(data) == len(data2)) + }) + } +}