Compare commits
	
		
			16 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 9889d9492b | ||
|  | 823368aea8 | ||
|  | 6b1ad46cf1 | ||
|  | ce8c0b1f06 | ||
|  | 43922d3734 | ||
|  | 48c5318737 | ||
|  | 002221e557 | ||
|  | 141d7205bf | ||
|  | af4d4be19a | ||
|  | 3b4d9b47ce | ||
|  | 4b0c4ae7d0 | ||
|  | dc821d8a02 | ||
|  | d18a6b5be0 | ||
|  | 1e56a70f21 | ||
|  | ab1e1aab75 | ||
|  | 00dd04f97d | 
							
								
								
									
										11
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -2,6 +2,17 @@ | |||||||
|  |  | ||||||
| ## 0.2 | ## 0.2 | ||||||
|  |  | ||||||
|  | ### 0.2.5 | ||||||
|  |  | ||||||
|  | - Fix windows support. (#46) | ||||||
|  | - Support `db_constraint` in fk, m2m should manual define table with fk. (#52) | ||||||
|  |  | ||||||
|  | ### 0.2.4 | ||||||
|  |  | ||||||
|  | - Raise error with SQLite unsupported features. | ||||||
|  | - Fix Postgres alter table. (#48) | ||||||
|  | - Add `Rename` support. | ||||||
|  |  | ||||||
| ### 0.2.3 | ### 0.2.3 | ||||||
|  |  | ||||||
| - Fix tortoise ssl config. | - Fix tortoise ssl config. | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								Makefile
									
									
									
									
									
								
							| @@ -40,10 +40,10 @@ test_sqlite: | |||||||
| 	$(py_warn) TEST_DB=sqlite://:memory: py.test | 	$(py_warn) TEST_DB=sqlite://:memory: py.test | ||||||
|  |  | ||||||
| test_mysql: | test_mysql: | ||||||
| 	$(py_warn) TEST_DB="mysql://root:$(MYSQL_PASS)@$(MYSQL_HOST):$(MYSQL_PORT)/test_\{\}" py.test | 	$(py_warn) TEST_DB="mysql://root:$(MYSQL_PASS)@$(MYSQL_HOST):$(MYSQL_PORT)/test_\{\}" pytest -vv -s | ||||||
|  |  | ||||||
| test_postgres: | test_postgres: | ||||||
| 	$(py_warn) TEST_DB="postgres://postgres:$(POSTGRES_PASS)@$(POSTGRES_HOST):$(POSTGRES_PORT)/test_\{\}" py.test | 	$(py_warn) TEST_DB="postgres://postgres:$(POSTGRES_PASS)@$(POSTGRES_HOST):$(POSTGRES_PORT)/test_\{\}" pytest | ||||||
|  |  | ||||||
| testall: deps test_sqlite test_postgres test_mysql | testall: deps test_sqlite test_postgres test_mysql | ||||||
|  |  | ||||||
|   | |||||||
| @@ -109,7 +109,11 @@ Success migrate 1_202029051520102929_drop_column.json | |||||||
| ``` | ``` | ||||||
|  |  | ||||||
| Format of migrate filename is | Format of migrate filename is | ||||||
| `{version_num}_{datetime}_{name|update}.json` | `{version_num}_{datetime}_{name|update}.json`. | ||||||
|  |  | ||||||
|  | And if `aerich` guess you are renaming a column, it will ask `Rename {old_column} to {new_column} [True]`, you can choice `True` to rename column without column drop, or choice `False` to drop column then create. | ||||||
|  |  | ||||||
|  | If you use `MySQL`, only MySQL8.0+ support `rename..to` syntax. | ||||||
|  |  | ||||||
| ### Upgrade to latest version | ### Upgrade to latest version | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1 +1 @@ | |||||||
| __version__ = "0.2.3" | __version__ = "0.2.5" | ||||||
|   | |||||||
| @@ -1,12 +1,12 @@ | |||||||
| import functools | import asyncio | ||||||
| import json | import json | ||||||
| import os | import os | ||||||
| import sys | import sys | ||||||
| from configparser import ConfigParser | from configparser import ConfigParser | ||||||
| from enum import Enum | from functools import wraps | ||||||
|  |  | ||||||
| import asyncclick as click | import click | ||||||
| from asyncclick import Context, UsageError | from click import Context, UsageError | ||||||
| from tortoise import Tortoise, generate_schema_for_client | from tortoise import Tortoise, generate_schema_for_client | ||||||
| from tortoise.exceptions import OperationalError | from tortoise.exceptions import OperationalError | ||||||
| from tortoise.transactions import in_transaction | from tortoise.transactions import in_transaction | ||||||
| @@ -16,26 +16,19 @@ from aerich.migrate import Migrate | |||||||
| from aerich.utils import get_app_connection, get_app_connection_name, get_tortoise_config | from aerich.utils import get_app_connection, get_app_connection_name, get_tortoise_config | ||||||
|  |  | ||||||
| from . import __version__ | from . import __version__ | ||||||
|  | from .enums import Color | ||||||
| from .models import Aerich | from .models import Aerich | ||||||
|  |  | ||||||
|  |  | ||||||
| class Color(str, Enum): |  | ||||||
|     green = "green" |  | ||||||
|     red = "red" |  | ||||||
|     yellow = "yellow" |  | ||||||
|  |  | ||||||
|  |  | ||||||
| parser = ConfigParser() | parser = ConfigParser() | ||||||
|  |  | ||||||
|  |  | ||||||
| def close_db(func): | def coro(f): | ||||||
|     @functools.wraps(func) |     @wraps(f) | ||||||
|     async def close_db_inner(*args, **kwargs): |     def wrapper(*args, **kwargs): | ||||||
|         result = await func(*args, **kwargs) |         loop = asyncio.get_event_loop() | ||||||
|         await Tortoise.close_connections() |         return loop.run_until_complete(f(*args, **kwargs)) | ||||||
|         return result |  | ||||||
|  |  | ||||||
|     return close_db_inner |     return wrapper | ||||||
|  |  | ||||||
|  |  | ||||||
| @click.group(context_settings={"help_option_names": ["-h", "--help"]}) | @click.group(context_settings={"help_option_names": ["-h", "--help"]}) | ||||||
| @@ -52,6 +45,7 @@ def close_db(func): | |||||||
|     help="Name of section in .ini file to use for aerich config.", |     help="Name of section in .ini file to use for aerich config.", | ||||||
| ) | ) | ||||||
| @click.pass_context | @click.pass_context | ||||||
|  | @coro | ||||||
| async def cli(ctx: Context, config, app, name): | async def cli(ctx: Context, config, app, name): | ||||||
|     ctx.ensure_object(dict) |     ctx.ensure_object(dict) | ||||||
|     ctx.obj["config_file"] = config |     ctx.obj["config_file"] = config | ||||||
| @@ -81,12 +75,11 @@ async def cli(ctx: Context, config, app, name): | |||||||
| @cli.command(help="Generate migrate changes file.") | @cli.command(help="Generate migrate changes file.") | ||||||
| @click.option("--name", default="update", show_default=True, help="Migrate name.") | @click.option("--name", default="update", show_default=True, help="Migrate name.") | ||||||
| @click.pass_context | @click.pass_context | ||||||
| @close_db | @coro | ||||||
| async def migrate(ctx: Context, name): | async def migrate(ctx: Context, name): | ||||||
|     config = ctx.obj["config"] |     config = ctx.obj["config"] | ||||||
|     location = ctx.obj["location"] |     location = ctx.obj["location"] | ||||||
|     app = ctx.obj["app"] |     app = ctx.obj["app"] | ||||||
|  |  | ||||||
|     ret = await Migrate.migrate(name) |     ret = await Migrate.migrate(name) | ||||||
|     if not ret: |     if not ret: | ||||||
|         return click.secho("No changes detected", fg=Color.yellow) |         return click.secho("No changes detected", fg=Color.yellow) | ||||||
| @@ -96,7 +89,7 @@ async def migrate(ctx: Context, name): | |||||||
|  |  | ||||||
| @cli.command(help="Upgrade to latest version.") | @cli.command(help="Upgrade to latest version.") | ||||||
| @click.pass_context | @click.pass_context | ||||||
| @close_db | @coro | ||||||
| async def upgrade(ctx: Context): | async def upgrade(ctx: Context): | ||||||
|     config = ctx.obj["config"] |     config = ctx.obj["config"] | ||||||
|     app = ctx.obj["app"] |     app = ctx.obj["app"] | ||||||
| @@ -123,7 +116,7 @@ async def upgrade(ctx: Context): | |||||||
|  |  | ||||||
| @cli.command(help="Downgrade to previous version.") | @cli.command(help="Downgrade to previous version.") | ||||||
| @click.pass_context | @click.pass_context | ||||||
| @close_db | @coro | ||||||
| async def downgrade(ctx: Context): | async def downgrade(ctx: Context): | ||||||
|     app = ctx.obj["app"] |     app = ctx.obj["app"] | ||||||
|     config = ctx.obj["config"] |     config = ctx.obj["config"] | ||||||
| @@ -146,7 +139,7 @@ async def downgrade(ctx: Context): | |||||||
|  |  | ||||||
| @cli.command(help="Show current available heads in migrate location.") | @cli.command(help="Show current available heads in migrate location.") | ||||||
| @click.pass_context | @click.pass_context | ||||||
| @close_db | @coro | ||||||
| async def heads(ctx: Context): | async def heads(ctx: Context): | ||||||
|     app = ctx.obj["app"] |     app = ctx.obj["app"] | ||||||
|     versions = Migrate.get_all_version_files() |     versions = Migrate.get_all_version_files() | ||||||
| @@ -161,7 +154,7 @@ async def heads(ctx: Context): | |||||||
|  |  | ||||||
| @cli.command(help="List all migrate items.") | @cli.command(help="List all migrate items.") | ||||||
| @click.pass_context | @click.pass_context | ||||||
| @close_db | @coro | ||||||
| async def history(ctx: Context): | async def history(ctx: Context): | ||||||
|     versions = Migrate.get_all_version_files() |     versions = Migrate.get_all_version_files() | ||||||
|     for version in versions: |     for version in versions: | ||||||
| @@ -181,6 +174,7 @@ async def history(ctx: Context): | |||||||
|     "--location", default="./migrations", show_default=True, help="Migrate store location." |     "--location", default="./migrations", show_default=True, help="Migrate store location." | ||||||
| ) | ) | ||||||
| @click.pass_context | @click.pass_context | ||||||
|  | @coro | ||||||
| async def init( | async def init( | ||||||
|     ctx: Context, tortoise_orm, location, |     ctx: Context, tortoise_orm, location, | ||||||
| ): | ): | ||||||
| @@ -212,7 +206,7 @@ async def init( | |||||||
|     show_default=True, |     show_default=True, | ||||||
| ) | ) | ||||||
| @click.pass_context | @click.pass_context | ||||||
| @close_db | @coro | ||||||
| async def init_db(ctx: Context, safe): | async def init_db(ctx: Context, safe): | ||||||
|     config = ctx.obj["config"] |     config = ctx.obj["config"] | ||||||
|     location = ctx.obj["location"] |     location = ctx.obj["location"] | ||||||
| @@ -245,4 +239,4 @@ async def init_db(ctx: Context, safe): | |||||||
|  |  | ||||||
| def main(): | def main(): | ||||||
|     sys.path.insert(0, ".") |     sys.path.insert(0, ".") | ||||||
|     cli(_anyio_backend="asyncio") |     cli() | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ from typing import List, Type | |||||||
|  |  | ||||||
| from tortoise import BaseDBAsyncClient, ForeignKeyFieldInstance, ManyToManyFieldInstance, Model | from tortoise import BaseDBAsyncClient, ForeignKeyFieldInstance, ManyToManyFieldInstance, Model | ||||||
| from tortoise.backends.base.schema_generator import BaseSchemaGenerator | from tortoise.backends.base.schema_generator import BaseSchemaGenerator | ||||||
| from tortoise.fields import Field, JSONField, TextField, UUIDField | from tortoise.fields import CASCADE, Field, JSONField, TextField, UUIDField | ||||||
|  |  | ||||||
|  |  | ||||||
| class BaseDDL: | class BaseDDL: | ||||||
| @@ -11,13 +11,16 @@ class BaseDDL: | |||||||
|     _DROP_TABLE_TEMPLATE = 'DROP TABLE IF EXISTS "{table_name}"' |     _DROP_TABLE_TEMPLATE = 'DROP TABLE IF EXISTS "{table_name}"' | ||||||
|     _ADD_COLUMN_TEMPLATE = 'ALTER TABLE "{table_name}" ADD {column}' |     _ADD_COLUMN_TEMPLATE = 'ALTER TABLE "{table_name}" ADD {column}' | ||||||
|     _DROP_COLUMN_TEMPLATE = 'ALTER TABLE "{table_name}" DROP COLUMN "{column_name}"' |     _DROP_COLUMN_TEMPLATE = 'ALTER TABLE "{table_name}" DROP COLUMN "{column_name}"' | ||||||
|  |     _RENAME_COLUMN_TEMPLATE = ( | ||||||
|  |         'ALTER TABLE "{table_name}" RENAME COLUMN "{old_column_name}" TO "{new_column_name}"' | ||||||
|  |     ) | ||||||
|     _ADD_INDEX_TEMPLATE = ( |     _ADD_INDEX_TEMPLATE = ( | ||||||
|         'ALTER TABLE "{table_name}" ADD {unique} INDEX "{index_name}" ({column_names})' |         'ALTER TABLE "{table_name}" ADD {unique} INDEX "{index_name}" ({column_names})' | ||||||
|     ) |     ) | ||||||
|     _DROP_INDEX_TEMPLATE = 'ALTER TABLE "{table_name}" DROP INDEX "{index_name}"' |     _DROP_INDEX_TEMPLATE = 'ALTER TABLE "{table_name}" DROP INDEX "{index_name}"' | ||||||
|     _ADD_FK_TEMPLATE = 'ALTER TABLE "{table_name}" ADD CONSTRAINT "{fk_name}" FOREIGN KEY ("{db_column}") REFERENCES "{table}" ("{field}") ON DELETE {on_delete}' |     _ADD_FK_TEMPLATE = 'ALTER TABLE "{table_name}" ADD CONSTRAINT "{fk_name}" FOREIGN KEY ("{db_column}") REFERENCES "{table}" ("{field}") ON DELETE {on_delete}' | ||||||
|     _DROP_FK_TEMPLATE = 'ALTER TABLE "{table_name}" DROP FOREIGN KEY "{fk_name}"' |     _DROP_FK_TEMPLATE = 'ALTER TABLE "{table_name}" DROP FOREIGN KEY "{fk_name}"' | ||||||
|     _M2M_TABLE_TEMPLATE = 'CREATE TABLE "{table_name}" ("{backward_key}" {backward_type} NOT NULL REFERENCES "{backward_table}" ("{backward_field}") ON DELETE CASCADE,"{forward_key}" {forward_type} NOT NULL REFERENCES "{forward_table}" ("{forward_field}") ON DELETE CASCADE){extra}{comment};' |     _M2M_TABLE_TEMPLATE = 'CREATE TABLE "{table_name}" ("{backward_key}" {backward_type} NOT NULL REFERENCES "{backward_table}" ("{backward_field}") ON DELETE CASCADE,"{forward_key}" {forward_type} NOT NULL REFERENCES "{forward_table}" ("{forward_field}") ON DELETE {on_delete}){extra}{comment};' | ||||||
|     _MODIFY_COLUMN_TEMPLATE = 'ALTER TABLE "{table_name}" MODIFY COLUMN {column}' |     _MODIFY_COLUMN_TEMPLATE = 'ALTER TABLE "{table_name}" MODIFY COLUMN {column}' | ||||||
|  |  | ||||||
|     def __init__(self, client: "BaseDBAsyncClient"): |     def __init__(self, client: "BaseDBAsyncClient"): | ||||||
| @@ -41,6 +44,7 @@ class BaseDDL: | |||||||
|             backward_type=model._meta.pk.get_for_dialect(self.DIALECT, "SQL_TYPE"), |             backward_type=model._meta.pk.get_for_dialect(self.DIALECT, "SQL_TYPE"), | ||||||
|             forward_key=field.forward_key, |             forward_key=field.forward_key, | ||||||
|             forward_type=field.related_model._meta.pk.get_for_dialect(self.DIALECT, "SQL_TYPE"), |             forward_type=field.related_model._meta.pk.get_for_dialect(self.DIALECT, "SQL_TYPE"), | ||||||
|  |             on_delete=CASCADE, | ||||||
|             extra=self.schema_generator._table_generate_extra(table=field.through), |             extra=self.schema_generator._table_generate_extra(table=field.through), | ||||||
|             comment=self.schema_generator._table_comment_generator( |             comment=self.schema_generator._table_comment_generator( | ||||||
|                 table=field.through, comment=field.description |                 table=field.through, comment=field.description | ||||||
| @@ -125,6 +129,13 @@ class BaseDDL: | |||||||
|             ), |             ), | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|  |     def rename_column(self, model: "Type[Model]", old_column_name: str, new_column_name: str): | ||||||
|  |         return self._RENAME_COLUMN_TEMPLATE.format( | ||||||
|  |             table_name=model._meta.db_table, | ||||||
|  |             old_column_name=old_column_name, | ||||||
|  |             new_column_name=new_column_name, | ||||||
|  |         ) | ||||||
|  |  | ||||||
|     def add_index(self, model: "Type[Model]", field_names: List[str], unique=False): |     def add_index(self, model: "Type[Model]", field_names: List[str], unique=False): | ||||||
|         return self._ADD_INDEX_TEMPLATE.format( |         return self._ADD_INDEX_TEMPLATE.format( | ||||||
|             unique="UNIQUE" if unique else "", |             unique="UNIQUE" if unique else "", | ||||||
|   | |||||||
| @@ -9,6 +9,9 @@ class MysqlDDL(BaseDDL): | |||||||
|     _DROP_TABLE_TEMPLATE = "DROP TABLE IF EXISTS `{table_name}`" |     _DROP_TABLE_TEMPLATE = "DROP TABLE IF EXISTS `{table_name}`" | ||||||
|     _ADD_COLUMN_TEMPLATE = "ALTER TABLE `{table_name}` ADD {column}" |     _ADD_COLUMN_TEMPLATE = "ALTER TABLE `{table_name}` ADD {column}" | ||||||
|     _DROP_COLUMN_TEMPLATE = "ALTER TABLE `{table_name}` DROP COLUMN `{column_name}`" |     _DROP_COLUMN_TEMPLATE = "ALTER TABLE `{table_name}` DROP COLUMN `{column_name}`" | ||||||
|  |     _RENAME_COLUMN_TEMPLATE = ( | ||||||
|  |         "ALTER TABLE `{table_name}` RENAME COLUMN `{old_column_name}` TO `{new_column_name}`" | ||||||
|  |     ) | ||||||
|     _ADD_INDEX_TEMPLATE = ( |     _ADD_INDEX_TEMPLATE = ( | ||||||
|         "ALTER TABLE `{table_name}` ADD {unique} INDEX `{index_name}` ({column_names})" |         "ALTER TABLE `{table_name}` ADD {unique} INDEX `{index_name}` ({column_names})" | ||||||
|     ) |     ) | ||||||
|   | |||||||
| @@ -1,8 +1,19 @@ | |||||||
|  | from typing import Type | ||||||
|  |  | ||||||
|  | from tortoise import Model | ||||||
| from tortoise.backends.sqlite.schema_generator import SqliteSchemaGenerator | from tortoise.backends.sqlite.schema_generator import SqliteSchemaGenerator | ||||||
|  | from tortoise.fields import Field | ||||||
|  |  | ||||||
| from aerich.ddl import BaseDDL | from aerich.ddl import BaseDDL | ||||||
|  | from aerich.exceptions import NotSupportError | ||||||
|  |  | ||||||
|  |  | ||||||
| class SqliteDDL(BaseDDL): | class SqliteDDL(BaseDDL): | ||||||
|     schema_generator_cls = SqliteSchemaGenerator |     schema_generator_cls = SqliteSchemaGenerator | ||||||
|     DIALECT = SqliteSchemaGenerator.DIALECT |     DIALECT = SqliteSchemaGenerator.DIALECT | ||||||
|  |  | ||||||
|  |     def drop_column(self, model: "Type[Model]", column_name: str): | ||||||
|  |         raise NotSupportError("Drop column is not support in SQLite.") | ||||||
|  |  | ||||||
|  |     def modify_column(self, model: "Type[Model]", field_object: Field): | ||||||
|  |         raise NotSupportError("Modify column is not support in SQLite.") | ||||||
|   | |||||||
							
								
								
									
										7
									
								
								aerich/enums.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								aerich/enums.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | |||||||
|  | from enum import Enum | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Color(str, Enum): | ||||||
|  |     green = "green" | ||||||
|  |     red = "red" | ||||||
|  |     yellow = "yellow" | ||||||
| @@ -1,6 +1,4 @@ | |||||||
| class ConfigurationError(Exception): | class NotSupportError(Exception): | ||||||
|     """ |     """ | ||||||
|     config error |     raise when features not support | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     pass |  | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ from datetime import datetime | |||||||
| from importlib import import_module | from importlib import import_module | ||||||
| from typing import Dict, List, Tuple, Type | from typing import Dict, List, Tuple, Type | ||||||
|  |  | ||||||
|  | import click | ||||||
| from tortoise import ( | from tortoise import ( | ||||||
|     BackwardFKRelation, |     BackwardFKRelation, | ||||||
|     BackwardOneToOneRelation, |     BackwardOneToOneRelation, | ||||||
| @@ -28,6 +29,8 @@ class Migrate: | |||||||
|     _upgrade_m2m: List[str] = [] |     _upgrade_m2m: List[str] = [] | ||||||
|     _downgrade_m2m: List[str] = [] |     _downgrade_m2m: List[str] = [] | ||||||
|     _aerich = Aerich.__name__ |     _aerich = Aerich.__name__ | ||||||
|  |     _rename_old = [] | ||||||
|  |     _rename_new = [] | ||||||
|  |  | ||||||
|     ddl: BaseDDL |     ddl: BaseDDL | ||||||
|     migrate_config: dict |     migrate_config: dict | ||||||
| @@ -129,7 +132,7 @@ class Migrate: | |||||||
|         return await cls._generate_diff_sql(name) |         return await cls._generate_diff_sql(name) | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def _add_operator(cls, operator: str, upgrade=True, fk=False): |     def _add_operator(cls, operator: str, upgrade=True, fk_m2m=False): | ||||||
|         """ |         """ | ||||||
|         add operator,differentiate fk because fk is order limit |         add operator,differentiate fk because fk is order limit | ||||||
|         :param operator: |         :param operator: | ||||||
| @@ -138,12 +141,12 @@ class Migrate: | |||||||
|         :return: |         :return: | ||||||
|         """ |         """ | ||||||
|         if upgrade: |         if upgrade: | ||||||
|             if fk: |             if fk_m2m: | ||||||
|                 cls._upgrade_fk_m2m_index_operators.append(operator) |                 cls._upgrade_fk_m2m_index_operators.append(operator) | ||||||
|             else: |             else: | ||||||
|                 cls.upgrade_operators.append(operator) |                 cls.upgrade_operators.append(operator) | ||||||
|         else: |         else: | ||||||
|             if fk: |             if fk_m2m: | ||||||
|                 cls._downgrade_fk_m2m_index_operators.append(operator) |                 cls._downgrade_fk_m2m_index_operators.append(operator) | ||||||
|             else: |             else: | ||||||
|                 cls.downgrade_operators.append(operator) |                 cls.downgrade_operators.append(operator) | ||||||
| @@ -166,7 +169,7 @@ class Migrate: | |||||||
|             ret = re.sub(pattern, rf"\2{cls.diff_app}\4\5", content) |             ret = re.sub(pattern, rf"\2{cls.diff_app}\4\5", content) | ||||||
|             mode = "w" if i == 0 else "a" |             mode = "w" if i == 0 else "a" | ||||||
|             with open(old_model_file, mode, encoding="utf-8") as f: |             with open(old_model_file, mode, encoding="utf-8") as f: | ||||||
|                 f.write(ret) |                 f.write(f"{ret}\n") | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def _get_migrate_config(cls, config: dict, app: str, location: str): |     def _get_migrate_config(cls, config: dict, app: str, location: str): | ||||||
| @@ -264,11 +267,35 @@ class Migrate: | |||||||
|             if cls._exclude_field(new_field, upgrade): |             if cls._exclude_field(new_field, upgrade): | ||||||
|                 continue |                 continue | ||||||
|             if new_key not in old_keys: |             if new_key not in old_keys: | ||||||
|                 cls._add_operator( |                 new_field_dict = new_field.describe(serializable=True) | ||||||
|                     cls._add_field(new_model, new_field), |                 new_field_dict.pop("name", None) | ||||||
|                     upgrade, |                 new_field_dict.pop("db_column", None) | ||||||
|                     isinstance(new_field, (ForeignKeyFieldInstance, ManyToManyFieldInstance)), |                 for diff_key in old_keys - new_keys: | ||||||
|                 ) |                     old_field = old_fields_map.get(diff_key) | ||||||
|  |                     old_field_dict = old_field.describe(serializable=True) | ||||||
|  |                     old_field_dict.pop("name", None) | ||||||
|  |                     old_field_dict.pop("db_column", None) | ||||||
|  |                     if old_field_dict == new_field_dict: | ||||||
|  |                         if upgrade: | ||||||
|  |                             is_rename = click.prompt( | ||||||
|  |                                 f"Rename {diff_key} to {new_key}?", | ||||||
|  |                                 default=True, | ||||||
|  |                                 type=bool, | ||||||
|  |                                 show_choices=True, | ||||||
|  |                             ) | ||||||
|  |                             cls._rename_new.append(new_key) | ||||||
|  |                             cls._rename_old.append(diff_key) | ||||||
|  |                         else: | ||||||
|  |                             is_rename = diff_key in cls._rename_new | ||||||
|  |                         if is_rename: | ||||||
|  |                             cls._add_operator( | ||||||
|  |                                 cls._rename_field(new_model, old_field, new_field), upgrade, | ||||||
|  |                             ) | ||||||
|  |                             break | ||||||
|  |                 else: | ||||||
|  |                     cls._add_operator( | ||||||
|  |                         cls._add_field(new_model, new_field), upgrade, cls._is_fk_m2m(new_field), | ||||||
|  |                     ) | ||||||
|             else: |             else: | ||||||
|                 old_field = old_fields_map.get(new_key) |                 old_field = old_fields_map.get(new_key) | ||||||
|                 new_field_dict = new_field.describe(serializable=True) |                 new_field_dict = new_field.describe(serializable=True) | ||||||
| @@ -291,7 +318,12 @@ class Migrate: | |||||||
|                             cls._add_operator( |                             cls._add_operator( | ||||||
|                                 cls._set_comment(new_model, new_field), upgrade=upgrade |                                 cls._set_comment(new_model, new_field), upgrade=upgrade | ||||||
|                             ) |                             ) | ||||||
|                     cls._add_operator(cls._modify_field(new_model, new_field), upgrade=upgrade) |                         if new_field.field_type != old_field.field_type: | ||||||
|  |                             cls._add_operator( | ||||||
|  |                                 cls._modify_field(new_model, new_field), upgrade=upgrade | ||||||
|  |                             ) | ||||||
|  |                     else: | ||||||
|  |                         cls._add_operator(cls._modify_field(new_model, new_field), upgrade=upgrade) | ||||||
|                 if (old_field.index and not new_field.index) or ( |                 if (old_field.index and not new_field.index) or ( | ||||||
|                     old_field.unique and not new_field.unique |                     old_field.unique and not new_field.unique | ||||||
|                 ): |                 ): | ||||||
| @@ -310,13 +342,25 @@ class Migrate: | |||||||
|                         upgrade, |                         upgrade, | ||||||
|                         cls._is_fk_m2m(new_field), |                         cls._is_fk_m2m(new_field), | ||||||
|                     ) |                     ) | ||||||
|  |                 if isinstance(new_field, ForeignKeyFieldInstance): | ||||||
|  |                     if old_field.db_constraint and not new_field.db_constraint: | ||||||
|  |                         cls._add_operator( | ||||||
|  |                             cls._drop_fk(new_model, new_field), upgrade, True, | ||||||
|  |                         ) | ||||||
|  |                     if new_field.db_constraint and not old_field.db_constraint: | ||||||
|  |                         cls._add_operator( | ||||||
|  |                             cls._add_fk(new_model, new_field), upgrade, True, | ||||||
|  |                         ) | ||||||
|  |  | ||||||
|         for old_key in old_keys: |         for old_key in old_keys: | ||||||
|             field = old_fields_map.get(old_key) |             field = old_fields_map.get(old_key) | ||||||
|             if old_key not in new_keys and not cls._exclude_field(field, upgrade): |             if old_key not in new_keys and not cls._exclude_field(field, upgrade): | ||||||
|                 cls._add_operator( |                 if (upgrade and old_key not in cls._rename_old) or ( | ||||||
|                     cls._remove_field(old_model, field), upgrade, cls._is_fk_m2m(field), |                     not upgrade and old_key not in cls._rename_new | ||||||
|                 ) |                 ): | ||||||
|  |                     cls._add_operator( | ||||||
|  |                         cls._remove_field(old_model, field), upgrade, cls._is_fk_m2m(field), | ||||||
|  |                     ) | ||||||
|  |  | ||||||
|         for new_index in new_indexes: |         for new_index in new_indexes: | ||||||
|             if new_index not in old_indexes: |             if new_index not in old_indexes: | ||||||
| @@ -400,6 +444,10 @@ class Migrate: | |||||||
|     def _modify_field(cls, model: Type[Model], field: Field): |     def _modify_field(cls, model: Type[Model], field: Field): | ||||||
|         return cls.ddl.modify_column(model, field) |         return cls.ddl.modify_column(model, field) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _drop_fk(cls, model: Type[Model], field: ForeignKeyFieldInstance): | ||||||
|  |         return cls.ddl.drop_fk(model, field) | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def _remove_field(cls, model: Type[Model], field: Field): |     def _remove_field(cls, model: Type[Model], field: Field): | ||||||
|         if isinstance(field, ForeignKeyFieldInstance): |         if isinstance(field, ForeignKeyFieldInstance): | ||||||
| @@ -408,6 +456,10 @@ class Migrate: | |||||||
|             return cls.ddl.drop_m2m(field) |             return cls.ddl.drop_m2m(field) | ||||||
|         return cls.ddl.drop_column(model, field.model_field_name) |         return cls.ddl.drop_column(model, field.model_field_name) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _rename_field(cls, model: Type[Model], old_field: Field, new_field: Field): | ||||||
|  |         return cls.ddl.rename_column(model, old_field.model_field_name, new_field.model_field_name) | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def _add_fk(cls, model: Type[Model], field: ForeignKeyFieldInstance): |     def _add_fk(cls, model: Type[Model], field: ForeignKeyFieldInstance): | ||||||
|         """ |         """ | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| import importlib | import importlib | ||||||
|  |  | ||||||
| from asyncclick import BadOptionUsage, Context | from click import BadOptionUsage, Context | ||||||
| from tortoise import BaseDBAsyncClient, Tortoise | from tortoise import BaseDBAsyncClient, Tortoise | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										25
									
								
								conftest.py
									
									
									
									
									
								
							
							
						
						
									
										25
									
								
								conftest.py
									
									
									
									
									
								
							| @@ -31,24 +31,29 @@ def reset_migrate(): | |||||||
|     Migrate._downgrade_m2m = [] |     Migrate._downgrade_m2m = [] | ||||||
|  |  | ||||||
|  |  | ||||||
| @pytest.fixture(scope="session") | @pytest.yield_fixture(scope="session") | ||||||
| def loop(): | def event_loop(): | ||||||
|     loop = asyncio.get_event_loop() |     policy = asyncio.get_event_loop_policy() | ||||||
|     return loop |     res = policy.new_event_loop() | ||||||
|  |     asyncio.set_event_loop(res) | ||||||
|  |     res._close = res.close | ||||||
|  |     res.close = lambda: None | ||||||
|  |  | ||||||
|  |     yield res | ||||||
|  |  | ||||||
|  |     res._close() | ||||||
|  |  | ||||||
|  |  | ||||||
| @pytest.fixture(scope="session", autouse=True) | @pytest.fixture(scope="session", autouse=True) | ||||||
| def initialize_tests(loop, request): | async def initialize_tests(event_loop, request): | ||||||
|     tortoise_orm["connections"]["diff_models"] = "sqlite://:memory:" |     tortoise_orm["connections"]["diff_models"] = "sqlite://:memory:" | ||||||
|     tortoise_orm["apps"]["diff_models"] = { |     tortoise_orm["apps"]["diff_models"] = { | ||||||
|         "models": ["tests.diff_models"], |         "models": ["tests.diff_models"], | ||||||
|         "default_connection": "diff_models", |         "default_connection": "diff_models", | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     loop.run_until_complete(Tortoise.init(config=tortoise_orm, _create_db=True)) |     await Tortoise.init(config=tortoise_orm, _create_db=True) | ||||||
|     loop.run_until_complete( |     await generate_schema_for_client(Tortoise.get_connection("default"), safe=True) | ||||||
|         generate_schema_for_client(Tortoise.get_connection("default"), safe=True) |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     client = Tortoise.get_connection("default") |     client = Tortoise.get_connection("default") | ||||||
|     if client.schema_generator is MySQLSchemaGenerator: |     if client.schema_generator is MySQLSchemaGenerator: | ||||||
| @@ -58,4 +63,4 @@ def initialize_tests(loop, request): | |||||||
|     elif client.schema_generator is AsyncpgSchemaGenerator: |     elif client.schema_generator is AsyncpgSchemaGenerator: | ||||||
|         Migrate.ddl = PostgresDDL(client) |         Migrate.ddl = PostgresDDL(client) | ||||||
|  |  | ||||||
|     request.addfinalizer(lambda: loop.run_until_complete(Tortoise._drop_databases())) |     request.addfinalizer(lambda: event_loop.run_until_complete(Tortoise._drop_databases())) | ||||||
|   | |||||||
							
								
								
									
										355
									
								
								poetry.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										355
									
								
								poetry.lock
									
									
									
										generated
									
									
									
								
							| @@ -23,25 +23,6 @@ version = "0.15.0" | |||||||
| [package.dependencies] | [package.dependencies] | ||||||
| typing_extensions = "*" | typing_extensions = "*" | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| category = "main" |  | ||||||
| description = "High level compatibility layer for multiple asynchronous event loop implementations" |  | ||||||
| name = "anyio" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=3.5.3" |  | ||||||
| version = "1.4.0" |  | ||||||
|  |  | ||||||
| [package.dependencies] |  | ||||||
| async-generator = "*" |  | ||||||
| idna = ">=2.8" |  | ||||||
| sniffio = ">=1.1" |  | ||||||
|  |  | ||||||
| [package.extras] |  | ||||||
| curio = ["curio (>=0.9)"] |  | ||||||
| doc = ["sphinx-rtd-theme", "sphinx-autodoc-typehints (>=1.2.0)"] |  | ||||||
| test = ["coverage (>=4.5)", "hypothesis (>=4.0)", "pytest (>=3.7.2)", "uvloop"] |  | ||||||
| trio = ["trio (>=0.12)"] |  | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| category = "dev" | category = "dev" | ||||||
| description = "apipkg: namespace control and lazy-import mechanism" | description = "apipkg: namespace control and lazy-import mechanism" | ||||||
| @@ -58,39 +39,16 @@ optional = false | |||||||
| python-versions = "*" | python-versions = "*" | ||||||
| version = "1.4.4" | version = "1.4.4" | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| category = "main" |  | ||||||
| description = "Async generators and context managers for Python 3.5+" |  | ||||||
| name = "async-generator" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=3.5" |  | ||||||
| version = "1.10" |  | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| category = "main" |  | ||||||
| description = "A simple anyio-compatible fork of Click, for powerful command line utilities." |  | ||||||
| name = "asyncclick" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=3.6" |  | ||||||
| version = "7.0.9" |  | ||||||
|  |  | ||||||
| [package.dependencies] |  | ||||||
| anyio = "*" |  | ||||||
|  |  | ||||||
| [package.extras] |  | ||||||
| dev = ["coverage", "pytest-runner", "pytest-trio", "pytest (>=3)", "sphinx", "tox"] |  | ||||||
| docs = ["sphinx"] |  | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| category = "main" | category = "main" | ||||||
| description = "An asyncio PostgreSQL driver" | description = "An asyncio PostgreSQL driver" | ||||||
| name = "asyncpg" | name = "asyncpg" | ||||||
| optional = true | optional = true | ||||||
| python-versions = ">=3.5.0" | python-versions = ">=3.5.0" | ||||||
| version = "0.20.1" | version = "0.21.0" | ||||||
|  |  | ||||||
| [package.extras] | [package.extras] | ||||||
| dev = ["Cython (0.29.14)", "pytest (>=3.6.0)", "Sphinx (>=1.7.3,<1.8.0)", "sphinxcontrib-asyncio (>=0.2.0,<0.3.0)", "sphinx-rtd-theme (>=0.2.4,<0.3.0)", "pycodestyle (>=2.5.0,<2.6.0)", "flake8 (>=3.7.9,<3.8.0)", "uvloop (>=0.14.0,<0.15.0)"] | dev = ["Cython (0.29.20)", "pytest (>=3.6.0)", "Sphinx (>=1.7.3,<1.8.0)", "sphinxcontrib-asyncio (>=0.2.0,<0.3.0)", "sphinx-rtd-theme (>=0.2.4,<0.3.0)", "pycodestyle (>=2.5.0,<2.6.0)", "flake8 (>=3.7.9,<3.8.0)", "uvloop (>=0.14.0,<0.15.0)"] | ||||||
| docs = ["Sphinx (>=1.7.3,<1.8.0)", "sphinxcontrib-asyncio (>=0.2.0,<0.3.0)", "sphinx-rtd-theme (>=0.2.4,<0.3.0)"] | docs = ["Sphinx (>=1.7.3,<1.8.0)", "sphinxcontrib-asyncio (>=0.2.0,<0.3.0)", "sphinx-rtd-theme (>=0.2.4,<0.3.0)"] | ||||||
| test = ["pycodestyle (>=2.5.0,<2.6.0)", "flake8 (>=3.7.9,<3.8.0)", "uvloop (>=0.14.0,<0.15.0)"] | test = ["pycodestyle (>=2.5.0,<2.6.0)", "flake8 (>=3.7.9,<3.8.0)", "uvloop (>=0.14.0,<0.15.0)"] | ||||||
|  |  | ||||||
| @@ -109,13 +67,13 @@ description = "Classes Without Boilerplate" | |||||||
| name = "attrs" | name = "attrs" | ||||||
| optional = false | optional = false | ||||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" | python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" | ||||||
| version = "19.3.0" | version = "20.2.0" | ||||||
|  |  | ||||||
| [package.extras] | [package.extras] | ||||||
| azure-pipelines = ["coverage", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "zope.interface", "pytest-azurepipelines"] | dev = ["coverage (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "zope.interface", "sphinx", "sphinx-rtd-theme", "pre-commit"] | ||||||
| dev = ["coverage", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "zope.interface", "sphinx", "pre-commit"] | docs = ["sphinx", "sphinx-rtd-theme", "zope.interface"] | ||||||
| docs = ["sphinx", "zope.interface"] | tests = ["coverage (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "zope.interface"] | ||||||
| tests = ["coverage", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "zope.interface"] | tests_no_zope = ["coverage (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six"] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| category = "dev" | category = "dev" | ||||||
| @@ -158,13 +116,13 @@ description = "Foreign Function Interface for Python calling C code." | |||||||
| name = "cffi" | name = "cffi" | ||||||
| optional = true | optional = true | ||||||
| python-versions = "*" | python-versions = "*" | ||||||
| version = "1.14.1" | version = "1.14.3" | ||||||
|  |  | ||||||
| [package.dependencies] | [package.dependencies] | ||||||
| pycparser = "*" | pycparser = "*" | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| category = "dev" | category = "main" | ||||||
| description = "Composable command line interface toolkit" | description = "Composable command line interface toolkit" | ||||||
| name = "click" | name = "click" | ||||||
| optional = false | optional = false | ||||||
| @@ -186,7 +144,7 @@ description = "cryptography is a package which provides cryptographic recipes an | |||||||
| name = "cryptography" | name = "cryptography" | ||||||
| optional = true | optional = true | ||||||
| python-versions = ">=2.7,!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*" | python-versions = ">=2.7,!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*" | ||||||
| version = "3.0" | version = "3.1.1" | ||||||
|  |  | ||||||
| [package.dependencies] | [package.dependencies] | ||||||
| cffi = ">=1.8,<1.11.3 || >1.11.3" | cffi = ">=1.8,<1.11.3 || >1.11.3" | ||||||
| @@ -195,7 +153,6 @@ six = ">=1.4.1" | |||||||
| [package.extras] | [package.extras] | ||||||
| docs = ["sphinx (>=1.6.5,<1.8.0 || >1.8.0,<3.1.0 || >3.1.0,<3.1.1 || >3.1.1)", "sphinx-rtd-theme"] | docs = ["sphinx (>=1.6.5,<1.8.0 || >1.8.0,<3.1.0 || >3.1.0,<3.1.1 || >3.1.1)", "sphinx-rtd-theme"] | ||||||
| docstest = ["doc8", "pyenchant (>=1.6.11)", "twine (>=1.12.0)", "sphinxcontrib-spelling (>=4.0.1)"] | docstest = ["doc8", "pyenchant (>=1.6.11)", "twine (>=1.12.0)", "sphinxcontrib-spelling (>=4.0.1)"] | ||||||
| idna = ["idna (>=2.1)"] |  | ||||||
| pep8test = ["black", "flake8", "flake8-import-order", "pep8-naming"] | pep8test = ["black", "flake8", "flake8-import-order", "pep8-naming"] | ||||||
| ssh = ["bcrypt (>=3.1.5)"] | ssh = ["bcrypt (>=3.1.5)"] | ||||||
| test = ["pytest (>=3.6.0,<3.9.0 || >3.9.0,<3.9.1 || >3.9.1,<3.9.2 || >3.9.2)", "pretend", "iso8601", "pytz", "hypothesis (>=1.11.4,<3.79.2 || >3.79.2)"] | test = ["pytest (>=3.6.0,<3.9.0 || >3.9.0,<3.9.1 || >3.9.1,<3.9.2 || >3.9.2)", "pretend", "iso8601", "pytz", "hypothesis (>=1.11.4,<3.79.2 || >3.79.2)"] | ||||||
| @@ -248,19 +205,11 @@ description = "Python Git Library" | |||||||
| name = "gitpython" | name = "gitpython" | ||||||
| optional = false | optional = false | ||||||
| python-versions = ">=3.4" | python-versions = ">=3.4" | ||||||
| version = "3.1.7" | version = "3.1.8" | ||||||
|  |  | ||||||
| [package.dependencies] | [package.dependencies] | ||||||
| gitdb = ">=4.0.1,<5" | gitdb = ">=4.0.1,<5" | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| category = "main" |  | ||||||
| description = "Internationalized Domain Names in Applications (IDNA)" |  | ||||||
| name = "idna" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" |  | ||||||
| version = "2.10" |  | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| category = "dev" | category = "dev" | ||||||
| description = "Read metadata from Python packages" | description = "Read metadata from Python packages" | ||||||
| @@ -268,7 +217,7 @@ marker = "python_version < \"3.8\"" | |||||||
| name = "importlib-metadata" | name = "importlib-metadata" | ||||||
| optional = false | optional = false | ||||||
| python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,>=2.7" | python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,>=2.7" | ||||||
| version = "1.7.0" | version = "2.0.0" | ||||||
|  |  | ||||||
| [package.dependencies] | [package.dependencies] | ||||||
| zipp = ">=0.5" | zipp = ">=0.5" | ||||||
| @@ -291,7 +240,7 @@ description = "Simple module to parse ISO 8601 dates" | |||||||
| name = "iso8601" | name = "iso8601" | ||||||
| optional = false | optional = false | ||||||
| python-versions = "*" | python-versions = "*" | ||||||
| version = "0.1.12" | version = "0.1.13" | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| category = "dev" | category = "dev" | ||||||
| @@ -299,11 +248,11 @@ description = "A Python utility / library to sort Python imports." | |||||||
| name = "isort" | name = "isort" | ||||||
| optional = false | optional = false | ||||||
| python-versions = ">=3.6,<4.0" | python-versions = ">=3.6,<4.0" | ||||||
| version = "5.3.2" | version = "5.5.3" | ||||||
|  |  | ||||||
| [package.extras] | [package.extras] | ||||||
| colors = ["colorama (>=0.4.3,<0.5.0)"] | colors = ["colorama (>=0.4.3,<0.5.0)"] | ||||||
| pipfile_deprecated_finder = ["pipreqs", "requirementslib", "tomlkit (>=0.5.3)"] | pipfile_deprecated_finder = ["pipreqs", "requirementslib"] | ||||||
| requirements_deprecated_finder = ["pipreqs", "pip-api"] | requirements_deprecated_finder = ["pipreqs", "pip-api"] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| @@ -320,7 +269,7 @@ description = "More routines for operating on iterables, beyond itertools" | |||||||
| name = "more-itertools" | name = "more-itertools" | ||||||
| optional = false | optional = false | ||||||
| python-versions = ">=3.5" | python-versions = ">=3.5" | ||||||
| version = "8.4.0" | version = "8.5.0" | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| category = "dev" | category = "dev" | ||||||
| @@ -347,8 +296,8 @@ category = "dev" | |||||||
| description = "Python Build Reasonableness" | description = "Python Build Reasonableness" | ||||||
| name = "pbr" | name = "pbr" | ||||||
| optional = false | optional = false | ||||||
| python-versions = "*" | python-versions = ">=2.6" | ||||||
| version = "5.4.5" | version = "5.5.0" | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| category = "dev" | category = "dev" | ||||||
| @@ -436,7 +385,7 @@ description = "A SQL query builder API for Python" | |||||||
| name = "pypika" | name = "pypika" | ||||||
| optional = false | optional = false | ||||||
| python-versions = "*" | python-versions = "*" | ||||||
| version = "0.38.0" | version = "0.40.0" | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| category = "dev" | category = "dev" | ||||||
| @@ -444,7 +393,7 @@ description = "pytest: simple powerful testing with Python" | |||||||
| name = "pytest" | name = "pytest" | ||||||
| optional = false | optional = false | ||||||
| python-versions = ">=3.5" | python-versions = ">=3.5" | ||||||
| version = "6.0.1" | version = "6.0.2" | ||||||
|  |  | ||||||
| [package.dependencies] | [package.dependencies] | ||||||
| atomicwrites = ">=1.0" | atomicwrites = ">=1.0" | ||||||
| @@ -497,10 +446,10 @@ description = "Thin-wrapper around the mock package for easier use with pytest" | |||||||
| name = "pytest-mock" | name = "pytest-mock" | ||||||
| optional = false | optional = false | ||||||
| python-versions = ">=3.5" | python-versions = ">=3.5" | ||||||
| version = "3.2.0" | version = "3.3.1" | ||||||
|  |  | ||||||
| [package.dependencies] | [package.dependencies] | ||||||
| pytest = ">=2.7" | pytest = ">=5.0" | ||||||
|  |  | ||||||
| [package.extras] | [package.extras] | ||||||
| dev = ["pre-commit", "tox", "pytest-asyncio"] | dev = ["pre-commit", "tox", "pytest-asyncio"] | ||||||
| @@ -510,16 +459,16 @@ category = "dev" | |||||||
| description = "pytest xdist plugin for distributed testing and loop-on-failing modes" | description = "pytest xdist plugin for distributed testing and loop-on-failing modes" | ||||||
| name = "pytest-xdist" | name = "pytest-xdist" | ||||||
| optional = false | optional = false | ||||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" | python-versions = ">=3.5" | ||||||
| version = "1.34.0" | version = "2.1.0" | ||||||
|  |  | ||||||
| [package.dependencies] | [package.dependencies] | ||||||
| execnet = ">=1.1" | execnet = ">=1.1" | ||||||
| pytest = ">=4.4.0" | pytest = ">=6.0.0" | ||||||
| pytest-forked = "*" | pytest-forked = "*" | ||||||
| six = "*" |  | ||||||
|  |  | ||||||
| [package.extras] | [package.extras] | ||||||
|  | psutil = ["psutil (>=3.0)"] | ||||||
| testing = ["filelock"] | testing = ["filelock"] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| @@ -554,21 +503,13 @@ optional = false | |||||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" | python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" | ||||||
| version = "3.0.4" | version = "3.0.4" | ||||||
|  |  | ||||||
| [[package]] |  | ||||||
| category = "main" |  | ||||||
| description = "Sniff out which async library your code is running under" |  | ||||||
| name = "sniffio" |  | ||||||
| optional = false |  | ||||||
| python-versions = ">=3.5" |  | ||||||
| version = "1.1.0" |  | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| category = "dev" | category = "dev" | ||||||
| description = "Manage dynamic plugins for Python applications" | description = "Manage dynamic plugins for Python applications" | ||||||
| name = "stevedore" | name = "stevedore" | ||||||
| optional = false | optional = false | ||||||
| python-versions = ">=3.6" | python-versions = ">=3.6" | ||||||
| version = "3.2.0" | version = "3.2.2" | ||||||
|  |  | ||||||
| [package.dependencies] | [package.dependencies] | ||||||
| pbr = ">=2.0.0,<2.1.0 || >2.1.0" | pbr = ">=2.0.0,<2.1.0 || >2.1.0" | ||||||
| @@ -591,12 +532,12 @@ description = "Easy async ORM for python, built with relations in mind" | |||||||
| name = "tortoise-orm" | name = "tortoise-orm" | ||||||
| optional = false | optional = false | ||||||
| python-versions = "*" | python-versions = "*" | ||||||
| version = "0.16.14" | version = "0.16.16" | ||||||
|  |  | ||||||
| [package.dependencies] | [package.dependencies] | ||||||
| aiosqlite = ">=0.11.0" | aiosqlite = ">=0.11.0" | ||||||
| iso8601 = ">=0.1.12" | iso8601 = ">=0.1.12" | ||||||
| pypika = ">=0.36.5" | pypika = ">=0.39.0" | ||||||
| typing-extensions = ">=3.7" | typing-extensions = ">=3.7" | ||||||
|  |  | ||||||
| [package.extras] | [package.extras] | ||||||
| @@ -616,7 +557,7 @@ description = "Backported and Experimental Type Hints for Python 3.5+" | |||||||
| name = "typing-extensions" | name = "typing-extensions" | ||||||
| optional = false | optional = false | ||||||
| python-versions = "*" | python-versions = "*" | ||||||
| version = "3.7.4.2" | version = "3.7.4.3" | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| category = "dev" | category = "dev" | ||||||
| @@ -625,17 +566,17 @@ marker = "python_version < \"3.8\"" | |||||||
| name = "zipp" | name = "zipp" | ||||||
| optional = false | optional = false | ||||||
| python-versions = ">=3.6" | python-versions = ">=3.6" | ||||||
| version = "3.1.0" | version = "3.2.0" | ||||||
|  |  | ||||||
| [package.extras] | [package.extras] | ||||||
| docs = ["sphinx", "jaraco.packaging (>=3.2)", "rst.linker (>=1.9)"] | docs = ["sphinx", "jaraco.packaging (>=3.2)", "rst.linker (>=1.9)"] | ||||||
| testing = ["jaraco.itertools", "func-timeout"] | testing = ["pytest (>=3.5,<3.7.3 || >3.7.3)", "pytest-checkdocs (>=1.2.3)", "pytest-flake8", "pytest-cov", "jaraco.test (>=3.2.0)", "jaraco.itertools", "func-timeout", "pytest-black (>=0.3.7)", "pytest-mypy"] | ||||||
|  |  | ||||||
| [extras] | [extras] | ||||||
| dbdrivers = ["aiomysql", "asyncpg"] | dbdrivers = ["aiomysql", "asyncpg"] | ||||||
|  |  | ||||||
| [metadata] | [metadata] | ||||||
| content-hash = "06f00778f783c4ad5b174a9c9ee80f4f0e38db9da9ff1012f09c7d306eaa0975" | content-hash = "43fe9c0036f4d55d38f82c263887d8a7d9a35a597e02036b70a631955ff73149" | ||||||
| lock-version = "1.0" | lock-version = "1.0" | ||||||
| python-versions = "^3.7" | python-versions = "^3.7" | ||||||
|  |  | ||||||
| @@ -648,10 +589,6 @@ aiosqlite = [ | |||||||
|     {file = "aiosqlite-0.15.0-py3-none-any.whl", hash = "sha256:19b984b6702aed9f1c85c023f37296954547fc4030dae8e9d027b2a930bed78b"}, |     {file = "aiosqlite-0.15.0-py3-none-any.whl", hash = "sha256:19b984b6702aed9f1c85c023f37296954547fc4030dae8e9d027b2a930bed78b"}, | ||||||
|     {file = "aiosqlite-0.15.0.tar.gz", hash = "sha256:a2884793f4dc8f2798d90e1dfecb2b56a6d479cf039f7ec52356a7fd5f3bdc57"}, |     {file = "aiosqlite-0.15.0.tar.gz", hash = "sha256:a2884793f4dc8f2798d90e1dfecb2b56a6d479cf039f7ec52356a7fd5f3bdc57"}, | ||||||
| ] | ] | ||||||
| anyio = [ |  | ||||||
|     {file = "anyio-1.4.0-py3-none-any.whl", hash = "sha256:9ee67e8131853f42957e214d4531cee6f2b66dda164a298d9686a768b7161a4f"}, |  | ||||||
|     {file = "anyio-1.4.0.tar.gz", hash = "sha256:95f60964fc4583f3f226f8dc275dfb02aefe7b39b85a999c6d14f4ec5323c1d8"}, |  | ||||||
| ] |  | ||||||
| apipkg = [ | apipkg = [ | ||||||
|     {file = "apipkg-1.5-py2.py3-none-any.whl", hash = "sha256:58587dd4dc3daefad0487f6d9ae32b4542b185e1c36db6993290e7c41ca2b47c"}, |     {file = "apipkg-1.5-py2.py3-none-any.whl", hash = "sha256:58587dd4dc3daefad0487f6d9ae32b4542b185e1c36db6993290e7c41ca2b47c"}, | ||||||
|     {file = "apipkg-1.5.tar.gz", hash = "sha256:37228cda29411948b422fae072f57e31d3396d2ee1c9783775980ee9c9990af6"}, |     {file = "apipkg-1.5.tar.gz", hash = "sha256:37228cda29411948b422fae072f57e31d3396d2ee1c9783775980ee9c9990af6"}, | ||||||
| @@ -660,43 +597,40 @@ appdirs = [ | |||||||
|     {file = "appdirs-1.4.4-py2.py3-none-any.whl", hash = "sha256:a841dacd6b99318a741b166adb07e19ee71a274450e68237b4650ca1055ab128"}, |     {file = "appdirs-1.4.4-py2.py3-none-any.whl", hash = "sha256:a841dacd6b99318a741b166adb07e19ee71a274450e68237b4650ca1055ab128"}, | ||||||
|     {file = "appdirs-1.4.4.tar.gz", hash = "sha256:7d5d0167b2b1ba821647616af46a749d1c653740dd0d2415100fe26e27afdf41"}, |     {file = "appdirs-1.4.4.tar.gz", hash = "sha256:7d5d0167b2b1ba821647616af46a749d1c653740dd0d2415100fe26e27afdf41"}, | ||||||
| ] | ] | ||||||
| async-generator = [ |  | ||||||
|     {file = "async_generator-1.10-py3-none-any.whl", hash = "sha256:01c7bf666359b4967d2cda0000cc2e4af16a0ae098cbffcb8472fb9e8ad6585b"}, |  | ||||||
|     {file = "async_generator-1.10.tar.gz", hash = "sha256:6ebb3d106c12920aaae42ccb6f787ef5eefdcdd166ea3d628fa8476abe712144"}, |  | ||||||
| ] |  | ||||||
| asyncclick = [ |  | ||||||
|     {file = "asyncclick-7.0.9.tar.gz", hash = "sha256:62cebf3eca36d973802e2dd521ca1db11c5bf4544e9795e093d1a53cb688a8c2"}, |  | ||||||
| ] |  | ||||||
| asyncpg = [ | asyncpg = [ | ||||||
|     {file = "asyncpg-0.20.1-cp35-cp35m-macosx_10_13_x86_64.whl", hash = "sha256:f7184689177eeb5a11fa1b2baf3f6f2e26bfd7a85acf4de1a3adbd0867d7c0e2"}, |     {file = "asyncpg-0.21.0-cp35-cp35m-macosx_10_13_x86_64.whl", hash = "sha256:09badce47a4645cfe523cc8a182bd047d5d62af0caaea77935e6a3c9e77dc364"}, | ||||||
|     {file = "asyncpg-0.20.1-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:f0c9719ac00615f097fe91082b785bce36dbf02a5ec4115ede0ebfd2cd9500cb"}, |     {file = "asyncpg-0.21.0-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:6b7807bfedd24dd15cfb2c17c60977ce01410615ecc285268b5144a944ec97ff"}, | ||||||
|     {file = "asyncpg-0.20.1-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:1388caa456070dab102be874205e3ae8fd1de2577d5de9fa22e65ba5c0f8b110"}, |     {file = "asyncpg-0.21.0-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:dfd491e9865e64a3e91f1587b1d88d71dde1cfb850429253a73d4d44b98c3a0f"}, | ||||||
|     {file = "asyncpg-0.20.1-cp35-cp35m-win32.whl", hash = "sha256:ec6e7046c98730cb2ba4df41387e10cb8963a3ac2918f69ae416f8aab9ca7b1b"}, |     {file = "asyncpg-0.21.0-cp35-cp35m-manylinux2014_aarch64.whl", hash = "sha256:8587e206d78e739ca83a40c9982e03b28f8904c95a54dc782da99e86cf768f73"}, | ||||||
|     {file = "asyncpg-0.20.1-cp35-cp35m-win_amd64.whl", hash = "sha256:25edb0b947eb632b6b53e5a4b36cba5677297bb34cbaba270019714d0a5fed76"}, |     {file = "asyncpg-0.21.0-cp35-cp35m-win32.whl", hash = "sha256:b1b10916c006e5c2c0dcd5dadeb38cbf61ecd20d66c50164e82f31c22c7e329d"}, | ||||||
|     {file = "asyncpg-0.20.1-cp36-cp36m-macosx_10_13_x86_64.whl", hash = "sha256:95cd2df61ee00b789bdcd04a080e6d9188693b841db2bf9a87ebaed9e53147e0"}, |     {file = "asyncpg-0.21.0-cp35-cp35m-win_amd64.whl", hash = "sha256:22d161618b59e4b56fb2a5cc956aa9eeb336d07cae924a5b90c9aa1c2d137f15"}, | ||||||
|     {file = "asyncpg-0.20.1-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:058baec9d6b75612412baa872a1aa47317d0ff88c318a49f9c4a2389043d5a8d"}, |     {file = "asyncpg-0.21.0-cp36-cp36m-macosx_10_13_x86_64.whl", hash = "sha256:f2d1aa890ffd1ad062a38b7ff7488764b3da4b0a24e0c83d7bbb1d1a6609df15"}, | ||||||
|     {file = "asyncpg-0.20.1-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:c773c7dbe2f4d3ebc9e3030e94303e45d6742e6c2fc25da0c46a56ea3d83caeb"}, |     {file = "asyncpg-0.21.0-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:e7bfb9269aeb11d78d50accf1be46823683ced99209b7199e307cdf7da849522"}, | ||||||
|     {file = "asyncpg-0.20.1-cp36-cp36m-win32.whl", hash = "sha256:5664d1bd8abe64fc60a0e701eb85fa1d8c9a4a8018a5a59164d27238f2caf395"}, |     {file = "asyncpg-0.21.0-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:68f7981f65317a5d5f497ec76919b488dbe0e838f8b924e7517a680bdca0f308"}, | ||||||
|     {file = "asyncpg-0.20.1-cp36-cp36m-win_amd64.whl", hash = "sha256:57666dfae38f4dbf84ffbf0c5c0f78733fef0e8e083230275dcb9ccad1d5ee09"}, |     {file = "asyncpg-0.21.0-cp36-cp36m-manylinux2014_aarch64.whl", hash = "sha256:a4c1feb285ec3807ecd5b54ab718a3d065bb55c93ebaf800670eadde31484be8"}, | ||||||
|     {file = "asyncpg-0.20.1-cp37-cp37m-macosx_10_13_x86_64.whl", hash = "sha256:0c336903c3b08e970f8af2f606332f1738dba156bca83ed0467dc2f5c70da796"}, |     {file = "asyncpg-0.21.0-cp36-cp36m-win32.whl", hash = "sha256:dddf4d4c5e781310a36529c3c87c1746837c2d2c7ec0f2ec4e4f06450d83c50a"}, | ||||||
|     {file = "asyncpg-0.20.1-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:ad5ba062e09673b1a4b8d0facaf5a6d9719bf7b337440d10b07fe994d90a9552"}, |     {file = "asyncpg-0.21.0-cp36-cp36m-win_amd64.whl", hash = "sha256:7ee29c4707eb8fb3d3a0348ac4495e06f4afaca3ee38c3bebedc9c8b239125ff"}, | ||||||
|     {file = "asyncpg-0.20.1-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:ba90d3578bc6dddcbce461875672fd9bdb34f0b8215b68612dd3b65a956ff51c"}, |     {file = "asyncpg-0.21.0-cp37-cp37m-macosx_10_13_x86_64.whl", hash = "sha256:4421407b07b4e22291a226d9de0bf6f3ea8158aa1c12d83bfedbf5c22e13cd55"}, | ||||||
|     {file = "asyncpg-0.20.1-cp37-cp37m-win32.whl", hash = "sha256:da238592235717419a6a7b5edc8564da410ebfd056ca4ecc41e70b1b5df86fba"}, |     {file = "asyncpg-0.21.0-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:aa2e0cb14c01a2f58caeeca7196681b30aa22dd22c82845560b401df5e98e171"}, | ||||||
|     {file = "asyncpg-0.20.1-cp37-cp37m-win_amd64.whl", hash = "sha256:74510234c294c6a6767089ba9c938f09a491426c24405634eb357bd91dffd734"}, |     {file = "asyncpg-0.21.0-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:28584783dd0d21b2a0db3bfe54fb12f21425a4cc015e4419083ea99e6de0de9b"}, | ||||||
|     {file = "asyncpg-0.20.1-cp38-cp38-macosx_10_13_x86_64.whl", hash = "sha256:391aea89871df8c1560750af6c7170f2772c2d133b34772acf3637e3cf4db93e"}, |     {file = "asyncpg-0.21.0-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:915cebc8a7693c8a5e89804fa106678dbedcc50d0270ebab0b75f16e668bd59b"}, | ||||||
|     {file = "asyncpg-0.20.1-cp38-cp38-manylinux1_i686.whl", hash = "sha256:a981500bf6947926e53c48f4d60ae080af1b4ad7fa78e363465a5b5ad4f2b65e"}, |     {file = "asyncpg-0.21.0-cp37-cp37m-win32.whl", hash = "sha256:308b8ba32c42ea1ed84c034320678ec307296bb4faf3fbbeb9f9e20b46db99a5"}, | ||||||
|     {file = "asyncpg-0.20.1-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:a9e6fd6f0f9e8bd77e9a4e1ef9a4f83a80674d9136a754ae3603e915da96b627"}, |     {file = "asyncpg-0.21.0-cp37-cp37m-win_amd64.whl", hash = "sha256:888593b6688faa7ec1c97ff7f2ca3b5a5b8abb15478fe2a13c5012b607a28737"}, | ||||||
|     {file = "asyncpg-0.20.1-cp38-cp38-win32.whl", hash = "sha256:e39aac2b3a2f839ce65aa255ce416de899c58b7d38d601d24ca35558e13b48e3"}, |     {file = "asyncpg-0.21.0-cp38-cp38-macosx_10_13_x86_64.whl", hash = "sha256:ecd5232cf64f58caac3b85103f1223fdf20e9eb43bfa053c56ef9e5dd76ab099"}, | ||||||
|     {file = "asyncpg-0.20.1-cp38-cp38-win_amd64.whl", hash = "sha256:2af6a5a705accd36e13292ea43d08c20b15e52d684beb522cb3a7d3c9c8f3f48"}, |     {file = "asyncpg-0.21.0-cp38-cp38-manylinux1_i686.whl", hash = "sha256:3ade59cef35bffae6dbc6f5f3ef56e1d53c67f0a7adc3cc4c714f07568d2d717"}, | ||||||
|     {file = "asyncpg-0.20.1.tar.gz", hash = "sha256:394bf19bdddbba07a38cd6fb526ebf66e120444d6b3097332b78efd5b26495b0"}, |     {file = "asyncpg-0.21.0-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:ea26604932719b3612541e606508d9d604211f56a65806ccf8c92c64104f4f8a"}, | ||||||
|  |     {file = "asyncpg-0.21.0-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:7e51d1a012b779e0ebf0195f80d004f65d3c60cc06f0fa1cef9d3e536262abbd"}, | ||||||
|  |     {file = "asyncpg-0.21.0-cp38-cp38-win32.whl", hash = "sha256:615c7e3adb46e1f2e3aff45e4ee9401b4f24f9f7153e5530a0753369be72a5c6"}, | ||||||
|  |     {file = "asyncpg-0.21.0-cp38-cp38-win_amd64.whl", hash = "sha256:823eca36108bd64a8600efe7bbf1230aa00f2defa3be42852f3b61ab40cf1226"}, | ||||||
|  |     {file = "asyncpg-0.21.0.tar.gz", hash = "sha256:53cb2a0eb326f61e34ef4da2db01d87ce9c0ebe396f65a295829df334e31863f"}, | ||||||
| ] | ] | ||||||
| atomicwrites = [ | atomicwrites = [ | ||||||
|     {file = "atomicwrites-1.4.0-py2.py3-none-any.whl", hash = "sha256:6d1784dea7c0c8d4a5172b6c620f40b6e4cbfdf96d783691f2e1302a7b88e197"}, |     {file = "atomicwrites-1.4.0-py2.py3-none-any.whl", hash = "sha256:6d1784dea7c0c8d4a5172b6c620f40b6e4cbfdf96d783691f2e1302a7b88e197"}, | ||||||
|     {file = "atomicwrites-1.4.0.tar.gz", hash = "sha256:ae70396ad1a434f9c7046fd2dd196fc04b12f9e91ffb859164193be8b6168a7a"}, |     {file = "atomicwrites-1.4.0.tar.gz", hash = "sha256:ae70396ad1a434f9c7046fd2dd196fc04b12f9e91ffb859164193be8b6168a7a"}, | ||||||
| ] | ] | ||||||
| attrs = [ | attrs = [ | ||||||
|     {file = "attrs-19.3.0-py2.py3-none-any.whl", hash = "sha256:08a96c641c3a74e44eb59afb61a24f2cb9f4d7188748e76ba4bb5edfa3cb7d1c"}, |     {file = "attrs-20.2.0-py2.py3-none-any.whl", hash = "sha256:fce7fc47dfc976152e82d53ff92fa0407700c21acd20886a13777a0d20e655dc"}, | ||||||
|     {file = "attrs-19.3.0.tar.gz", hash = "sha256:f7b7ce16570fe9965acd6d30101a28f62fb4a7f9e926b3bbc9b61f8b04247e72"}, |     {file = "attrs-20.2.0.tar.gz", hash = "sha256:26b54ddbbb9ee1d34d5d3668dd37d6cf74990ab23c828c2888dccdceee395594"}, | ||||||
| ] | ] | ||||||
| bandit = [ | bandit = [ | ||||||
|     {file = "bandit-1.6.2-py2.py3-none-any.whl", hash = "sha256:336620e220cf2d3115877685e264477ff9d9abaeb0afe3dc7264f55fa17a3952"}, |     {file = "bandit-1.6.2-py2.py3-none-any.whl", hash = "sha256:336620e220cf2d3115877685e264477ff9d9abaeb0afe3dc7264f55fa17a3952"}, | ||||||
| @@ -707,34 +641,42 @@ black = [ | |||||||
|     {file = "black-19.10b0.tar.gz", hash = "sha256:c2edb73a08e9e0e6f65a0e6af18b059b8b1cdd5bef997d7a0b181df93dc81539"}, |     {file = "black-19.10b0.tar.gz", hash = "sha256:c2edb73a08e9e0e6f65a0e6af18b059b8b1cdd5bef997d7a0b181df93dc81539"}, | ||||||
| ] | ] | ||||||
| cffi = [ | cffi = [ | ||||||
|     {file = "cffi-1.14.1-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:66dd45eb9530e3dde8f7c009f84568bc7cac489b93d04ac86e3111fb46e470c2"}, |     {file = "cffi-1.14.3-2-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:3eeeb0405fd145e714f7633a5173318bd88d8bbfc3dd0a5751f8c4f70ae629bc"}, | ||||||
|     {file = "cffi-1.14.1-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:4f53e4128c81ca3212ff4cf097c797ab44646a40b42ec02a891155cd7a2ba4d8"}, |     {file = "cffi-1.14.3-2-cp35-cp35m-macosx_10_9_x86_64.whl", hash = "sha256:cb763ceceae04803adcc4e2d80d611ef201c73da32d8f2722e9d0ab0c7f10768"}, | ||||||
|     {file = "cffi-1.14.1-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:833401b15de1bb92791d7b6fb353d4af60dc688eaa521bd97203dcd2d124a7c1"}, |     {file = "cffi-1.14.3-2-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:44f60519595eaca110f248e5017363d751b12782a6f2bd6a7041cba275215f5d"}, | ||||||
|     {file = "cffi-1.14.1-cp27-cp27m-win32.whl", hash = "sha256:26f33e8f6a70c255767e3c3f957ccafc7f1f706b966e110b855bfe944511f1f9"}, |     {file = "cffi-1.14.3-2-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:c53af463f4a40de78c58b8b2710ade243c81cbca641e34debf3396a9640d6ec1"}, | ||||||
|     {file = "cffi-1.14.1-cp27-cp27m-win_amd64.whl", hash = "sha256:b87dfa9f10a470eee7f24234a37d1d5f51e5f5fa9eeffda7c282e2b8f5162eb1"}, |     {file = "cffi-1.14.3-2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:33c6cdc071ba5cd6d96769c8969a0531be2d08c2628a0143a10a7dcffa9719ca"}, | ||||||
|     {file = "cffi-1.14.1-cp27-cp27mu-manylinux1_i686.whl", hash = "sha256:effd2ba52cee4ceff1a77f20d2a9f9bf8d50353c854a282b8760ac15b9833168"}, |     {file = "cffi-1.14.3-2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:c11579638288e53fc94ad60022ff1b67865363e730ee41ad5e6f0a17188b327a"}, | ||||||
|     {file = "cffi-1.14.1-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:bac0d6f7728a9cc3c1e06d4fcbac12aaa70e9379b3025b27ec1226f0e2d404cf"}, |     {file = "cffi-1.14.3-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:3cb3e1b9ec43256c4e0f8d2837267a70b0e1ca8c4f456685508ae6106b1f504c"}, | ||||||
|     {file = "cffi-1.14.1-cp35-cp35m-macosx_10_9_x86_64.whl", hash = "sha256:d6033b4ffa34ef70f0b8086fd4c3df4bf801fee485a8a7d4519399818351aa8e"}, |     {file = "cffi-1.14.3-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:f0620511387790860b249b9241c2f13c3a80e21a73e0b861a2df24e9d6f56730"}, | ||||||
|     {file = "cffi-1.14.1-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:8416ed88ddc057bab0526d4e4e9f3660f614ac2394b5e019a628cdfff3733849"}, |     {file = "cffi-1.14.3-cp27-cp27m-win32.whl", hash = "sha256:005f2bfe11b6745d726dbb07ace4d53f057de66e336ff92d61b8c7e9c8f4777d"}, | ||||||
|     {file = "cffi-1.14.1-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:892daa86384994fdf4856cb43c93f40cbe80f7f95bb5da94971b39c7f54b3a9c"}, |     {file = "cffi-1.14.3-cp27-cp27m-win_amd64.whl", hash = "sha256:2f9674623ca39c9ebe38afa3da402e9326c245f0f5ceff0623dccdac15023e05"}, | ||||||
|     {file = "cffi-1.14.1-cp35-cp35m-win32.whl", hash = "sha256:c991112622baee0ae4d55c008380c32ecfd0ad417bcd0417ba432e6ba7328caa"}, |     {file = "cffi-1.14.3-cp27-cp27mu-manylinux1_i686.whl", hash = "sha256:09e96138280241bd355cd585148dec04dbbedb4f46128f340d696eaafc82dd7b"}, | ||||||
|     {file = "cffi-1.14.1-cp35-cp35m-win_amd64.whl", hash = "sha256:fcf32bf76dc25e30ed793145a57426064520890d7c02866eb93d3e4abe516948"}, |     {file = "cffi-1.14.3-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:3363e77a6176afb8823b6e06db78c46dbc4c7813b00a41300a4873b6ba63b171"}, | ||||||
|     {file = "cffi-1.14.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:f960375e9823ae6a07072ff7f8a85954e5a6434f97869f50d0e41649a1c8144f"}, |     {file = "cffi-1.14.3-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:0ef488305fdce2580c8b2708f22d7785ae222d9825d3094ab073e22e93dfe51f"}, | ||||||
|     {file = "cffi-1.14.1-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:a6d28e7f14ecf3b2ad67c4f106841218c8ab12a0683b1528534a6c87d2307af3"}, |     {file = "cffi-1.14.3-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:0b1ad452cc824665ddc682400b62c9e4f5b64736a2ba99110712fdee5f2505c4"}, | ||||||
|     {file = "cffi-1.14.1-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:cda422d54ee7905bfc53ee6915ab68fe7b230cacf581110df4272ee10462aadc"}, |     {file = "cffi-1.14.3-cp35-cp35m-win32.whl", hash = "sha256:85ba797e1de5b48aa5a8427b6ba62cf69607c18c5d4eb747604b7302f1ec382d"}, | ||||||
|     {file = "cffi-1.14.1-cp36-cp36m-win32.whl", hash = "sha256:4a03416915b82b81af5502459a8a9dd62a3c299b295dcdf470877cb948d655f2"}, |     {file = "cffi-1.14.3-cp35-cp35m-win_amd64.whl", hash = "sha256:e66399cf0fc07de4dce4f588fc25bfe84a6d1285cc544e67987d22663393926d"}, | ||||||
|     {file = "cffi-1.14.1-cp36-cp36m-win_amd64.whl", hash = "sha256:4ce1e995aeecf7cc32380bc11598bfdfa017d592259d5da00fc7ded11e61d022"}, |     {file = "cffi-1.14.3-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:15f351bed09897fbda218e4db5a3d5c06328862f6198d4fb385f3e14e19decb3"}, | ||||||
|     {file = "cffi-1.14.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:e23cb7f1d8e0f93addf0cae3c5b6f00324cccb4a7949ee558d7b6ca973ab8ae9"}, |     {file = "cffi-1.14.3-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:4d7c26bfc1ea9f92084a1d75e11999e97b62d63128bcc90c3624d07813c52808"}, | ||||||
|     {file = "cffi-1.14.1-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:ddff0b2bd7edcc8c82d1adde6dbbf5e60d57ce985402541cd2985c27f7bec2a0"}, |     {file = "cffi-1.14.3-cp36-cp36m-manylinux2014_aarch64.whl", hash = "sha256:23e5d2040367322824605bc29ae8ee9175200b92cb5483ac7d466927a9b3d537"}, | ||||||
|     {file = "cffi-1.14.1-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:f90c2267101010de42f7273c94a1f026e56cbc043f9330acd8a80e64300aba33"}, |     {file = "cffi-1.14.3-cp36-cp36m-win32.whl", hash = "sha256:a624fae282e81ad2e4871bdb767e2c914d0539708c0f078b5b355258293c98b0"}, | ||||||
|     {file = "cffi-1.14.1-cp37-cp37m-win32.whl", hash = "sha256:3cd2c044517f38d1b577f05927fb9729d3396f1d44d0c659a445599e79519792"}, |     {file = "cffi-1.14.3-cp36-cp36m-win_amd64.whl", hash = "sha256:de31b5164d44ef4943db155b3e8e17929707cac1e5bd2f363e67a56e3af4af6e"}, | ||||||
|     {file = "cffi-1.14.1-cp37-cp37m-win_amd64.whl", hash = "sha256:4fa72a52a906425416f41738728268072d5acfd48cbe7796af07a923236bcf96"}, |     {file = "cffi-1.14.3-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:f92cdecb618e5fa4658aeb97d5eb3d2f47aa94ac6477c6daf0f306c5a3b9e6b1"}, | ||||||
|     {file = "cffi-1.14.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:267adcf6e68d77ba154334a3e4fc921b8e63cbb38ca00d33d40655d4228502bc"}, |     {file = "cffi-1.14.3-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:22399ff4870fb4c7ef19fff6eeb20a8bbf15571913c181c78cb361024d574579"}, | ||||||
|     {file = "cffi-1.14.1-cp38-cp38-manylinux1_i686.whl", hash = "sha256:d3148b6ba3923c5850ea197a91a42683f946dba7e8eb82dfa211ab7e708de939"}, |     {file = "cffi-1.14.3-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:f4eae045e6ab2bb54ca279733fe4eb85f1effda392666308250714e01907f394"}, | ||||||
|     {file = "cffi-1.14.1-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:98be759efdb5e5fa161e46d404f4e0ce388e72fbf7d9baf010aff16689e22abe"}, |     {file = "cffi-1.14.3-cp37-cp37m-win32.whl", hash = "sha256:b0358e6fefc74a16f745afa366acc89f979040e0cbc4eec55ab26ad1f6a9bfbc"}, | ||||||
|     {file = "cffi-1.14.1-cp38-cp38-win32.whl", hash = "sha256:6923d077d9ae9e8bacbdb1c07ae78405a9306c8fd1af13bfa06ca891095eb995"}, |     {file = "cffi-1.14.3-cp37-cp37m-win_amd64.whl", hash = "sha256:6642f15ad963b5092d65aed022d033c77763515fdc07095208f15d3563003869"}, | ||||||
|     {file = "cffi-1.14.1-cp38-cp38-win_amd64.whl", hash = "sha256:b1d6ebc891607e71fd9da71688fcf332a6630b7f5b7f5549e6e631821c0e5d90"}, |     {file = "cffi-1.14.3-cp38-cp38-manylinux1_i686.whl", hash = "sha256:2791f68edc5749024b4722500e86303a10d342527e1e3bcac47f35fbd25b764e"}, | ||||||
|     {file = "cffi-1.14.1.tar.gz", hash = "sha256:b2a2b0d276a136146e012154baefaea2758ef1f56ae9f4e01c612b0831e0bd2f"}, |     {file = "cffi-1.14.3-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:529c4ed2e10437c205f38f3691a68be66c39197d01062618c55f74294a4a4828"}, | ||||||
|  |     {file = "cffi-1.14.3-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:8f0f1e499e4000c4c347a124fa6a27d37608ced4fe9f7d45070563b7c4c370c9"}, | ||||||
|  |     {file = "cffi-1.14.3-cp38-cp38-win32.whl", hash = "sha256:3b8eaf915ddc0709779889c472e553f0d3e8b7bdf62dab764c8921b09bf94522"}, | ||||||
|  |     {file = "cffi-1.14.3-cp38-cp38-win_amd64.whl", hash = "sha256:bbd2f4dfee1079f76943767fce837ade3087b578aeb9f69aec7857d5bf25db15"}, | ||||||
|  |     {file = "cffi-1.14.3-cp39-cp39-manylinux1_i686.whl", hash = "sha256:cc75f58cdaf043fe6a7a6c04b3b5a0e694c6a9e24050967747251fb80d7bce0d"}, | ||||||
|  |     {file = "cffi-1.14.3-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:bf39a9e19ce7298f1bd6a9758fa99707e9e5b1ebe5e90f2c3913a47bc548747c"}, | ||||||
|  |     {file = "cffi-1.14.3-cp39-cp39-win32.whl", hash = "sha256:d80998ed59176e8cba74028762fbd9b9153b9afc71ea118e63bbf5d4d0f9552b"}, | ||||||
|  |     {file = "cffi-1.14.3-cp39-cp39-win_amd64.whl", hash = "sha256:c150eaa3dadbb2b5339675b88d4573c1be3cb6f2c33a6c83387e10cc0bf05bd3"}, | ||||||
|  |     {file = "cffi-1.14.3.tar.gz", hash = "sha256:f92f789e4f9241cd262ad7a555ca2c648a98178a953af117ef7fad46aa1d5591"}, | ||||||
| ] | ] | ||||||
| click = [ | click = [ | ||||||
|     {file = "click-7.1.2-py2.py3-none-any.whl", hash = "sha256:dacca89f4bfadd5de3d7489b7c8a566eee0d3676333fbb50030263894c38c0dc"}, |     {file = "click-7.1.2-py2.py3-none-any.whl", hash = "sha256:dacca89f4bfadd5de3d7489b7c8a566eee0d3676333fbb50030263894c38c0dc"}, | ||||||
| @@ -745,25 +687,28 @@ colorama = [ | |||||||
|     {file = "colorama-0.4.3.tar.gz", hash = "sha256:e96da0d330793e2cb9485e9ddfd918d456036c7149416295932478192f4436a1"}, |     {file = "colorama-0.4.3.tar.gz", hash = "sha256:e96da0d330793e2cb9485e9ddfd918d456036c7149416295932478192f4436a1"}, | ||||||
| ] | ] | ||||||
| cryptography = [ | cryptography = [ | ||||||
|     {file = "cryptography-3.0-cp27-cp27m-macosx_10_10_x86_64.whl", hash = "sha256:ab49edd5bea8d8b39a44b3db618e4783ef84c19c8b47286bf05dfdb3efb01c83"}, |     {file = "cryptography-3.1.1-cp27-cp27m-macosx_10_10_x86_64.whl", hash = "sha256:65beb15e7f9c16e15934569d29fb4def74ea1469d8781f6b3507ab896d6d8719"}, | ||||||
|     {file = "cryptography-3.0-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:124af7255ffc8e964d9ff26971b3a6153e1a8a220b9a685dc407976ecb27a06a"}, |     {file = "cryptography-3.1.1-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:983c0c3de4cb9fcba68fd3f45ed846eb86a2a8b8d8bc5bb18364c4d00b3c61fe"}, | ||||||
|     {file = "cryptography-3.0-cp27-cp27m-manylinux2010_x86_64.whl", hash = "sha256:51e40123083d2f946794f9fe4adeeee2922b581fa3602128ce85ff813d85b81f"}, |     {file = "cryptography-3.1.1-cp27-cp27m-manylinux2010_x86_64.whl", hash = "sha256:e97a3b627e3cb63c415a16245d6cef2139cca18bb1183d1b9375a1c14e83f3b3"}, | ||||||
|     {file = "cryptography-3.0-cp27-cp27m-win32.whl", hash = "sha256:dea0ba7fe6f9461d244679efa968d215ea1f989b9c1957d7f10c21e5c7c09ad6"}, |     {file = "cryptography-3.1.1-cp27-cp27m-win32.whl", hash = "sha256:cb179acdd4ae1e4a5a160d80b87841b3d0e0be84af46c7bb2cd7ece57a39c4ba"}, | ||||||
|     {file = "cryptography-3.0-cp27-cp27m-win_amd64.whl", hash = "sha256:8ecf9400d0893836ff41b6f977a33972145a855b6efeb605b49ee273c5e6469f"}, |     {file = "cryptography-3.1.1-cp27-cp27m-win_amd64.whl", hash = "sha256:b372026ebf32fe2523159f27d9f0e9f485092e43b00a5adacf732192a70ba118"}, | ||||||
|     {file = "cryptography-3.0-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:0c608ff4d4adad9e39b5057de43657515c7da1ccb1807c3a27d4cf31fc923b4b"}, |     {file = "cryptography-3.1.1-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:680da076cad81cdf5ffcac50c477b6790be81768d30f9da9e01960c4b18a66db"}, | ||||||
|     {file = "cryptography-3.0-cp27-cp27mu-manylinux2010_x86_64.whl", hash = "sha256:bec7568c6970b865f2bcebbe84d547c52bb2abadf74cefce396ba07571109c67"}, |     {file = "cryptography-3.1.1-cp27-cp27mu-manylinux2010_x86_64.whl", hash = "sha256:5d52c72449bb02dd45a773a203196e6d4fae34e158769c896012401f33064396"}, | ||||||
|     {file = "cryptography-3.0-cp35-abi3-macosx_10_10_x86_64.whl", hash = "sha256:0cbfed8ea74631fe4de00630f4bb592dad564d57f73150d6f6796a24e76c76cd"}, |     {file = "cryptography-3.1.1-cp35-abi3-macosx_10_10_x86_64.whl", hash = "sha256:f0e099fc4cc697450c3dd4031791559692dd941a95254cb9aeded66a7aa8b9bc"}, | ||||||
|     {file = "cryptography-3.0-cp35-abi3-manylinux1_x86_64.whl", hash = "sha256:a09fd9c1cca9a46b6ad4bea0a1f86ab1de3c0c932364dbcf9a6c2a5eeb44fa77"}, |     {file = "cryptography-3.1.1-cp35-abi3-manylinux1_x86_64.whl", hash = "sha256:a7597ffc67987b37b12e09c029bd1dc43965f75d328076ae85721b84046e9ca7"}, | ||||||
|     {file = "cryptography-3.0-cp35-abi3-manylinux2010_x86_64.whl", hash = "sha256:ce82cc06588e5cbc2a7df3c8a9c778f2cb722f56835a23a68b5a7264726bb00c"}, |     {file = "cryptography-3.1.1-cp35-abi3-manylinux2010_x86_64.whl", hash = "sha256:4549b137d8cbe3c2eadfa56c0c858b78acbeff956bd461e40000b2164d9167c6"}, | ||||||
|     {file = "cryptography-3.0-cp35-cp35m-win32.whl", hash = "sha256:9367d00e14dee8d02134c6c9524bb4bd39d4c162456343d07191e2a0b5ec8b3b"}, |     {file = "cryptography-3.1.1-cp35-abi3-manylinux2014_aarch64.whl", hash = "sha256:89aceb31cd5f9fc2449fe8cf3810797ca52b65f1489002d58fe190bfb265c536"}, | ||||||
|     {file = "cryptography-3.0-cp35-cp35m-win_amd64.whl", hash = "sha256:384d7c681b1ab904fff3400a6909261cae1d0939cc483a68bdedab282fb89a07"}, |     {file = "cryptography-3.1.1-cp35-cp35m-win32.whl", hash = "sha256:559d622aef2a2dff98a892eef321433ba5bc55b2485220a8ca289c1ecc2bd54f"}, | ||||||
|     {file = "cryptography-3.0-cp36-cp36m-win32.whl", hash = "sha256:4d355f2aee4a29063c10164b032d9fa8a82e2c30768737a2fd56d256146ad559"}, |     {file = "cryptography-3.1.1-cp35-cp35m-win_amd64.whl", hash = "sha256:451cdf60be4dafb6a3b78802006a020e6cd709c22d240f94f7a0696240a17154"}, | ||||||
|     {file = "cryptography-3.0-cp36-cp36m-win_amd64.whl", hash = "sha256:45741f5499150593178fc98d2c1a9c6722df88b99c821ad6ae298eff0ba1ae71"}, |     {file = "cryptography-3.1.1-cp36-abi3-win32.whl", hash = "sha256:762bc5a0df03c51ee3f09c621e1cee64e3a079a2b5020de82f1613873d79ee70"}, | ||||||
|     {file = "cryptography-3.0-cp37-cp37m-win32.whl", hash = "sha256:8ecef21ac982aa78309bb6f092d1677812927e8b5ef204a10c326fc29f1367e2"}, |     {file = "cryptography-3.1.1-cp36-abi3-win_amd64.whl", hash = "sha256:b12e715c10a13ca1bd27fbceed9adc8c5ff640f8e1f7ea76416352de703523c8"}, | ||||||
|     {file = "cryptography-3.0-cp37-cp37m-win_amd64.whl", hash = "sha256:4b9303507254ccb1181d1803a2080a798910ba89b1a3c9f53639885c90f7a756"}, |     {file = "cryptography-3.1.1-cp36-cp36m-win32.whl", hash = "sha256:21b47c59fcb1c36f1113f3709d37935368e34815ea1d7073862e92f810dc7499"}, | ||||||
|     {file = "cryptography-3.0-cp38-cp38-win32.whl", hash = "sha256:8713ddb888119b0d2a1462357d5946b8911be01ddbf31451e1d07eaa5077a261"}, |     {file = "cryptography-3.1.1-cp36-cp36m-win_amd64.whl", hash = "sha256:48ee615a779ffa749d7d50c291761dc921d93d7cf203dca2db663b4f193f0e49"}, | ||||||
|     {file = "cryptography-3.0-cp38-cp38-win_amd64.whl", hash = "sha256:bea0b0468f89cdea625bb3f692cd7a4222d80a6bdafd6fb923963f2b9da0e15f"}, |     {file = "cryptography-3.1.1-cp37-cp37m-win32.whl", hash = "sha256:b2bded09c578d19e08bd2c5bb8fed7f103e089752c9cf7ca7ca7de522326e921"}, | ||||||
|     {file = "cryptography-3.0.tar.gz", hash = "sha256:8e924dbc025206e97756e8903039662aa58aa9ba357d8e1d8fc29e3092322053"}, |     {file = "cryptography-3.1.1-cp37-cp37m-win_amd64.whl", hash = "sha256:f99317a0fa2e49917689b8cf977510addcfaaab769b3f899b9c481bbd76730c2"}, | ||||||
|  |     {file = "cryptography-3.1.1-cp38-cp38-win32.whl", hash = "sha256:ab010e461bb6b444eaf7f8c813bb716be2d78ab786103f9608ffd37a4bd7d490"}, | ||||||
|  |     {file = "cryptography-3.1.1-cp38-cp38-win_amd64.whl", hash = "sha256:99d4984aabd4c7182050bca76176ce2dbc9fa9748afe583a7865c12954d714ba"}, | ||||||
|  |     {file = "cryptography-3.1.1.tar.gz", hash = "sha256:9d9fc6a16357965d282dd4ab6531013935425d0dc4950df2e0cf2a1b1ac1017d"}, | ||||||
| ] | ] | ||||||
| execnet = [ | execnet = [ | ||||||
|     {file = "execnet-1.7.1-py2.py3-none-any.whl", hash = "sha256:d4efd397930c46415f62f8a31388d6be4f27a91d7550eb79bc64a756e0056547"}, |     {file = "execnet-1.7.1-py2.py3-none-any.whl", hash = "sha256:d4efd397930c46415f62f8a31388d6be4f27a91d7550eb79bc64a756e0056547"}, | ||||||
| @@ -778,37 +723,33 @@ gitdb = [ | |||||||
|     {file = "gitdb-4.0.5.tar.gz", hash = "sha256:c9e1f2d0db7ddb9a704c2a0217be31214e91a4fe1dea1efad19ae42ba0c285c9"}, |     {file = "gitdb-4.0.5.tar.gz", hash = "sha256:c9e1f2d0db7ddb9a704c2a0217be31214e91a4fe1dea1efad19ae42ba0c285c9"}, | ||||||
| ] | ] | ||||||
| gitpython = [ | gitpython = [ | ||||||
|     {file = "GitPython-3.1.7-py3-none-any.whl", hash = "sha256:fa3b92da728a457dd75d62bb5f3eb2816d99a7fe6c67398e260637a40e3fafb5"}, |     {file = "GitPython-3.1.8-py3-none-any.whl", hash = "sha256:1858f4fd089abe92ae465f01d5aaaf55e937eca565fb2c1fce35a51b5f85c910"}, | ||||||
|     {file = "GitPython-3.1.7.tar.gz", hash = "sha256:2db287d71a284e22e5c2846042d0602465c7434d910406990d5b74df4afb0858"}, |     {file = "GitPython-3.1.8.tar.gz", hash = "sha256:080bf8e2cf1a2b907634761c2eaefbe83b69930c94c66ad11b65a8252959f912"}, | ||||||
| ] |  | ||||||
| idna = [ |  | ||||||
|     {file = "idna-2.10-py2.py3-none-any.whl", hash = "sha256:b97d804b1e9b523befed77c48dacec60e6dcb0b5391d57af6a65a312a90648c0"}, |  | ||||||
|     {file = "idna-2.10.tar.gz", hash = "sha256:b307872f855b18632ce0c21c5e45be78c0ea7ae4c15c828c20788b26921eb3f6"}, |  | ||||||
| ] | ] | ||||||
| importlib-metadata = [ | importlib-metadata = [ | ||||||
|     {file = "importlib_metadata-1.7.0-py2.py3-none-any.whl", hash = "sha256:dc15b2969b4ce36305c51eebe62d418ac7791e9a157911d58bfb1f9ccd8e2070"}, |     {file = "importlib_metadata-2.0.0-py2.py3-none-any.whl", hash = "sha256:cefa1a2f919b866c5beb7c9f7b0ebb4061f30a8a9bf16d609b000e2dfaceb9c3"}, | ||||||
|     {file = "importlib_metadata-1.7.0.tar.gz", hash = "sha256:90bb658cdbbf6d1735b6341ce708fc7024a3e14e99ffdc5783edea9f9b077f83"}, |     {file = "importlib_metadata-2.0.0.tar.gz", hash = "sha256:77a540690e24b0305878c37ffd421785a6f7e53c8b5720d211b211de8d0e95da"}, | ||||||
| ] | ] | ||||||
| iniconfig = [ | iniconfig = [ | ||||||
|     {file = "iniconfig-1.0.1-py3-none-any.whl", hash = "sha256:80cf40c597eb564e86346103f609d74efce0f6b4d4f30ec8ce9e2c26411ba437"}, |     {file = "iniconfig-1.0.1-py3-none-any.whl", hash = "sha256:80cf40c597eb564e86346103f609d74efce0f6b4d4f30ec8ce9e2c26411ba437"}, | ||||||
|     {file = "iniconfig-1.0.1.tar.gz", hash = "sha256:e5f92f89355a67de0595932a6c6c02ab4afddc6fcdc0bfc5becd0d60884d3f69"}, |     {file = "iniconfig-1.0.1.tar.gz", hash = "sha256:e5f92f89355a67de0595932a6c6c02ab4afddc6fcdc0bfc5becd0d60884d3f69"}, | ||||||
| ] | ] | ||||||
| iso8601 = [ | iso8601 = [ | ||||||
|     {file = "iso8601-0.1.12-py2.py3-none-any.whl", hash = "sha256:210e0134677cc0d02f6028087fee1df1e1d76d372ee1db0bf30bf66c5c1c89a3"}, |     {file = "iso8601-0.1.13-py2.py3-none-any.whl", hash = "sha256:694be0743e9f1535ea873bfc7bd6fb62380c62b75822761859428073a17fd39c"}, | ||||||
|     {file = "iso8601-0.1.12-py3-none-any.whl", hash = "sha256:bbbae5fb4a7abfe71d4688fd64bff70b91bbd74ef6a99d964bab18f7fdf286dd"}, |     {file = "iso8601-0.1.13-py3-none-any.whl", hash = "sha256:6f02f01dd13320a7f280e58516dc8d1950dfaf77527cc365a398cd9de4d3c692"}, | ||||||
|     {file = "iso8601-0.1.12.tar.gz", hash = "sha256:49c4b20e1f38aa5cf109ddcd39647ac419f928512c869dc01d5c7098eddede82"}, |     {file = "iso8601-0.1.13.tar.gz", hash = "sha256:f7dec22af52025d4526be94cc1303c7d8f5379b746a3f54a8c8446384392eeb1"}, | ||||||
| ] | ] | ||||||
| isort = [ | isort = [ | ||||||
|     {file = "isort-5.3.2-py3-none-any.whl", hash = "sha256:5196bd2f5b23dc91215734b1c96c6d28390061d69860a948094c12635d6d64e6"}, |     {file = "isort-5.5.3-py3-none-any.whl", hash = "sha256:c16eaa7432a1c004c585d79b12ad080c6c421dd18fe27982ca11f95e6898e432"}, | ||||||
|     {file = "isort-5.3.2.tar.gz", hash = "sha256:ba83762132a8661d3525f87a86549712fb7d8da79eeb452e01f327ada9e87920"}, |     {file = "isort-5.5.3.tar.gz", hash = "sha256:6187a9f1ce8784cbc6d1b88790a43e6083a6302f03e9ae482acc0f232a98c843"}, | ||||||
| ] | ] | ||||||
| mccabe = [ | mccabe = [ | ||||||
|     {file = "mccabe-0.6.1-py2.py3-none-any.whl", hash = "sha256:ab8a6258860da4b6677da4bd2fe5dc2c659cff31b3ee4f7f5d64e79735b80d42"}, |     {file = "mccabe-0.6.1-py2.py3-none-any.whl", hash = "sha256:ab8a6258860da4b6677da4bd2fe5dc2c659cff31b3ee4f7f5d64e79735b80d42"}, | ||||||
|     {file = "mccabe-0.6.1.tar.gz", hash = "sha256:dd8d182285a0fe56bace7f45b5e7d1a6ebcbf524e8f3bd87eb0f125271b8831f"}, |     {file = "mccabe-0.6.1.tar.gz", hash = "sha256:dd8d182285a0fe56bace7f45b5e7d1a6ebcbf524e8f3bd87eb0f125271b8831f"}, | ||||||
| ] | ] | ||||||
| more-itertools = [ | more-itertools = [ | ||||||
|     {file = "more-itertools-8.4.0.tar.gz", hash = "sha256:68c70cc7167bdf5c7c9d8f6954a7837089c6a36bf565383919bb595efb8a17e5"}, |     {file = "more-itertools-8.5.0.tar.gz", hash = "sha256:6f83822ae94818eae2612063a5101a7311e68ae8002005b5e05f03fd74a86a20"}, | ||||||
|     {file = "more_itertools-8.4.0-py3-none-any.whl", hash = "sha256:b78134b2063dd214000685165d81c154522c3ee0a1c0d4d113c80361c234c5a2"}, |     {file = "more_itertools-8.5.0-py3-none-any.whl", hash = "sha256:9b30f12df9393f0d28af9210ff8efe48d10c94f73e5daf886f10c4b0b0b4f03c"}, | ||||||
| ] | ] | ||||||
| packaging = [ | packaging = [ | ||||||
|     {file = "packaging-20.4-py2.py3-none-any.whl", hash = "sha256:998416ba6962ae7fbd6596850b80e17859a5753ba17c32284f67bfff33784181"}, |     {file = "packaging-20.4-py2.py3-none-any.whl", hash = "sha256:998416ba6962ae7fbd6596850b80e17859a5753ba17c32284f67bfff33784181"}, | ||||||
| @@ -819,8 +760,8 @@ pathspec = [ | |||||||
|     {file = "pathspec-0.8.0.tar.gz", hash = "sha256:da45173eb3a6f2a5a487efba21f050af2b41948be6ab52b6a1e3ff22bb8b7061"}, |     {file = "pathspec-0.8.0.tar.gz", hash = "sha256:da45173eb3a6f2a5a487efba21f050af2b41948be6ab52b6a1e3ff22bb8b7061"}, | ||||||
| ] | ] | ||||||
| pbr = [ | pbr = [ | ||||||
|     {file = "pbr-5.4.5-py2.py3-none-any.whl", hash = "sha256:579170e23f8e0c2f24b0de612f71f648eccb79fb1322c814ae6b3c07b5ba23e8"}, |     {file = "pbr-5.5.0-py2.py3-none-any.whl", hash = "sha256:5adc0f9fc64319d8df5ca1e4e06eea674c26b80e6f00c530b18ce6a6592ead15"}, | ||||||
|     {file = "pbr-5.4.5.tar.gz", hash = "sha256:07f558fece33b05caf857474a366dfcc00562bca13dd8b47b2b3e22d9f9bf55c"}, |     {file = "pbr-5.5.0.tar.gz", hash = "sha256:14bfd98f51c78a3dd22a1ef45cf194ad79eee4a19e8e1a0d5c7f8e81ffe182ea"}, | ||||||
| ] | ] | ||||||
| pluggy = [ | pluggy = [ | ||||||
|     {file = "pluggy-0.13.1-py2.py3-none-any.whl", hash = "sha256:966c145cd83c96502c3c3868f50408687b38434af77734af1e9ca461a4081d2d"}, |     {file = "pluggy-0.13.1-py2.py3-none-any.whl", hash = "sha256:966c145cd83c96502c3c3868f50408687b38434af77734af1e9ca461a4081d2d"}, | ||||||
| @@ -870,11 +811,11 @@ pyparsing = [ | |||||||
|     {file = "pyparsing-2.4.7.tar.gz", hash = "sha256:c203ec8783bf771a155b207279b9bccb8dea02d8f0c9e5f8ead507bc3246ecc1"}, |     {file = "pyparsing-2.4.7.tar.gz", hash = "sha256:c203ec8783bf771a155b207279b9bccb8dea02d8f0c9e5f8ead507bc3246ecc1"}, | ||||||
| ] | ] | ||||||
| pypika = [ | pypika = [ | ||||||
|     {file = "PyPika-0.38.0.tar.gz", hash = "sha256:abf85d7fc3da6c4213125b58ca989a1eabfcc1e9b1f5fc3f524eba5cd7a25107"}, |     {file = "PyPika-0.40.0.tar.gz", hash = "sha256:659d307f7e531b66813619cbce08ecb97eeb302feabbd816ae8844b99496298b"}, | ||||||
| ] | ] | ||||||
| pytest = [ | pytest = [ | ||||||
|     {file = "pytest-6.0.1-py3-none-any.whl", hash = "sha256:8b6007800c53fdacd5a5c192203f4e531eb2a1540ad9c752e052ec0f7143dbad"}, |     {file = "pytest-6.0.2-py3-none-any.whl", hash = "sha256:0e37f61339c4578776e090c3b8f6b16ce4db333889d65d0efb305243ec544b40"}, | ||||||
|     {file = "pytest-6.0.1.tar.gz", hash = "sha256:85228d75db9f45e06e57ef9bf4429267f81ac7c0d742cc9ed63d09886a9fe6f4"}, |     {file = "pytest-6.0.2.tar.gz", hash = "sha256:c8f57c2a30983f469bf03e68cdfa74dc474ce56b8f280ddcb080dfd91df01043"}, | ||||||
| ] | ] | ||||||
| pytest-asyncio = [ | pytest-asyncio = [ | ||||||
|     {file = "pytest-asyncio-0.14.0.tar.gz", hash = "sha256:9882c0c6b24429449f5f969a5158b528f39bde47dc32e85b9f0403965017e700"}, |     {file = "pytest-asyncio-0.14.0.tar.gz", hash = "sha256:9882c0c6b24429449f5f969a5158b528f39bde47dc32e85b9f0403965017e700"}, | ||||||
| @@ -885,12 +826,12 @@ pytest-forked = [ | |||||||
|     {file = "pytest_forked-1.3.0-py2.py3-none-any.whl", hash = "sha256:dc4147784048e70ef5d437951728825a131b81714b398d5d52f17c7c144d8815"}, |     {file = "pytest_forked-1.3.0-py2.py3-none-any.whl", hash = "sha256:dc4147784048e70ef5d437951728825a131b81714b398d5d52f17c7c144d8815"}, | ||||||
| ] | ] | ||||||
| pytest-mock = [ | pytest-mock = [ | ||||||
|     {file = "pytest-mock-3.2.0.tar.gz", hash = "sha256:7122d55505d5ed5a6f3df940ad174b3f606ecae5e9bc379569cdcbd4cd9d2b83"}, |     {file = "pytest-mock-3.3.1.tar.gz", hash = "sha256:a4d6d37329e4a893e77d9ffa89e838dd2b45d5dc099984cf03c703ac8411bb82"}, | ||||||
|     {file = "pytest_mock-3.2.0-py3-none-any.whl", hash = "sha256:5564c7cd2569b603f8451ec77928083054d8896046830ca763ed68f4112d17c7"}, |     {file = "pytest_mock-3.3.1-py3-none-any.whl", hash = "sha256:024e405ad382646318c4281948aadf6fe1135632bea9cc67366ea0c4098ef5f2"}, | ||||||
| ] | ] | ||||||
| pytest-xdist = [ | pytest-xdist = [ | ||||||
|     {file = "pytest-xdist-1.34.0.tar.gz", hash = "sha256:340e8e83e2a4c0d861bdd8d05c5d7b7143f6eea0aba902997db15c2a86be04ee"}, |     {file = "pytest-xdist-2.1.0.tar.gz", hash = "sha256:82d938f1a24186520e2d9d3a64ef7d9ac7ecdf1a0659e095d18e596b8cbd0672"}, | ||||||
|     {file = "pytest_xdist-1.34.0-py2.py3-none-any.whl", hash = "sha256:ba5d10729372d65df3ac150872f9df5d2ed004a3b0d499cc0164aafedd8c7b66"}, |     {file = "pytest_xdist-2.1.0-py3-none-any.whl", hash = "sha256:7c629016b3bb006b88ac68e2b31551e7becf173c76b977768848e2bbed594d90"}, | ||||||
| ] | ] | ||||||
| pyyaml = [ | pyyaml = [ | ||||||
|     {file = "PyYAML-5.3.1-cp27-cp27m-win32.whl", hash = "sha256:74809a57b329d6cc0fdccee6318f44b9b8649961fa73144a98735b0aaf029f1f"}, |     {file = "PyYAML-5.3.1-cp27-cp27m-win32.whl", hash = "sha256:74809a57b329d6cc0fdccee6318f44b9b8649961fa73144a98735b0aaf029f1f"}, | ||||||
| @@ -935,20 +876,16 @@ smmap = [ | |||||||
|     {file = "smmap-3.0.4-py2.py3-none-any.whl", hash = "sha256:54c44c197c819d5ef1991799a7e30b662d1e520f2ac75c9efbeb54a742214cf4"}, |     {file = "smmap-3.0.4-py2.py3-none-any.whl", hash = "sha256:54c44c197c819d5ef1991799a7e30b662d1e520f2ac75c9efbeb54a742214cf4"}, | ||||||
|     {file = "smmap-3.0.4.tar.gz", hash = "sha256:9c98bbd1f9786d22f14b3d4126894d56befb835ec90cef151af566c7e19b5d24"}, |     {file = "smmap-3.0.4.tar.gz", hash = "sha256:9c98bbd1f9786d22f14b3d4126894d56befb835ec90cef151af566c7e19b5d24"}, | ||||||
| ] | ] | ||||||
| sniffio = [ |  | ||||||
|     {file = "sniffio-1.1.0-py3-none-any.whl", hash = "sha256:20ed6d5b46f8ae136d00b9dcb807615d83ed82ceea6b2058cecb696765246da5"}, |  | ||||||
|     {file = "sniffio-1.1.0.tar.gz", hash = "sha256:8e3810100f69fe0edd463d02ad407112542a11ffdc29f67db2bf3771afb87a21"}, |  | ||||||
| ] |  | ||||||
| stevedore = [ | stevedore = [ | ||||||
|     {file = "stevedore-3.2.0-py3-none-any.whl", hash = "sha256:c8f4f0ebbc394e52ddf49de8bcc3cf8ad2b4425ebac494106bbc5e3661ac7633"}, |     {file = "stevedore-3.2.2-py3-none-any.whl", hash = "sha256:5e1ab03eaae06ef6ce23859402de785f08d97780ed774948ef16c4652c41bc62"}, | ||||||
|     {file = "stevedore-3.2.0.tar.gz", hash = "sha256:38791aa5bed922b0a844513c5f9ed37774b68edc609e5ab8ab8d8fe0ce4315e5"}, |     {file = "stevedore-3.2.2.tar.gz", hash = "sha256:f845868b3a3a77a2489d226568abe7328b5c2d4f6a011cc759dfa99144a521f0"}, | ||||||
| ] | ] | ||||||
| toml = [ | toml = [ | ||||||
|     {file = "toml-0.10.1-py2.py3-none-any.whl", hash = "sha256:bda89d5935c2eac546d648028b9901107a595863cb36bae0c73ac804a9b4ce88"}, |     {file = "toml-0.10.1-py2.py3-none-any.whl", hash = "sha256:bda89d5935c2eac546d648028b9901107a595863cb36bae0c73ac804a9b4ce88"}, | ||||||
|     {file = "toml-0.10.1.tar.gz", hash = "sha256:926b612be1e5ce0634a2ca03470f95169cf16f939018233a670519cb4ac58b0f"}, |     {file = "toml-0.10.1.tar.gz", hash = "sha256:926b612be1e5ce0634a2ca03470f95169cf16f939018233a670519cb4ac58b0f"}, | ||||||
| ] | ] | ||||||
| tortoise-orm = [ | tortoise-orm = [ | ||||||
|     {file = "tortoise-orm-0.16.14.tar.gz", hash = "sha256:d6ffd4f4cd46d469fa946ca5309768f381640b65d5f5c9e848fb9b50185febee"}, |     {file = "tortoise-orm-0.16.16.tar.gz", hash = "sha256:dc0c7909324a6c224076936adb3b1b154a8d2b13ff51e4232f0bdc3d65841185"}, | ||||||
| ] | ] | ||||||
| typed-ast = [ | typed-ast = [ | ||||||
|     {file = "typed_ast-1.4.1-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:73d785a950fc82dd2a25897d525d003f6378d1cb23ab305578394694202a58c3"}, |     {file = "typed_ast-1.4.1-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:73d785a950fc82dd2a25897d525d003f6378d1cb23ab305578394694202a58c3"}, | ||||||
| @@ -974,11 +911,11 @@ typed-ast = [ | |||||||
|     {file = "typed_ast-1.4.1.tar.gz", hash = "sha256:8c8aaad94455178e3187ab22c8b01a3837f8ee50e09cf31f1ba129eb293ec30b"}, |     {file = "typed_ast-1.4.1.tar.gz", hash = "sha256:8c8aaad94455178e3187ab22c8b01a3837f8ee50e09cf31f1ba129eb293ec30b"}, | ||||||
| ] | ] | ||||||
| typing-extensions = [ | typing-extensions = [ | ||||||
|     {file = "typing_extensions-3.7.4.2-py2-none-any.whl", hash = "sha256:f8d2bd89d25bc39dabe7d23df520442fa1d8969b82544370e03d88b5a591c392"}, |     {file = "typing_extensions-3.7.4.3-py2-none-any.whl", hash = "sha256:dafc7639cde7f1b6e1acc0f457842a83e722ccca8eef5270af2d74792619a89f"}, | ||||||
|     {file = "typing_extensions-3.7.4.2-py3-none-any.whl", hash = "sha256:6e95524d8a547a91e08f404ae485bbb71962de46967e1b71a0cb89af24e761c5"}, |     {file = "typing_extensions-3.7.4.3-py3-none-any.whl", hash = "sha256:7cb407020f00f7bfc3cb3e7881628838e69d8f3fcab2f64742a5e76b2f841918"}, | ||||||
|     {file = "typing_extensions-3.7.4.2.tar.gz", hash = "sha256:79ee589a3caca649a9bfd2a8de4709837400dfa00b6cc81962a1e6a1815969ae"}, |     {file = "typing_extensions-3.7.4.3.tar.gz", hash = "sha256:99d4073b617d30288f569d3f13d2bd7548c3a7e4c8de87db09a9d29bb3a4a60c"}, | ||||||
| ] | ] | ||||||
| zipp = [ | zipp = [ | ||||||
|     {file = "zipp-3.1.0-py3-none-any.whl", hash = "sha256:aa36550ff0c0b7ef7fa639055d797116ee891440eac1a56f378e2d3179e0320b"}, |     {file = "zipp-3.2.0-py3-none-any.whl", hash = "sha256:43f4fa8d8bb313e65d8323a3952ef8756bf40f9a5c3ea7334be23ee4ec8278b6"}, | ||||||
|     {file = "zipp-3.1.0.tar.gz", hash = "sha256:c599e4d75c98f6798c509911d08a22e6c021d074469042177c8c86fb92eefd96"}, |     {file = "zipp-3.2.0.tar.gz", hash = "sha256:b52f22895f4cfce194bc8172f3819ee8de7540aa6d873535a8668b730b8b411f"}, | ||||||
| ] | ] | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| [tool.poetry] | [tool.poetry] | ||||||
| name = "aerich" | name = "aerich" | ||||||
| version = "0.2.3" | version = "0.2.5" | ||||||
| description = "A database migrations tool for Tortoise ORM." | description = "A database migrations tool for Tortoise ORM." | ||||||
| authors = ["long2ice <long2ice@gmail.com>"] | authors = ["long2ice <long2ice@gmail.com>"] | ||||||
| license = "Apache-2.0" | license = "Apache-2.0" | ||||||
| @@ -17,7 +17,7 @@ include = ["CHANGELOG.md", "LICENSE", "README.md"] | |||||||
| [tool.poetry.dependencies] | [tool.poetry.dependencies] | ||||||
| python = "^3.7" | python = "^3.7" | ||||||
| tortoise-orm = "*" | tortoise-orm = "*" | ||||||
| asyncclick = "*" | click = "*" | ||||||
| pydantic = "*" | pydantic = "*" | ||||||
| aiomysql = {version = "*", optional = true} | aiomysql = {version = "*", optional = true} | ||||||
| asyncpg = {version = "*", optional = true} | asyncpg = {version = "*", optional = true} | ||||||
|   | |||||||
| @@ -24,13 +24,19 @@ class Status(IntEnum): | |||||||
| class User(Model): | class User(Model): | ||||||
|     username = fields.CharField(max_length=20) |     username = fields.CharField(max_length=20) | ||||||
|     password = fields.CharField(max_length=200) |     password = fields.CharField(max_length=200) | ||||||
|     last_login = fields.DatetimeField(description="Last Login", default=datetime.datetime.now) |     last_login_at = fields.DatetimeField(description="Last Login", default=datetime.datetime.now) | ||||||
|     is_active = fields.BooleanField(default=True, description="Is Active") |     is_active = fields.BooleanField(default=True, description="Is Active") | ||||||
|     is_superuser = fields.BooleanField(default=False, description="Is SuperUser") |     is_superuser = fields.BooleanField(default=False, description="Is SuperUser") | ||||||
|     avatar = fields.CharField(max_length=200, default="") |     avatar = fields.CharField(max_length=200, default="") | ||||||
|     intro = fields.TextField(default="") |     intro = fields.TextField(default="") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Email(Model): | ||||||
|  |     email = fields.CharField(max_length=200) | ||||||
|  |     is_primary = fields.BooleanField(default=False) | ||||||
|  |     user = fields.ForeignKeyField("diff_models.User", db_constraint=True) | ||||||
|  |  | ||||||
|  |  | ||||||
| class Category(Model): | class Category(Model): | ||||||
|     slug = fields.CharField(max_length=200) |     slug = fields.CharField(max_length=200) | ||||||
|     user = fields.ForeignKeyField("diff_models.User", description="User") |     user = fields.ForeignKeyField("diff_models.User", description="User") | ||||||
|   | |||||||
| @@ -31,6 +31,12 @@ class User(Model): | |||||||
|     intro = fields.TextField(default="") |     intro = fields.TextField(default="") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Email(Model): | ||||||
|  |     email = fields.CharField(max_length=200) | ||||||
|  |     is_primary = fields.BooleanField(default=False) | ||||||
|  |     user = fields.ForeignKeyField("models.User", db_constraint=False) | ||||||
|  |  | ||||||
|  |  | ||||||
| class Category(Model): | class Category(Model): | ||||||
|     slug = fields.CharField(max_length=200) |     slug = fields.CharField(max_length=200) | ||||||
|     name = fields.CharField(max_length=200) |     name = fields.CharField(max_length=200) | ||||||
|   | |||||||
| @@ -1,6 +1,9 @@ | |||||||
|  | import pytest | ||||||
|  |  | ||||||
| from aerich.ddl.mysql import MysqlDDL | from aerich.ddl.mysql import MysqlDDL | ||||||
| from aerich.ddl.postgres import PostgresDDL | from aerich.ddl.postgres import PostgresDDL | ||||||
| from aerich.ddl.sqlite import SqliteDDL | from aerich.ddl.sqlite import SqliteDDL | ||||||
|  | from aerich.exceptions import NotSupportError | ||||||
| from aerich.migrate import Migrate | from aerich.migrate import Migrate | ||||||
| from tests.models import Category, User | from tests.models import Category, User | ||||||
|  |  | ||||||
| @@ -63,27 +66,26 @@ def test_add_column(): | |||||||
|  |  | ||||||
|  |  | ||||||
| def test_modify_column(): | def test_modify_column(): | ||||||
|     ret = Migrate.ddl.modify_column(Category, Category._meta.fields_map.get("name")) |     if isinstance(Migrate.ddl, SqliteDDL): | ||||||
|     if isinstance(Migrate.ddl, MysqlDDL): |         with pytest.raises(NotSupportError): | ||||||
|         assert ret == "ALTER TABLE `category` MODIFY COLUMN `name` VARCHAR(200) NOT NULL" |             ret0 = Migrate.ddl.modify_column(Category, Category._meta.fields_map.get("name")) | ||||||
|     elif isinstance(Migrate.ddl, PostgresDDL): |             ret1 = Migrate.ddl.modify_column(User, User._meta.fields_map.get("is_active")) | ||||||
|         assert ret == 'ALTER TABLE "category" ALTER COLUMN "name" TYPE VARCHAR(200)' |  | ||||||
|     else: |     else: | ||||||
|         assert ret == 'ALTER TABLE "category" MODIFY COLUMN "name" VARCHAR(200) NOT NULL' |         ret0 = Migrate.ddl.modify_column(Category, Category._meta.fields_map.get("name")) | ||||||
|  |         ret1 = Migrate.ddl.modify_column(User, User._meta.fields_map.get("is_active")) | ||||||
|  |     if isinstance(Migrate.ddl, MysqlDDL): | ||||||
|  |         assert ret0 == "ALTER TABLE `category` MODIFY COLUMN `name` VARCHAR(200) NOT NULL" | ||||||
|  |     elif isinstance(Migrate.ddl, PostgresDDL): | ||||||
|  |         assert ret0 == 'ALTER TABLE "category" ALTER COLUMN "name" TYPE VARCHAR(200)' | ||||||
|  |  | ||||||
|     ret = Migrate.ddl.modify_column(User, User._meta.fields_map.get("is_active")) |  | ||||||
|     if isinstance(Migrate.ddl, MysqlDDL): |     if isinstance(Migrate.ddl, MysqlDDL): | ||||||
|         assert ( |         assert ( | ||||||
|             ret |             ret1 | ||||||
|             == "ALTER TABLE `user` MODIFY COLUMN `is_active` BOOL NOT NULL  COMMENT 'Is Active' DEFAULT 1" |             == "ALTER TABLE `user` MODIFY COLUMN `is_active` BOOL NOT NULL  COMMENT 'Is Active' DEFAULT 1" | ||||||
|         ) |         ) | ||||||
|     elif isinstance(Migrate.ddl, PostgresDDL): |     elif isinstance(Migrate.ddl, PostgresDDL): | ||||||
|         assert ret == 'ALTER TABLE "user" ALTER COLUMN "is_active" TYPE BOOL' |         assert ret1 == 'ALTER TABLE "user" ALTER COLUMN "is_active" TYPE BOOL' | ||||||
|     else: |  | ||||||
|         assert ( |  | ||||||
|             ret |  | ||||||
|             == 'ALTER TABLE "user" MODIFY COLUMN "is_active" INT NOT NULL  DEFAULT 1 /* Is Active */' |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_alter_column_default(): | def test_alter_column_default(): | ||||||
| @@ -131,10 +133,14 @@ def test_set_comment(): | |||||||
|  |  | ||||||
|  |  | ||||||
| def test_drop_column(): | def test_drop_column(): | ||||||
|     ret = Migrate.ddl.drop_column(Category, "name") |     if isinstance(Migrate.ddl, SqliteDDL): | ||||||
|  |         with pytest.raises(NotSupportError): | ||||||
|  |             ret = Migrate.ddl.drop_column(Category, "name") | ||||||
|  |     else: | ||||||
|  |         ret = Migrate.ddl.drop_column(Category, "name") | ||||||
|     if isinstance(Migrate.ddl, MysqlDDL): |     if isinstance(Migrate.ddl, MysqlDDL): | ||||||
|         assert ret == "ALTER TABLE `category` DROP COLUMN `name`" |         assert ret == "ALTER TABLE `category` DROP COLUMN `name`" | ||||||
|     else: |     elif isinstance(Migrate.ddl, PostgresDDL): | ||||||
|         assert ret == 'ALTER TABLE "category" DROP COLUMN "name"' |         assert ret == 'ALTER TABLE "category" DROP COLUMN "name"' | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,43 +1,61 @@ | |||||||
|  | import pytest | ||||||
|  | from pytest_mock import MockerFixture | ||||||
| from tortoise import Tortoise | from tortoise import Tortoise | ||||||
|  |  | ||||||
| from aerich.ddl.mysql import MysqlDDL | from aerich.ddl.mysql import MysqlDDL | ||||||
| from aerich.ddl.postgres import PostgresDDL | from aerich.ddl.postgres import PostgresDDL | ||||||
|  | from aerich.ddl.sqlite import SqliteDDL | ||||||
|  | from aerich.exceptions import NotSupportError | ||||||
| from aerich.migrate import Migrate | from aerich.migrate import Migrate | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_migrate(): | def test_migrate(mocker: MockerFixture): | ||||||
|  |     mocker.patch("click.prompt", return_value=True) | ||||||
|     apps = Tortoise.apps |     apps = Tortoise.apps | ||||||
|     models = apps.get("models") |     models = apps.get("models") | ||||||
|     diff_models = apps.get("diff_models") |     diff_models = apps.get("diff_models") | ||||||
|     Migrate.diff_models(diff_models, models) |     Migrate.diff_models(diff_models, models) | ||||||
|     Migrate.diff_models(models, diff_models, False) |     if isinstance(Migrate.ddl, SqliteDDL): | ||||||
|  |         with pytest.raises(NotSupportError): | ||||||
|  |             Migrate.diff_models(models, diff_models, False) | ||||||
|  |     else: | ||||||
|  |         Migrate.diff_models(models, diff_models, False) | ||||||
|  |     Migrate._merge_operators() | ||||||
|     if isinstance(Migrate.ddl, MysqlDDL): |     if isinstance(Migrate.ddl, MysqlDDL): | ||||||
|         assert Migrate.upgrade_operators == [ |         assert Migrate.upgrade_operators == [ | ||||||
|  |             "ALTER TABLE `email` DROP FOREIGN KEY `fk_email_user_5b58673d`", | ||||||
|             "ALTER TABLE `category` ADD `name` VARCHAR(200) NOT NULL", |             "ALTER TABLE `category` ADD `name` VARCHAR(200) NOT NULL", | ||||||
|             "ALTER TABLE `user` ADD UNIQUE INDEX `uid_user_usernam_9987ab` (`username`)", |             "ALTER TABLE `user` ADD UNIQUE INDEX `uid_user_usernam_9987ab` (`username`)", | ||||||
|  |             "ALTER TABLE `user` RENAME COLUMN `last_login_at` TO `last_login`", | ||||||
|         ] |         ] | ||||||
|         assert Migrate.downgrade_operators == [ |         assert Migrate.downgrade_operators == [ | ||||||
|             "ALTER TABLE `category` DROP COLUMN `name`", |             "ALTER TABLE `category` DROP COLUMN `name`", | ||||||
|             "ALTER TABLE `user` DROP INDEX `uid_user_usernam_9987ab`", |             "ALTER TABLE `user` DROP INDEX `uid_user_usernam_9987ab`", | ||||||
|  |             "ALTER TABLE `user` RENAME COLUMN `last_login` TO `last_login_at`", | ||||||
|  |             "ALTER TABLE `email` ADD CONSTRAINT `fk_email_user_5b58673d` FOREIGN KEY " | ||||||
|  |             "(`user_id`) REFERENCES `user` (`id`) ON DELETE CASCADE", | ||||||
|         ] |         ] | ||||||
|     elif isinstance(Migrate.ddl, PostgresDDL): |     elif isinstance(Migrate.ddl, PostgresDDL): | ||||||
|         assert Migrate.upgrade_operators == [ |         assert Migrate.upgrade_operators == [ | ||||||
|  |             'ALTER TABLE "email" DROP CONSTRAINT "fk_email_user_5b58673d"', | ||||||
|             'ALTER TABLE "category" ADD "name" VARCHAR(200) NOT NULL', |             'ALTER TABLE "category" ADD "name" VARCHAR(200) NOT NULL', | ||||||
|             'ALTER TABLE "user" ADD CONSTRAINT "uid_user_usernam_9987ab" UNIQUE ("username")', |             'ALTER TABLE "user" ADD CONSTRAINT "uid_user_usernam_9987ab" UNIQUE ("username")', | ||||||
|  |             'ALTER TABLE "user" RENAME COLUMN "last_login_at" TO "last_login"', | ||||||
|         ] |         ] | ||||||
|         assert Migrate.downgrade_operators == [ |         assert Migrate.downgrade_operators == [ | ||||||
|             'ALTER TABLE "category" DROP COLUMN "name"', |             'ALTER TABLE "category" DROP COLUMN "name"', | ||||||
|             'ALTER TABLE "user" DROP CONSTRAINT "uid_user_usernam_9987ab"', |             'ALTER TABLE "user" DROP CONSTRAINT "uid_user_usernam_9987ab"', | ||||||
|  |             'ALTER TABLE "user" RENAME COLUMN "last_login" TO "last_login_at"', | ||||||
|  |             'ALTER TABLE "email" ADD CONSTRAINT "fk_email_user_5b58673d" FOREIGN KEY ("user_id") REFERENCES "user" ("id") ON DELETE CASCADE', | ||||||
|         ] |         ] | ||||||
|     else: |     elif isinstance(Migrate.ddl, SqliteDDL): | ||||||
|         assert Migrate.upgrade_operators == [ |         assert Migrate.upgrade_operators == [ | ||||||
|  |             'ALTER TABLE "email" DROP FOREIGN KEY "fk_email_user_5b58673d"', | ||||||
|             'ALTER TABLE "category" ADD "name" VARCHAR(200) NOT NULL', |             'ALTER TABLE "category" ADD "name" VARCHAR(200) NOT NULL', | ||||||
|             'ALTER TABLE "user" ADD UNIQUE INDEX "uid_user_usernam_9987ab" ("username")', |             'ALTER TABLE "user" ADD UNIQUE INDEX "uid_user_usernam_9987ab" ("username")', | ||||||
|  |             'ALTER TABLE "user" RENAME COLUMN "last_login_at" TO "last_login"', | ||||||
|         ] |         ] | ||||||
|         assert Migrate.downgrade_operators == [ |         assert Migrate.downgrade_operators == [] | ||||||
|             'ALTER TABLE "category" DROP COLUMN "name"', |  | ||||||
|             'ALTER TABLE "user" DROP INDEX "uid_user_usernam_9987ab"', |  | ||||||
|         ] |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_sort_all_version_files(mocker): | def test_sort_all_version_files(mocker): | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user