晋太元中,武陵人捕鱼为业。缘溪行,忘路之远近。忽逢桃花林,夹岸数百步,中无杂树,芳草鲜美,落英缤纷。渔人甚异之,复前行,欲穷其林。 林尽水源,便得一山,山有小口,仿佛若有光。便舍船,从口入。初极狭,才通人。复行数十步,豁然开朗。土地平旷,屋舍俨然,有良田、美池、桑竹之属。阡陌交通,鸡犬相闻。其中往来种作,男女衣着,悉如外人。黄发垂髫,并怡然自乐。 见渔人,乃大惊,问所从来。具答之。便要还家,设酒杀鸡作食。村中闻有此人,咸来问讯。自云先世避秦时乱,率妻子邑人来此绝境,不复出焉,遂与外人间隔。问今是何世,乃不知有汉,无论魏晋。此人一一为具言所闻,皆叹惋。余人各复延至其家,皆出酒食。停数日,辞去。此中人语云:“不足为外人道也。”(间隔 一作:隔绝) 既出,得其船,便扶向路,处处志之。及郡下,诣太守,说如此。太守即遣人随其往,寻向所志,遂迷,不复得路。 南阳刘子骥,高尚士也,闻之,欣然规往。未果,寻病终。后遂无问津者。
|
Server : Apache System : Linux srv.rainic.com 4.18.0-553.47.1.el8_10.x86_64 #1 SMP Wed Apr 2 05:45:37 EDT 2025 x86_64 User : rainic ( 1014) PHP Version : 7.4.33 Disable Function : exec,passthru,shell_exec,system Directory : /proc/thread-self/root/opt/imunify360/venv/lib64/python3.11/site-packages/peewee_migrate/ |
Upload File : |
"""CLI integration."""
from __future__ import annotations
import logging
import re
import sys
from pathlib import Path
from typing import TYPE_CHECKING, List, Optional, Pattern, Union
import click
from playhouse.db_url import connect
from .logs import logger
from .models import MIGRATE_TABLE
from .router import Router
if TYPE_CHECKING:
from peewee_migrate.types import TParams
CLEAN_RE: Pattern = re.compile(r"\s+$", re.M)
VERBOSE: List[str] = ["WARNING", "INFO", "DEBUG", "NOTSET"]
def get_router(
directory: Optional[Union[str, Path]] = None,
database: Optional[str] = None,
migratetable: str = MIGRATE_TABLE,
verbose: int = 0,
) -> Router:
"""Load and initialize a router."""
config: TParams = {}
logging_level: str = VERBOSE[verbose]
ignore = schema = None
if directory:
directory = Path(directory)
try:
with directory.joinpath("conf.py").open() as cfg:
code = compile(cfg.read(), "<string>", "exec", dont_inherit=True)
exec(code, config, config)
database = config.get("DATABASE", database)
ignore = config.get("IGNORE", ignore)
schema = config.get("SCHEMA", schema)
migratetable = config.get("MIGRATE_TABLE", migratetable)
logging_level = config.get("LOGGING_LEVEL", logging_level).upper()
except IOError:
pass
if isinstance(database, str):
database = connect(database)
logger.setLevel(logging_level)
if not database:
logger.error("Database is undefined")
return sys.exit(1)
try:
return Router(
database,
migrate_table=migratetable,
migrate_dir=directory,
ignore=ignore,
schema=schema,
)
except RuntimeError:
logger.exception("Failed to initialize router")
return sys.exit(1)
@click.group()
def cli():
"""Migrate database with Peewee ORM."""
logging.basicConfig(level=logging.INFO)
@cli.command()
@click.option("--name", default=None, help="Select migration")
@click.option("--database", default=None, help="Database connection")
@click.option("--directory", default="migrations", help="Directory where migrations are stored")
@click.option("--fake", is_flag=True, default=False, help="Run migration as fake.")
@click.option("--migratetable", default="migratehistory", help="Migration table.")
@click.option("-v", "--verbose", count=True)
def migrate( # noqa:
name: Optional[str] = None,
database: Optional[str] = None,
directory: Optional[str] = None,
migratetable: str = MIGRATE_TABLE,
verbose: int = 0,
fake: bool = False, # noqa:
):
"""Migrate database."""
router = get_router(directory, database, migratetable, verbose)
migrations = router.run(name, fake=fake)
if migrations:
click.echo("Migrations completed: %s" % ", ".join(migrations))
@cli.command()
@click.argument("name")
@click.option(
"--auto",
default=False,
is_flag=True,
help="Scan sources and create db migrations automatically. Supports autodiscovery.",
)
@click.option(
"--auto-source",
default=None,
help=(
"Set to python module path for changes autoscan (e.g. 'package.models'). "
"Current directory will be recursively scanned by default."
),
)
@click.option("--database", default=None, help="Database connection")
@click.option("--directory", default="migrations", help="Directory where migrations are stored")
@click.option("--migratetable", default="migratehistory", help="Migration table.")
@click.option("-v", "--verbose", count=True)
def create( # noqa:
name: Optional[str] = None,
database: Optional[str] = None,
directory: Optional[str] = None,
migratetable: Optional[str] = None,
verbose: int = 0,
auto: bool = False, # noqa:
auto_source: Optional[str] = None,
):
"""Create a migration."""
router: Router = get_router(directory, database, migratetable or MIGRATE_TABLE, verbose)
router.create(name or "auto", auto=auto_source if auto and auto_source else auto)
@cli.command()
@click.option(
"--count",
required=False,
default=1,
type=int,
help="Number of last migrations to be rolled back.Ignored in case of non-empty name",
)
@click.option("--database", default=None, help="Database connection")
@click.option("--directory", default="migrations", help="Directory where migrations are stored")
@click.option("--migratetable", default="migratehistory", help="Migration table.")
@click.option("-v", "--verbose", count=True)
def rollback(
database: Optional[str] = None,
directory: Optional[str] = None,
migratetable: Optional[str] = None,
verbose: int = 0,
count: int = 1,
):
"""Rollback a migration with the given steps --count of last migrations as integer number"""
router: Router = get_router(directory, database, migratetable or MIGRATE_TABLE, verbose)
if len(router.done) < count:
raise RuntimeError(
"Unable to rollback %s migrations from %s: %s" % (count, len(router.done), router.done)
)
for _ in range(count):
router.rollback()
@cli.command()
@click.option("--database", default=None, help="Database connection")
@click.option("--directory", default="migrations", help="Directory where migrations are stored")
@click.option("--migratetable", default="migratehistory", help="Migration table.")
@click.option("-v", "--verbose", count=True)
def list( # noqa:
database: Optional[str] = None,
directory: Optional[str] = None,
migratetable: Optional[str] = None,
verbose: int = 0,
):
"""List migrations."""
router: Router = get_router(directory, database, migratetable or MIGRATE_TABLE, verbose)
click.echo("Migrations are done:")
click.echo("\n".join(router.done))
click.echo("")
click.echo("Migrations are undone:")
click.echo("\n".join(router.diff))
@cli.command()
@click.option("--database", default=None, help="Database connection")
@click.option("--directory", default="migrations", help="Directory where migrations are stored")
@click.option("--migratetable", default="migratehistory", help="Migration table.")
@click.option("-v", "--verbose", count=True)
def merge(
database: Optional[str] = None,
directory: Optional[str] = None,
migratetable: Optional[str] = None,
verbose: int = 0,
):
"""Merge migrations into one."""
router: Router = get_router(directory, database, migratetable or MIGRATE_TABLE, verbose)
router.merge()