Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
D
doctrine-dbal
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Tomáš Trávníček
doctrine-dbal
Commits
17edd0d5
Commit
17edd0d5
authored
Dec 20, 2013
by
Steve Müller
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
fix column default value lifecycle
parent
ba4ab893
Changes
9
Hide whitespace changes
Inline
Side-by-side
Showing
9 changed files
with
88 additions
and
14 deletions
+88
-14
PostgreSqlPlatform.php
lib/Doctrine/DBAL/Platforms/PostgreSqlPlatform.php
+4
-1
SQLAnywherePlatform.php
lib/Doctrine/DBAL/Platforms/SQLAnywherePlatform.php
+8
-2
SQLServerPlatform.php
lib/Doctrine/DBAL/Platforms/SQLServerPlatform.php
+7
-7
Comparator.php
lib/Doctrine/DBAL/Schema/Comparator.php
+9
-1
DrizzleSchemaManager.php
lib/Doctrine/DBAL/Schema/DrizzleSchemaManager.php
+1
-1
SQLAnywhereSchemaManager.php
lib/Doctrine/DBAL/Schema/SQLAnywhereSchemaManager.php
+4
-0
SchemaManagerFunctionalTestCase.php
...BAL/Functional/Schema/SchemaManagerFunctionalTestCase.php
+52
-0
PostgreSqlPlatformTest.php
.../Doctrine/Tests/DBAL/Platforms/PostgreSqlPlatformTest.php
+2
-2
SQLServerPlatformTest.php
...s/Doctrine/Tests/DBAL/Platforms/SQLServerPlatformTest.php
+1
-0
No files found.
lib/Doctrine/DBAL/Platforms/PostgreSqlPlatform.php
View file @
17edd0d5
...
@@ -470,7 +470,10 @@ class PostgreSqlPlatform extends AbstractPlatform
...
@@ -470,7 +470,10 @@ class PostgreSqlPlatform extends AbstractPlatform
}
}
if
(
$columnDiff
->
hasChanged
(
'default'
))
{
if
(
$columnDiff
->
hasChanged
(
'default'
))
{
$query
=
'ALTER '
.
$oldColumnName
.
' SET '
.
$this
->
getDefaultValueDeclarationSQL
(
$column
->
toArray
());
$defaultClause
=
null
===
$column
->
getDefault
()
?
' DROP DEFAULT'
:
' SET'
.
$this
->
getDefaultValueDeclarationSQL
(
$column
->
toArray
());
$query
=
'ALTER '
.
$oldColumnName
.
$defaultClause
;
$sql
[]
=
'ALTER TABLE '
.
$diff
->
getName
()
->
getQuotedName
(
$this
)
.
' '
.
$query
;
$sql
[]
=
'ALTER TABLE '
.
$diff
->
getName
()
->
getQuotedName
(
$this
)
.
' '
.
$query
;
}
}
...
...
lib/Doctrine/DBAL/Platforms/SQLAnywherePlatform.php
View file @
17edd0d5
...
@@ -22,7 +22,6 @@ namespace Doctrine\DBAL\Platforms;
...
@@ -22,7 +22,6 @@ namespace Doctrine\DBAL\Platforms;
use
Doctrine\DBAL\Connection
;
use
Doctrine\DBAL\Connection
;
use
Doctrine\DBAL\DBALException
;
use
Doctrine\DBAL\DBALException
;
use
Doctrine\DBAL\LockMode
;
use
Doctrine\DBAL\LockMode
;
use
Doctrine\DBAL\Platforms\AbstractPlatform
;
use
Doctrine\DBAL\Schema\Column
;
use
Doctrine\DBAL\Schema\Column
;
use
Doctrine\DBAL\Schema\ColumnDiff
;
use
Doctrine\DBAL\Schema\ColumnDiff
;
use
Doctrine\DBAL\Schema\Constraint
;
use
Doctrine\DBAL\Schema\Constraint
;
...
@@ -287,7 +286,14 @@ class SQLAnywherePlatform extends AbstractPlatform
...
@@ -287,7 +286,14 @@ class SQLAnywherePlatform extends AbstractPlatform
// Do not return alter clause if only comment has changed.
// Do not return alter clause if only comment has changed.
if
(
!
(
$columnDiff
->
hasChanged
(
'comment'
)
&&
count
(
$columnDiff
->
changedProperties
)
===
1
))
{
if
(
!
(
$columnDiff
->
hasChanged
(
'comment'
)
&&
count
(
$columnDiff
->
changedProperties
)
===
1
))
{
return
'ALTER '
.
$this
->
getColumnDeclarationSQL
(
$column
->
getQuotedName
(
$this
),
$column
->
toArray
());
$columnAlterationClause
=
'ALTER '
.
$this
->
getColumnDeclarationSQL
(
$column
->
getQuotedName
(
$this
),
$column
->
toArray
());
if
(
$columnDiff
->
hasChanged
(
'default'
)
&&
null
===
$column
->
getDefault
())
{
$columnAlterationClause
.=
', ALTER '
.
$column
->
getQuotedName
(
$this
)
.
' DROP DEFAULT'
;
}
return
$columnAlterationClause
;
}
}
}
}
...
...
lib/Doctrine/DBAL/Platforms/SQLServerPlatform.php
View file @
17edd0d5
...
@@ -246,7 +246,7 @@ class SQLServerPlatform extends AbstractPlatform
...
@@ -246,7 +246,7 @@ class SQLServerPlatform extends AbstractPlatform
}
}
// Build default constraints SQL statements.
// Build default constraints SQL statements.
if
(
!
empty
(
$column
[
'default'
])
||
is_numeric
(
$column
[
'default'
]))
{
if
(
isset
(
$column
[
'default'
]))
{
$defaultConstraintsSql
[]
=
'ALTER TABLE '
.
$tableName
.
$defaultConstraintsSql
[]
=
'ALTER TABLE '
.
$tableName
.
' ADD'
.
$this
->
getDefaultConstraintDeclarationSQL
(
$tableName
,
$column
);
' ADD'
.
$this
->
getDefaultConstraintDeclarationSQL
(
$tableName
,
$column
);
}
}
...
@@ -352,7 +352,7 @@ class SQLServerPlatform extends AbstractPlatform
...
@@ -352,7 +352,7 @@ class SQLServerPlatform extends AbstractPlatform
*/
*/
public
function
getDefaultConstraintDeclarationSQL
(
$table
,
array
$column
)
public
function
getDefaultConstraintDeclarationSQL
(
$table
,
array
$column
)
{
{
if
(
empty
(
$column
[
'default'
])
&&
!
is_numeric
(
$column
[
'default'
]))
{
if
(
!
isset
(
$column
[
'default'
]))
{
throw
new
\InvalidArgumentException
(
"Incomplete column definition. 'default' required."
);
throw
new
\InvalidArgumentException
(
"Incomplete column definition. 'default' required."
);
}
}
...
@@ -446,7 +446,7 @@ class SQLServerPlatform extends AbstractPlatform
...
@@ -446,7 +446,7 @@ class SQLServerPlatform extends AbstractPlatform
$columnDef
=
$column
->
toArray
();
$columnDef
=
$column
->
toArray
();
$queryParts
[]
=
'ADD '
.
$this
->
getColumnDeclarationSQL
(
$column
->
getQuotedName
(
$this
),
$columnDef
);
$queryParts
[]
=
'ADD '
.
$this
->
getColumnDeclarationSQL
(
$column
->
getQuotedName
(
$this
),
$columnDef
);
if
(
!
empty
(
$columnDef
[
'default'
])
||
is_numeric
(
$columnDef
[
'default'
]))
{
if
(
isset
(
$columnDef
[
'default'
]))
{
$columnDef
[
'name'
]
=
$column
->
getQuotedName
(
$this
);
$columnDef
[
'name'
]
=
$column
->
getQuotedName
(
$this
);
$queryParts
[]
=
'ADD'
.
$this
->
getDefaultConstraintDeclarationSQL
(
$diff
->
name
,
$columnDef
);
$queryParts
[]
=
'ADD'
.
$this
->
getDefaultConstraintDeclarationSQL
(
$diff
->
name
,
$columnDef
);
}
}
...
@@ -517,7 +517,7 @@ class SQLServerPlatform extends AbstractPlatform
...
@@ -517,7 +517,7 @@ class SQLServerPlatform extends AbstractPlatform
* if default value has changed and another
* if default value has changed and another
* default constraint already exists for the column.
* default constraint already exists for the column.
*/
*/
if
(
$columnDefaultHasChanged
&&
(
!
empty
(
$fromColumnDefault
)
||
is_numeric
(
$fromColumnDefault
))
)
{
if
(
$columnDefaultHasChanged
&&
null
!==
$fromColumnDefault
)
{
$queryParts
[]
=
'DROP CONSTRAINT '
.
$queryParts
[]
=
'DROP CONSTRAINT '
.
$this
->
generateDefaultConstraintName
(
$diff
->
name
,
$columnDiff
->
oldColumnName
);
$this
->
generateDefaultConstraintName
(
$diff
->
name
,
$columnDiff
->
oldColumnName
);
}
}
...
@@ -525,7 +525,7 @@ class SQLServerPlatform extends AbstractPlatform
...
@@ -525,7 +525,7 @@ class SQLServerPlatform extends AbstractPlatform
$queryParts
[]
=
'ALTER COLUMN '
.
$queryParts
[]
=
'ALTER COLUMN '
.
$this
->
getColumnDeclarationSQL
(
$column
->
getQuotedName
(
$this
),
$columnDef
);
$this
->
getColumnDeclarationSQL
(
$column
->
getQuotedName
(
$this
),
$columnDef
);
if
(
$columnDefaultHasChanged
&&
(
!
empty
(
$columnDef
[
'default'
])
||
is_numeric
(
$columnDef
[
'default'
])
))
{
if
(
$columnDefaultHasChanged
&&
isset
(
$columnDef
[
'default'
]
))
{
$columnDef
[
'name'
]
=
$column
->
getQuotedName
(
$this
);
$columnDef
[
'name'
]
=
$column
->
getQuotedName
(
$this
);
$queryParts
[]
=
'ADD'
.
$this
->
getDefaultConstraintDeclarationSQL
(
$diff
->
name
,
$columnDef
);
$queryParts
[]
=
'ADD'
.
$this
->
getDefaultConstraintDeclarationSQL
(
$diff
->
name
,
$columnDef
);
}
}
...
@@ -546,7 +546,7 @@ class SQLServerPlatform extends AbstractPlatform
...
@@ -546,7 +546,7 @@ class SQLServerPlatform extends AbstractPlatform
* Drop existing default constraint for the old column name
* Drop existing default constraint for the old column name
* if column has default value.
* if column has default value.
*/
*/
if
(
!
empty
(
$columnDef
[
'default'
])
||
is_numeric
(
$columnDef
[
'default'
]))
{
if
(
isset
(
$columnDef
[
'default'
]))
{
$queryParts
[]
=
'DROP CONSTRAINT '
.
$queryParts
[]
=
'DROP CONSTRAINT '
.
$this
->
generateDefaultConstraintName
(
$diff
->
name
,
$oldColumnName
);
$this
->
generateDefaultConstraintName
(
$diff
->
name
,
$oldColumnName
);
}
}
...
@@ -557,7 +557,7 @@ class SQLServerPlatform extends AbstractPlatform
...
@@ -557,7 +557,7 @@ class SQLServerPlatform extends AbstractPlatform
/**
/**
* Readd default constraint for the new column name.
* Readd default constraint for the new column name.
*/
*/
if
(
!
empty
(
$columnDef
[
'default'
])
||
is_numeric
(
$columnDef
[
'default'
]))
{
if
(
isset
(
$columnDef
[
'default'
]))
{
$columnDef
[
'name'
]
=
$column
->
getQuotedName
(
$this
);
$columnDef
[
'name'
]
=
$column
->
getQuotedName
(
$this
);
$queryParts
[]
=
'ADD'
.
$this
->
getDefaultConstraintDeclarationSQL
(
$diff
->
name
,
$columnDef
);
$queryParts
[]
=
'ADD'
.
$this
->
getDefaultConstraintDeclarationSQL
(
$diff
->
name
,
$columnDef
);
}
}
...
...
lib/Doctrine/DBAL/Schema/Comparator.php
View file @
17edd0d5
...
@@ -369,7 +369,15 @@ class Comparator
...
@@ -369,7 +369,15 @@ class Comparator
$changedProperties
[]
=
'notnull'
;
$changedProperties
[]
=
'notnull'
;
}
}
if
(
$column1
->
getDefault
()
!=
$column2
->
getDefault
())
{
$column1Default
=
$column1
->
getDefault
();
$column2Default
=
$column2
->
getDefault
();
if
(
$column1Default
!=
$column2Default
||
// Null values need to be checked additionally as they tell whether to create or drop a default value.
// null != 0, null != false, null != '' etc. This affects platform's table alteration SQL generation.
(
null
===
$column1Default
&&
null
!==
$column2Default
)
||
(
null
===
$column2Default
&&
null
!==
$column1Default
)
)
{
$changedProperties
[]
=
'default'
;
$changedProperties
[]
=
'default'
;
}
}
...
...
lib/Doctrine/DBAL/Schema/DrizzleSchemaManager.php
View file @
17edd0d5
...
@@ -41,7 +41,7 @@ class DrizzleSchemaManager extends AbstractSchemaManager
...
@@ -41,7 +41,7 @@ class DrizzleSchemaManager extends AbstractSchemaManager
$options
=
array
(
$options
=
array
(
'notnull'
=>
!
(
bool
)
$tableColumn
[
'IS_NULLABLE'
],
'notnull'
=>
!
(
bool
)
$tableColumn
[
'IS_NULLABLE'
],
'length'
=>
(
int
)
$tableColumn
[
'CHARACTER_MAXIMUM_LENGTH'
],
'length'
=>
(
int
)
$tableColumn
[
'CHARACTER_MAXIMUM_LENGTH'
],
'default'
=>
empty
(
$tableColumn
[
'COLUMN_DEFAULT'
])
?
null
:
$tableColumn
[
'COLUMN_DEFAULT'
]
,
'default'
=>
isset
(
$tableColumn
[
'COLUMN_DEFAULT'
])
?
$tableColumn
[
'COLUMN_DEFAULT'
]
:
null
,
'autoincrement'
=>
(
bool
)
$tableColumn
[
'IS_AUTO_INCREMENT'
],
'autoincrement'
=>
(
bool
)
$tableColumn
[
'IS_AUTO_INCREMENT'
],
'scale'
=>
(
int
)
$tableColumn
[
'NUMERIC_SCALE'
],
'scale'
=>
(
int
)
$tableColumn
[
'NUMERIC_SCALE'
],
'precision'
=>
(
int
)
$tableColumn
[
'NUMERIC_PRECISION'
],
'precision'
=>
(
int
)
$tableColumn
[
'NUMERIC_PRECISION'
],
...
...
lib/Doctrine/DBAL/Schema/SQLAnywhereSchemaManager.php
View file @
17edd0d5
...
@@ -112,6 +112,10 @@ class SQLAnywhereSchemaManager extends AbstractSchemaManager
...
@@ -112,6 +112,10 @@ class SQLAnywhereSchemaManager extends AbstractSchemaManager
if
(
$tableColumn
[
'default'
])
{
if
(
$tableColumn
[
'default'
])
{
// Strip quotes from default value.
// Strip quotes from default value.
$default
=
preg_replace
(
array
(
"/^'(.*)'$/"
,
"/''/"
),
array
(
"$1"
,
"'"
),
$tableColumn
[
'default'
]);
$default
=
preg_replace
(
array
(
"/^'(.*)'$/"
,
"/''/"
),
array
(
"$1"
,
"'"
),
$tableColumn
[
'default'
]);
if
(
'autoincrement'
==
$default
)
{
$default
=
null
;
}
}
}
switch
(
$tableColumn
[
'type'
])
{
switch
(
$tableColumn
[
'type'
])
{
...
...
tests/Doctrine/Tests/DBAL/Functional/Schema/SchemaManagerFunctionalTestCase.php
View file @
17edd0d5
...
@@ -2,6 +2,8 @@
...
@@ -2,6 +2,8 @@
namespace
Doctrine\Tests\DBAL\Functional\Schema
;
namespace
Doctrine\Tests\DBAL\Functional\Schema
;
use
Doctrine\DBAL\Schema\Comparator
;
use
Doctrine\DBAL\Schema\Table
;
use
Doctrine\DBAL\Types\Type
,
use
Doctrine\DBAL\Types\Type
,
Doctrine\DBAL\Schema\AbstractSchemaManager
;
Doctrine\DBAL\Schema\AbstractSchemaManager
;
use
Doctrine\DBAL\Platforms\AbstractPlatform
;
use
Doctrine\DBAL\Platforms\AbstractPlatform
;
...
@@ -670,4 +672,54 @@ class SchemaManagerFunctionalTestCase extends \Doctrine\Tests\DbalFunctionalTest
...
@@ -670,4 +672,54 @@ class SchemaManagerFunctionalTestCase extends \Doctrine\Tests\DbalFunctionalTest
$this
->
assertEquals
(
array
(
'id'
,
'foreign_key_test'
),
array_map
(
'strtolower'
,
$fkeys
[
0
]
->
getLocalColumns
()));
$this
->
assertEquals
(
array
(
'id'
,
'foreign_key_test'
),
array_map
(
'strtolower'
,
$fkeys
[
0
]
->
getLocalColumns
()));
$this
->
assertEquals
(
array
(
'id'
,
'other_id'
),
array_map
(
'strtolower'
,
$fkeys
[
0
]
->
getForeignColumns
()));
$this
->
assertEquals
(
array
(
'id'
,
'other_id'
),
array_map
(
'strtolower'
,
$fkeys
[
0
]
->
getForeignColumns
()));
}
}
/**
* @group DBAL-44
*/
public
function
testColumnDefaultLifecycle
()
{
$table
=
new
Table
(
"col_def_lifecycle"
);
$table
->
addColumn
(
'id'
,
'integer'
,
array
(
'primary'
=>
true
,
'autoincrement'
=>
true
));
$table
->
addColumn
(
'column1'
,
'string'
,
array
(
'default'
=>
null
));
$table
->
addColumn
(
'column2'
,
'string'
,
array
(
'default'
=>
false
));
$table
->
addColumn
(
'column3'
,
'string'
,
array
(
'default'
=>
true
));
$table
->
addColumn
(
'column4'
,
'string'
,
array
(
'default'
=>
0
));
$table
->
addColumn
(
'column5'
,
'string'
,
array
(
'default'
=>
''
));
$table
->
addColumn
(
'column6'
,
'string'
,
array
(
'default'
=>
'def'
));
$table
->
setPrimaryKey
(
array
(
'id'
));
$this
->
_sm
->
dropAndCreateTable
(
$table
);
$columns
=
$this
->
_sm
->
listTableColumns
(
'col_def_lifecycle'
);
$this
->
assertNull
(
$columns
[
'id'
]
->
getDefault
());
$this
->
assertNull
(
$columns
[
'column1'
]
->
getDefault
());
$this
->
assertSame
(
''
,
$columns
[
'column2'
]
->
getDefault
());
$this
->
assertSame
(
'1'
,
$columns
[
'column3'
]
->
getDefault
());
$this
->
assertSame
(
'0'
,
$columns
[
'column4'
]
->
getDefault
());
$this
->
assertSame
(
''
,
$columns
[
'column5'
]
->
getDefault
());
$this
->
assertSame
(
'def'
,
$columns
[
'column6'
]
->
getDefault
());
$diffTable
=
clone
$table
;
$diffTable
->
changeColumn
(
'column1'
,
array
(
'default'
=>
false
));
$diffTable
->
changeColumn
(
'column2'
,
array
(
'default'
=>
null
));
$diffTable
->
changeColumn
(
'column3'
,
array
(
'default'
=>
false
));
$diffTable
->
changeColumn
(
'column4'
,
array
(
'default'
=>
null
));
$diffTable
->
changeColumn
(
'column5'
,
array
(
'default'
=>
false
));
$diffTable
->
changeColumn
(
'column6'
,
array
(
'default'
=>
666
));
$comparator
=
new
Comparator
();
$this
->
_sm
->
alterTable
(
$comparator
->
diffTable
(
$table
,
$diffTable
));
$columns
=
$this
->
_sm
->
listTableColumns
(
'col_def_lifecycle'
);
$this
->
assertSame
(
''
,
$columns
[
'column1'
]
->
getDefault
());
$this
->
assertNull
(
$columns
[
'column2'
]
->
getDefault
());
$this
->
assertSame
(
''
,
$columns
[
'column3'
]
->
getDefault
());
$this
->
assertNull
(
$columns
[
'column4'
]
->
getDefault
());
$this
->
assertSame
(
''
,
$columns
[
'column5'
]
->
getDefault
());
$this
->
assertSame
(
'666'
,
$columns
[
'column6'
]
->
getDefault
());
}
}
}
tests/Doctrine/Tests/DBAL/Platforms/PostgreSqlPlatformTest.php
View file @
17edd0d5
...
@@ -33,10 +33,10 @@ class PostgreSqlPlatformTest extends AbstractPlatformTestCase
...
@@ -33,10 +33,10 @@ class PostgreSqlPlatformTest extends AbstractPlatformTestCase
'ALTER TABLE mytable ADD quota INT DEFAULT NULL'
,
'ALTER TABLE mytable ADD quota INT DEFAULT NULL'
,
'ALTER TABLE mytable DROP foo'
,
'ALTER TABLE mytable DROP foo'
,
'ALTER TABLE mytable ALTER bar TYPE VARCHAR(255)'
,
'ALTER TABLE mytable ALTER bar TYPE VARCHAR(255)'
,
"ALTER TABLE mytable ALTER bar SET
DEFAULT 'def'"
,
"ALTER TABLE mytable ALTER bar SET DEFAULT 'def'"
,
'ALTER TABLE mytable ALTER bar SET NOT NULL'
,
'ALTER TABLE mytable ALTER bar SET NOT NULL'
,
'ALTER TABLE mytable ALTER bloo TYPE BOOLEAN'
,
'ALTER TABLE mytable ALTER bloo TYPE BOOLEAN'
,
"ALTER TABLE mytable ALTER bloo SET
DEFAULT 'false'"
,
"ALTER TABLE mytable ALTER bloo SET DEFAULT 'false'"
,
'ALTER TABLE mytable ALTER bloo SET NOT NULL'
,
'ALTER TABLE mytable ALTER bloo SET NOT NULL'
,
'ALTER TABLE mytable RENAME TO userlist'
,
'ALTER TABLE mytable RENAME TO userlist'
,
);
);
...
...
tests/Doctrine/Tests/DBAL/Platforms/SQLServerPlatformTest.php
View file @
17edd0d5
...
@@ -37,6 +37,7 @@ class SQLServerPlatformTest extends AbstractPlatformTestCase
...
@@ -37,6 +37,7 @@ class SQLServerPlatformTest extends AbstractPlatformTestCase
'ALTER TABLE mytable ALTER COLUMN baz NVARCHAR(255) NOT NULL'
,
'ALTER TABLE mytable ALTER COLUMN baz NVARCHAR(255) NOT NULL'
,
"ALTER TABLE mytable ADD CONSTRAINT DF_6B2BD609_78240498 DEFAULT 'def' FOR baz"
,
"ALTER TABLE mytable ADD CONSTRAINT DF_6B2BD609_78240498 DEFAULT 'def' FOR baz"
,
'ALTER TABLE mytable ALTER COLUMN bloo BIT NOT NULL'
,
'ALTER TABLE mytable ALTER COLUMN bloo BIT NOT NULL'
,
"ALTER TABLE mytable ADD CONSTRAINT DF_6B2BD609_CECED971 DEFAULT '0' FOR bloo"
,
"sp_RENAME 'mytable', 'userlist'"
,
"sp_RENAME 'mytable', 'userlist'"
,
"DECLARE @sql NVARCHAR(MAX) = N''; "
.
"DECLARE @sql NVARCHAR(MAX) = N''; "
.
"SELECT @sql += N'EXEC sp_rename N''' + dc.name + ''', N''' "
.
"SELECT @sql += N'EXEC sp_rename N''' + dc.name + ''', N''' "
.
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment