Commit ef8f8b1f authored by Sergei Golubchik's avatar Sergei Golubchik

find_files(): don't sort files in my_dir(), sort table names

after all engines have discovered their tables
  
side effect: correct alphabetical sorting as in ORDER BY ... COLLATE utf8_bin,
information_schema is no longer the first after find_files(),
tables like #mysql50#zzz are sorted first (as per table name),
not last (as per file name zzz).
parent c8ba9e10
...@@ -142,9 +142,9 @@ create table `#mysql50#aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa ...@@ -142,9 +142,9 @@ create table `#mysql50#aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
ERROR 42000: Incorrect table name '#mysql50#aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa12345' ERROR 42000: Incorrect table name '#mysql50#aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa12345'
show tables; show tables;
Tables_in_mysqltestbug26703 Tables_in_mysqltestbug26703
#mysql50#abc`def
aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa1234 aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa1234
aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
#mysql50#abc`def
use test; use test;
drop database mysqltestbug26703; drop database mysqltestbug26703;
End of 5.1 tests End of 5.1 tests
......
...@@ -176,15 +176,15 @@ t1 ...@@ -176,15 +176,15 @@ t1
v1 v1
show tables; show tables;
Tables_in_test Tables_in_test
t1
#mysql50#v-1 #mysql50#v-1
t1
v1 v1
test.t1 OK test.t1 OK
show tables; show tables;
Tables_in_test Tables_in_test
t1 t1
v1
v-1 v-1
v1
drop view v1, `v-1`; drop view v1, `v-1`;
drop table t1; drop table t1;
SET NAMES utf8; SET NAMES utf8;
......
...@@ -234,8 +234,8 @@ test ...@@ -234,8 +234,8 @@ test
# 'bug58090' database should be present. # 'bug58090' database should be present.
SHOW DATABASES; SHOW DATABASES;
Database Database
information_schema
bug58090 bug58090
information_schema
mtr mtr
mysql mysql
performance_schema performance_schema
......
...@@ -5,8 +5,8 @@ Database Create Database ...@@ -5,8 +5,8 @@ Database Create Database
foo CREATE DATABASE `foo` /*!40100 DEFAULT CHARACTER SET latin1 */ foo CREATE DATABASE `foo` /*!40100 DEFAULT CHARACTER SET latin1 */
show schemas; show schemas;
Database Database
information_schema
foo foo
information_schema
mtr mtr
mysql mysql
performance_schema performance_schema
......
...@@ -33,12 +33,12 @@ show create database `#mysql50#mysqltest-1`; ...@@ -33,12 +33,12 @@ show create database `#mysql50#mysqltest-1`;
ERROR 42000: Unknown database '#mysql50#mysqltest-1' ERROR 42000: Unknown database '#mysql50#mysqltest-1'
show tables in `mysqltest1`; show tables in `mysqltest1`;
Tables_in_mysqltest1 Tables_in_mysqltest1
t1
t-1 t-1
t1
show tables in `mysqltest-1`; show tables in `mysqltest-1`;
Tables_in_mysqltest-1 Tables_in_mysqltest-1
t1
t-1 t-1
t1
drop database `mysqltest1`; drop database `mysqltest1`;
drop database `mysqltest-1`; drop database `mysqltest-1`;
drop table if exists `txu@0023p@0023p1`; drop table if exists `txu@0023p@0023p1`;
......
...@@ -9,12 +9,12 @@ CREATE DATABASE ` ...@@ -9,12 +9,12 @@ CREATE DATABASE `
SELECT schema_name from information_schema.schemata where schema_name <> 'mtr'; SELECT schema_name from information_schema.schemata where schema_name <> 'mtr';
schema_name schema_name
information_schema information_schema
ソ十表
日本語
ニホンゴ
mysql mysql
performance_schema performance_schema
test test
ソ十表
日本語
ニホンゴ
USE `ニホンゴ`; USE `ニホンゴ`;
USE `日本語`; USE `日本語`;
USE `ソ十表`; USE `ソ十表`;
......
...@@ -10,12 +10,12 @@ CREATE DATABASE ` ...@@ -10,12 +10,12 @@ CREATE DATABASE `
SELECT schema_name from information_schema.schemata where schema_name <> 'mtr'; SELECT schema_name from information_schema.schemata where schema_name <> 'mtr';
schema_name schema_name
information_schema information_schema
日本語
龔龖龗
ニホンゴ
mysql mysql
performance_schema performance_schema
test test
日本語
龔龖龗
ニホンゴ
USE `ニホンゴ`; USE `ニホンゴ`;
USE `日本語`; USE `日本語`;
USE `龔龖龗`; USE `龔龖龗`;
......
...@@ -9,12 +9,12 @@ CREATE DATABASE ` ...@@ -9,12 +9,12 @@ CREATE DATABASE `
SELECT schema_name from information_schema.schemata where schema_name <> 'mtr'; SELECT schema_name from information_schema.schemata where schema_name <> 'mtr';
schema_name schema_name
information_schema information_schema
日本語
龔龖龗
ニホンゴ
mysql mysql
performance_schema performance_schema
test test
日本語
龔龖龗
ニホンゴ
USE `ニホンゴ`; USE `ニホンゴ`;
USE `日本語`; USE `日本語`;
USE `龔龖龗`; USE `龔龖龗`;
......
...@@ -9,12 +9,12 @@ CREATE DATABASE `龔龖龗`; ...@@ -9,12 +9,12 @@ CREATE DATABASE `龔龖龗`;
SELECT schema_name from information_schema.schemata where schema_name <> 'mtr'; SELECT schema_name from information_schema.schemata where schema_name <> 'mtr';
schema_name schema_name
information_schema information_schema
日本語
龔龖龗
ニホンゴ
mysql mysql
performance_schema performance_schema
test test
日本語
龔龖龗
ニホンゴ
USE `ニホンゴ`; USE `ニホンゴ`;
USE `日本語`; USE `日本語`;
USE `龔龖龗`; USE `龔龖龗`;
......
...@@ -6,8 +6,8 @@ e,lost+found,.mysqlgui,ignored_db ...@@ -6,8 +6,8 @@ e,lost+found,.mysqlgui,ignored_db
# with '.' # with '.'
SHOW DATABASES; SHOW DATABASES;
Database Database
information_schema
#mysql50#.otherdir #mysql50#.otherdir
information_schema
mtr mtr
mysql mysql
performance_schema performance_schema
...@@ -33,8 +33,8 @@ id ...@@ -33,8 +33,8 @@ id
2 2
SHOW DATABASES; SHOW DATABASES;
Database Database
information_schema
#mysql50#.otherdir #mysql50#.otherdir
information_schema
lost+found lost+found
mtr mtr
mysql mysql
......
...@@ -687,6 +687,15 @@ db_name_is_in_ignore_db_dirs_list(const char *directory) ...@@ -687,6 +687,15 @@ db_name_is_in_ignore_db_dirs_list(const char *directory)
return my_hash_search(&ignore_db_dirs_hash, (uchar *) buff, buff_len)!=NULL; return my_hash_search(&ignore_db_dirs_hash, (uchar *) buff, buff_len)!=NULL;
} }
extern "C" {
static int cmp_table_names(LEX_STRING * const *a, LEX_STRING * const *b)
{
return my_strnncoll(&my_charset_bin,
(const uchar*)((*a)->str), (*a)->length,
(const uchar*)((*b)->str), (*b)->length);
}
}
enum find_files_result { enum find_files_result {
FIND_FILES_OK, FIND_FILES_OK,
FIND_FILES_OOM, FIND_FILES_OOM,
...@@ -740,7 +749,7 @@ find_files(THD *thd, Dynamic_array<LEX_STRING*> *files, const char *db, ...@@ -740,7 +749,7 @@ find_files(THD *thd, Dynamic_array<LEX_STRING*> *files, const char *db,
bzero((char*) &table_list,sizeof(table_list)); bzero((char*) &table_list,sizeof(table_list));
if (!(dirp = my_dir(path, MYF((dir ? MY_WANT_STAT : 0) | MY_WANT_SORT | if (!(dirp = my_dir(path, MYF((dir ? MY_WANT_STAT : 0) |
MY_THREAD_SPECIFIC)))) MY_THREAD_SPECIFIC))))
{ {
if (my_errno == ENOENT) if (my_errno == ENOENT)
...@@ -838,6 +847,8 @@ find_files(THD *thd, Dynamic_array<LEX_STRING*> *files, const char *db, ...@@ -838,6 +847,8 @@ find_files(THD *thd, Dynamic_array<LEX_STRING*> *files, const char *db,
DBUG_PRINT("info",("found: %d files", files->elements())); DBUG_PRINT("info",("found: %d files", files->elements()));
my_dirend(dirp); my_dirend(dirp);
files->sort(cmp_table_names);
DBUG_RETURN(FIND_FILES_OK); DBUG_RETURN(FIND_FILES_OK);
} }
...@@ -3694,8 +3705,6 @@ enum enum_schema_tables get_schema_table_idx(ST_SCHEMA_TABLE *schema_table) ...@@ -3694,8 +3705,6 @@ enum enum_schema_tables get_schema_table_idx(ST_SCHEMA_TABLE *schema_table)
wild wild string wild wild string
idx_field_vals idx_field_vals->db_name contains db name or idx_field_vals idx_field_vals->db_name contains db name or
wild string wild string
with_i_schema returns 1 if we added 'IS' name to list
otherwise returns 0
RETURN RETURN
zero success zero success
...@@ -3703,13 +3712,8 @@ enum enum_schema_tables get_schema_table_idx(ST_SCHEMA_TABLE *schema_table) ...@@ -3703,13 +3712,8 @@ enum enum_schema_tables get_schema_table_idx(ST_SCHEMA_TABLE *schema_table)
*/ */
int make_db_list(THD *thd, Dynamic_array<LEX_STRING*> *files, int make_db_list(THD *thd, Dynamic_array<LEX_STRING*> *files,
LOOKUP_FIELD_VALUES *lookup_field_vals, LOOKUP_FIELD_VALUES *lookup_field_vals)
bool *with_i_schema)
{ {
LEX_STRING *i_s_name_copy= 0;
i_s_name_copy= thd->make_lex_string(INFORMATION_SCHEMA_NAME.str,
INFORMATION_SCHEMA_NAME.length);
*with_i_schema= 0;
if (lookup_field_vals->wild_db_value) if (lookup_field_vals->wild_db_value)
{ {
/* /*
...@@ -3722,8 +3726,7 @@ int make_db_list(THD *thd, Dynamic_array<LEX_STRING*> *files, ...@@ -3722,8 +3726,7 @@ int make_db_list(THD *thd, Dynamic_array<LEX_STRING*> *files,
INFORMATION_SCHEMA_NAME.str, INFORMATION_SCHEMA_NAME.str,
lookup_field_vals->db_value.str)) lookup_field_vals->db_value.str))
{ {
*with_i_schema= 1; if (files->append_val(&INFORMATION_SCHEMA_NAME))
if (files->append(i_s_name_copy))
return 1; return 1;
} }
return (find_files(thd, files, NullS, mysql_data_home, return (find_files(thd, files, NullS, mysql_data_home,
...@@ -3743,8 +3746,7 @@ int make_db_list(THD *thd, Dynamic_array<LEX_STRING*> *files, ...@@ -3743,8 +3746,7 @@ int make_db_list(THD *thd, Dynamic_array<LEX_STRING*> *files,
if (is_infoschema_db(lookup_field_vals->db_value.str, if (is_infoschema_db(lookup_field_vals->db_value.str,
lookup_field_vals->db_value.length)) lookup_field_vals->db_value.length))
{ {
*with_i_schema= 1; if (files->append_val(&INFORMATION_SCHEMA_NAME))
if (files->append(i_s_name_copy))
return 1; return 1;
return 0; return 0;
} }
...@@ -3757,9 +3759,8 @@ int make_db_list(THD *thd, Dynamic_array<LEX_STRING*> *files, ...@@ -3757,9 +3759,8 @@ int make_db_list(THD *thd, Dynamic_array<LEX_STRING*> *files,
Create list of existing databases. It is used in case Create list of existing databases. It is used in case
of select from information schema table of select from information schema table
*/ */
if (files->append(i_s_name_copy)) if (files->append_val(&INFORMATION_SCHEMA_NAME))
return 1; return 1;
*with_i_schema= 1;
return (find_files(thd, files, NullS, return (find_files(thd, files, NullS,
mysql_data_home, NullS, 1) != FIND_FILES_OK); mysql_data_home, NullS, 1) != FIND_FILES_OK);
} }
...@@ -3857,7 +3858,6 @@ int schema_tables_add(THD *thd, Dynamic_array<LEX_STRING*> *files, ...@@ -3857,7 +3858,6 @@ int schema_tables_add(THD *thd, Dynamic_array<LEX_STRING*> *files,
@param[in] table_names List of table names in database @param[in] table_names List of table names in database
@param[in] lex pointer to LEX struct @param[in] lex pointer to LEX struct
@param[in] lookup_field_vals pointer to LOOKUP_FIELD_VALUE struct @param[in] lookup_field_vals pointer to LOOKUP_FIELD_VALUE struct
@param[in] with_i_schema TRUE means that we add I_S tables to list
@param[in] db_name database name @param[in] db_name database name
@return Operation status @return Operation status
...@@ -3869,14 +3869,14 @@ int schema_tables_add(THD *thd, Dynamic_array<LEX_STRING*> *files, ...@@ -3869,14 +3869,14 @@ int schema_tables_add(THD *thd, Dynamic_array<LEX_STRING*> *files,
static int static int
make_table_name_list(THD *thd, Dynamic_array<LEX_STRING*> *table_names, make_table_name_list(THD *thd, Dynamic_array<LEX_STRING*> *table_names,
LEX *lex, LOOKUP_FIELD_VALUES *lookup_field_vals, LEX *lex, LOOKUP_FIELD_VALUES *lookup_field_vals,
bool with_i_schema, LEX_STRING *db_name) LEX_STRING *db_name)
{ {
char path[FN_REFLEN + 1]; char path[FN_REFLEN + 1];
build_table_filename(path, sizeof(path) - 1, db_name->str, "", "", 0); build_table_filename(path, sizeof(path) - 1, db_name->str, "", "", 0);
if (!lookup_field_vals->wild_table_value && if (!lookup_field_vals->wild_table_value &&
lookup_field_vals->table_value.str) lookup_field_vals->table_value.str)
{ {
if (with_i_schema) if (db_name == &INFORMATION_SCHEMA_NAME)
{ {
LEX_STRING *name; LEX_STRING *name;
ST_SCHEMA_TABLE *schema_table= ST_SCHEMA_TABLE *schema_table=
...@@ -3901,7 +3901,7 @@ make_table_name_list(THD *thd, Dynamic_array<LEX_STRING*> *table_names, ...@@ -3901,7 +3901,7 @@ make_table_name_list(THD *thd, Dynamic_array<LEX_STRING*> *table_names,
This call will add all matching the wildcards (if specified) IS tables This call will add all matching the wildcards (if specified) IS tables
to the list to the list
*/ */
if (with_i_schema) if (db_name == &INFORMATION_SCHEMA_NAME)
return (schema_tables_add(thd, table_names, return (schema_tables_add(thd, table_names,
lookup_field_vals->table_value.str)); lookup_field_vals->table_value.str));
...@@ -4129,7 +4129,6 @@ fill_schema_table_by_open(THD *thd, bool is_show_fields_or_keys, ...@@ -4129,7 +4129,6 @@ fill_schema_table_by_open(THD *thd, bool is_show_fields_or_keys,
@param[in] table TABLE struct for I_S table @param[in] table TABLE struct for I_S table
@param[in] db_name database name @param[in] db_name database name
@param[in] table_name table name @param[in] table_name table name
@param[in] with_i_schema I_S table if TRUE
@return Operation status @return Operation status
@retval 0 success @retval 0 success
...@@ -4137,11 +4136,10 @@ fill_schema_table_by_open(THD *thd, bool is_show_fields_or_keys, ...@@ -4137,11 +4136,10 @@ fill_schema_table_by_open(THD *thd, bool is_show_fields_or_keys,
*/ */
static int fill_schema_table_names(THD *thd, TABLE_LIST *tables, static int fill_schema_table_names(THD *thd, TABLE_LIST *tables,
LEX_STRING *db_name, LEX_STRING *table_name, LEX_STRING *db_name, LEX_STRING *table_name)
bool with_i_schema)
{ {
TABLE *table= tables->table; TABLE *table= tables->table;
if (with_i_schema) if (db_name == &INFORMATION_SCHEMA_NAME)
{ {
table->field[3]->store(STRING_WITH_LEN("SYSTEM VIEW"), table->field[3]->store(STRING_WITH_LEN("SYSTEM VIEW"),
system_charset_info); system_charset_info);
...@@ -4550,8 +4548,6 @@ int get_all_tables(THD *thd, TABLE_LIST *tables, COND *cond) ...@@ -4550,8 +4548,6 @@ int get_all_tables(THD *thd, TABLE_LIST *tables, COND *cond)
SELECT_LEX *lsel= tables->schema_select_lex; SELECT_LEX *lsel= tables->schema_select_lex;
ST_SCHEMA_TABLE *schema_table= tables->schema_table; ST_SCHEMA_TABLE *schema_table= tables->schema_table;
LOOKUP_FIELD_VALUES lookup_field_vals; LOOKUP_FIELD_VALUES lookup_field_vals;
LEX_STRING *db_name;
bool with_i_schema;
enum enum_schema_tables schema_table_idx; enum enum_schema_tables schema_table_idx;
Dynamic_array<LEX_STRING*> db_names; Dynamic_array<LEX_STRING*> db_names;
COND *partial_cond= 0; COND *partial_cond= 0;
...@@ -4655,11 +4651,11 @@ int get_all_tables(THD *thd, TABLE_LIST *tables, COND *cond) ...@@ -4655,11 +4651,11 @@ int get_all_tables(THD *thd, TABLE_LIST *tables, COND *cond)
goto err; goto err;
} }
if (make_db_list(thd, &db_names, &lookup_field_vals, &with_i_schema)) if (make_db_list(thd, &db_names, &lookup_field_vals))
goto err; goto err;
for (int i=0; i < db_names.elements(); i++) for (int i=0; i < db_names.elements(); i++)
{ {
db_name= db_names.at(i); LEX_STRING *db_name= db_names.at(i);
#ifndef NO_EMBEDDED_ACCESS_CHECKS #ifndef NO_EMBEDDED_ACCESS_CHECKS
if (!(check_access(thd, SELECT_ACL, db_name->str, if (!(check_access(thd, SELECT_ACL, db_name->str,
&thd->col_access, NULL, 0, 1) || &thd->col_access, NULL, 0, 1) ||
...@@ -4670,8 +4666,7 @@ int get_all_tables(THD *thd, TABLE_LIST *tables, COND *cond) ...@@ -4670,8 +4666,7 @@ int get_all_tables(THD *thd, TABLE_LIST *tables, COND *cond)
{ {
Dynamic_array<LEX_STRING*> table_names; Dynamic_array<LEX_STRING*> table_names;
int res= make_table_name_list(thd, &table_names, lex, int res= make_table_name_list(thd, &table_names, lex,
&lookup_field_vals, &lookup_field_vals, db_name);
with_i_schema, db_name);
if (res == 2) /* Not fatal error, continue */ if (res == 2) /* Not fatal error, continue */
continue; continue;
if (res) if (res)
...@@ -4707,14 +4702,13 @@ int get_all_tables(THD *thd, TABLE_LIST *tables, COND *cond) ...@@ -4707,14 +4702,13 @@ int get_all_tables(THD *thd, TABLE_LIST *tables, COND *cond)
/* SHOW TABLE NAMES command */ /* SHOW TABLE NAMES command */
if (schema_table_idx == SCH_TABLE_NAMES) if (schema_table_idx == SCH_TABLE_NAMES)
{ {
if (fill_schema_table_names(thd, tables, db_name, if (fill_schema_table_names(thd, tables, db_name, table_name))
table_name, with_i_schema))
continue; continue;
} }
else else
{ {
if (!(table_open_method & ~OPEN_FRM_ONLY) && if (!(table_open_method & ~OPEN_FRM_ONLY) &&
!with_i_schema) db_name != &INFORMATION_SCHEMA_NAME)
{ {
/* /*
Here we need to filter out warnings, which can happen Here we need to filter out warnings, which can happen
...@@ -4748,11 +4742,6 @@ int get_all_tables(THD *thd, TABLE_LIST *tables, COND *cond) ...@@ -4748,11 +4742,6 @@ int get_all_tables(THD *thd, TABLE_LIST *tables, COND *cond)
} }
} }
} }
/*
If we have information schema its always the first table and only
the first table. Reset for other tables.
*/
with_i_schema= 0;
} }
} }
...@@ -4785,7 +4774,6 @@ int fill_schema_schemata(THD *thd, TABLE_LIST *tables, COND *cond) ...@@ -4785,7 +4774,6 @@ int fill_schema_schemata(THD *thd, TABLE_LIST *tables, COND *cond)
LOOKUP_FIELD_VALUES lookup_field_vals; LOOKUP_FIELD_VALUES lookup_field_vals;
Dynamic_array<LEX_STRING*> db_names; Dynamic_array<LEX_STRING*> db_names;
bool with_i_schema;
HA_CREATE_INFO create; HA_CREATE_INFO create;
TABLE *table= tables->table; TABLE *table= tables->table;
#ifndef NO_EMBEDDED_ACCESS_CHECKS #ifndef NO_EMBEDDED_ACCESS_CHECKS
...@@ -4798,15 +4786,14 @@ int fill_schema_schemata(THD *thd, TABLE_LIST *tables, COND *cond) ...@@ -4798,15 +4786,14 @@ int fill_schema_schemata(THD *thd, TABLE_LIST *tables, COND *cond)
DBUG_PRINT("INDEX VALUES",("db_name: %s table_name: %s", DBUG_PRINT("INDEX VALUES",("db_name: %s table_name: %s",
lookup_field_vals.db_value.str, lookup_field_vals.db_value.str,
lookup_field_vals.table_value.str)); lookup_field_vals.table_value.str));
if (make_db_list(thd, &db_names, &lookup_field_vals, if (make_db_list(thd, &db_names, &lookup_field_vals))
&with_i_schema))
DBUG_RETURN(1); DBUG_RETURN(1);
/* /*
If we have lookup db value we should check that the database exists If we have lookup db value we should check that the database exists
*/ */
if(lookup_field_vals.db_value.str && !lookup_field_vals.wild_db_value && if(lookup_field_vals.db_value.str && !lookup_field_vals.wild_db_value &&
!with_i_schema) db_names.at(0) != &INFORMATION_SCHEMA_NAME)
{ {
char path[FN_REFLEN+16]; char path[FN_REFLEN+16];
uint path_len; uint path_len;
...@@ -4823,12 +4810,11 @@ int fill_schema_schemata(THD *thd, TABLE_LIST *tables, COND *cond) ...@@ -4823,12 +4810,11 @@ int fill_schema_schemata(THD *thd, TABLE_LIST *tables, COND *cond)
for (int i=0; i < db_names.elements(); i++) for (int i=0; i < db_names.elements(); i++)
{ {
LEX_STRING *db_name= db_names.at(i); LEX_STRING *db_name= db_names.at(i);
if (with_i_schema) // information schema name is always first in list if (db_name == &INFORMATION_SCHEMA_NAME)
{ {
if (store_schema_shemata(thd, table, db_name, if (store_schema_shemata(thd, table, db_name,
system_charset_info)) system_charset_info))
DBUG_RETURN(1); DBUG_RETURN(1);
with_i_schema= 0;
continue; continue;
} }
#ifndef NO_EMBEDDED_ACCESS_CHECKS #ifndef NO_EMBEDDED_ACCESS_CHECKS
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment