title: 多数据库迁移的艺术:Alembic在复杂环境中的精妙应用

date: 2025/05/11 00:35:52

updated: 2025/05/11 00:35:52

author: cmdragon

excerpt:

现代Web应用中,多数据库场景包括主从架构、多租户系统、混合数据库和分库分表。Alembic支持多数据库配置,通过创建多版本目录结构和修改alembic.ini文件实现。环境脚本(env.py)需改造以支持多数据库迁移。模型定义推荐使用pydantic结合SQLAlchemy ORM。迁移操作包括生成独立脚本、执行迁移和查看历史。常见报错如未初始化版本表、缺少数据库配置和模型类未绑定元数据,均有相应解决方案。

categories:

  • 后端开发
  • FastAPI

tags:

  • 多数据库迁移
  • Alembic配置
  • SQLAlchemy ORM
  • 数据库架构
  • 数据模型
  • 迁移脚本
  • 错误处理


扫描二维码

关注或者微信搜一搜:编程智域 前端至全栈交流与成长

探索数千个预构建的 AI 应用,开启你的下一个伟大创意https://tools.cmdragon.cn/

  1. 多数据库环境下的迁移需求分析

    现代Web应用中常见以下多数据库场景:
  • 主从数据库架构(读写分离)
  • 多租户系统(不同租户使用独立数据库)
  • 混合数据库类型(MySQL+PostgreSQL组合使用)
  • 分库分表架构(水平拆分业务模块)
  1. Alembic多数据库配置方法

    创建多版本目录结构:
project/
├── alembic/
│ ├── versions/
│ │ ├── db1/
│ │ └── db2/
│ └── env.py
├── alembic.ini
└── app/
└── models.py

修改alembic.ini配置:

[alembic]
script_location = alembic [db1]
sqlalchemy.url = postgresql://user:pass@localhost/db1 [db2]
sqlalchemy.url = mysql://user:pass@localhost/db2

环境脚本改造(env.py):

from alembic import context

def run_migrations(engine_name):
config = context.config
section = config.get_section(engine_name)
url = section["sqlalchemy.url"] engine = create_engine(url)
with engine.connect() as connection:
context.configure(
connection=connection,
target_metadata=get_metadata(engine_name),
version_table=f'alembic_version_{engine_name}'
)
with context.begin_transaction():
context.run_migrations(engine_name) if context.is_offline_mode():
run_migrations('db1')
run_migrations('db2')
else:
for engine_name in ['db1', 'db2']:
run_migrations(engine_name)
  1. 模型定义最佳实践

    使用pydantic结合SQLAlchemy ORM:
from sqlalchemy.ext.declarative import declarative_base
from pydantic import BaseModel class UserBase(BaseModel):
email: str
is_active: bool = True class UserCreate(UserBase):
password: str class UserDB(UserBase):
id: int Db1Base = declarative_base()
Db2Base = declarative_base() class Db1User(Db1Base):
__tablename__ = "users"
id = Column(Integer, primary_key=True)
email = Column(String)
password_hash = Column(String)
is_active = Column(Boolean) class Db2Log(Db2Base):
__tablename__ = "logs"
id = Column(Integer, primary_key=True)
action = Column(String)
user_id = Column(Integer)
  1. 多数据库迁移操作指南

    生成独立迁移脚本:
alembic -n db1 revision --autogenerate -m "add users table"
alembic -n db2 revision --autogenerate -m "add logs table"

执行迁移命令:

alembic -n db1 upgrade head
alembic -n db2 upgrade head

查看迁移历史:

alembic -n db1 history --verbose
alembic -n db2 history --verbose
  1. 课后Quiz

    Q1:当需要为第三个数据库添加迁移支持时,应该修改哪些配置文件?

    A) 只需修改alembic.ini

    B) 修改env.py和alembic.ini

    C) 修改models.py和env.py

    D) 需要创建新的版本目录并修改所有配置文件

正确答案:B

解析:需要修改alembic.ini添加新的数据库配置段落,并在env.py中扩展迁移执行逻辑。模型定义可能需要新增基类,但不需要修改现有文件。

Q2:如何为不同数据库设置独立的迁移版本表?

A) 在模型类中指定__version_table__属性

B) 在env.py的context.configure中设置version_table参数

C) 修改alembic.ini的version_table配置项

D) 使用不同的迁移目录结构

正确答案:B

解析:context.configure()中的version_table参数允许为每个数据库引擎指定独立的版本表名称。

  1. 常见报错解决方案

    错误1:sqlalchemy.exc.NoSuchTableError: alembic_version

    原因:目标数据库未初始化迁移版本表

    解决:
alembic -n db1 revision --autogenerate --version-path=alembic/versions/db1
alembic -n db1 stamp head

错误2:KeyError: 'No such section: 'db3'

原因:alembic.ini缺少对应数据库配置

解决:

[db3]
sqlalchemy.url = sqlite:///db3.sqlite

错误3:SAWarning: Class <class 'app.models.Db1User'> does not have a __table__

原因:模型类未正确绑定到元数据

解决:

Db1Base.metadata.create_all(engine)  # 在应用启动时执行

余下文章内容请点击跳转至 个人博客页面 或者 扫码关注或者微信搜一搜:编程智域 前端至全栈交流与成长,阅读完整的文章:多数据库迁移的艺术:Alembic在复杂环境中的精妙应用 | cmdragon's Blog

往期文章归档:

多数据库迁移的艺术:Alembic在复杂环境中的精妙应用的更多相关文章

  1. Flask学习笔记:数据库迁移操作flask-script+alembic/flask-migrate

    数据库迁移是将代码中模型类(即表)的修改同步到数据库中, flask-sqlalchemy的模型类一旦使用create_all()映射到数据库中后,对这个模型类的修改(例如添加了一个新的字段)就不会再 ...

  2. 转 【TTS】AIX平台数据库迁移到Linux--基于RMAN(真实环境)

    [TTS]AIX平台数据库迁移到Linux--基于RMAN(真实环境) http://www.cnblogs.com/lhrbest/articles/5186933.html 各位技术爱好者,看完本 ...

  3. SQL Server中使用数据库快照的方式来完成测试环境中数据库的轻量级备份还原操作

    在开发或者测试环境的数据库中,经常会发现有开发或者测试人员误删除表或者数据的情况,对于开发或者测试库,一般都没有安排定时的备份任务去备份数据库,一方面是由于存储资源有限,不太可能给开发或者测试环境准备 ...

  4. Code First 数据库迁移

    当 Entity Framework Code First 的数据模型发生改变时,默认会引发一个System.InvalidOperationException 的异常.解决方法是使用DropCrea ...

  5. EntityFramework CodeFirst 数据库迁移

    参考: https://msdn.microsoft.com/en-us/data/jj591621 https://msdn.microsoft.com/en-us/library/dd394698 ...

  6. 使用 Code First 数据库迁移

    当 Entity Framework Code First 的数据模型发生改变时,默认会引发一个System.InvalidOperationException 的异常.解决方法是使用DropCrea ...

  7. 数据库上云实践:使用Ora2pg进行数据库迁移

    目录 概述 重要 前置条件 配置环境 Win环境配置 linux环境配置 定义ORACLE_HOME环境变量 Ora2Pg使用方法 配置文件ora2pg_table.conf 配置文件解释:Oracl ...

  8. Flask从入门到精通之使用Flask-Migrate实现数据库迁移

    在开发程序的过程中,你会发现有时需要修改数据库模型,而且修改之后还需要更新数据库.仅当数据库表不存在时,Flask-SQLAlchemy 才会根据模型进行创建.因此,更新表的唯一方式就是先删除旧表,不 ...

  9. Flask flask-migrate 数据库迁移

    简介 flask-migrate是flask的一个扩展模块,主要是扩展数据库表结构的. 官方文档:http://flask-migrate.readthedocs.io/en/latest/ 使用: ...

  10. EF Code First Migrations数据库迁移

    1.EF Code First创建数据库 新建控制台应用程序Portal,通过程序包管理器控制台添加EntityFramework. 在程序包管理器控制台中执行以下语句,安装EntityFramewo ...

随机推荐

  1. 实战AI大模型辅助编程:新安江水文模型和SCE-UA优化算法的移植与实现

    新安江水文模型与 SCE-UA 优化算法是水文学和水资源管理领域的重要工具,二者结合使用可以有效模拟流域的水文过程并优化模型参数. 新安江水文模型是一种概念性水文模型,主要用于模拟流域的降雨-径流关系 ...

  2. ubuntu更换国内镜像源备忘

    源的路径: /etc/apt/sources.list 更换前备份一下: sudo cp /etc/apt/sources.list /etc/apt/sources_init.list 打开文档,修 ...

  3. mysql 卸载安装教程链接

    https://blog.csdn.net/weixin_56952690/article/details/129678685 https://blog.51cto.com/u_16213646/70 ...

  4. leaflet生成地图封装成jquery插件使用

    公司业务里一直都有使用leaflet地图插件来做地图展示.绘图等操作.公司有个项目已经有好几年了,由于项目原因一直在使用,今年由于google 地图 api过期,导致已经使用的地图无法加载.我作为现在 ...

  5. 给react native 添加transform translateY动画报错:Transform with key of "translateY" must be a number:{translateY“:0}

    初学react native,想实现一个相机扫描功能时,报错,报错描述如标题 这是我的主要逻辑代码 const fadeAnim = useRef(new Animated.Value(0)).cur ...

  6. Mac下打开进入/usr/local等隐藏目录

    教程 Mac下/usr/local目录默认是对于Finder是隐藏,如果需要到/usr/local下去,打开Finder,然后使用command+shift+G,在弹出的目录中填写/usr/local ...

  7. go 组合函数 Collection

    我们经常需要程序在数据集上执行操作,比如选择满足给定条件的所有项,或者将所有的项通过一个自定义函数映射到一个新的集合上. 在某些语言中,会习惯使用泛型. Go 不支持泛型,在 Go 中,当你的程序或者 ...

  8. vue学习一(指令1.v-text,v-html,插值表达式{{msg}})

    一.1.v-text,v-html,插值表达式{{msg}} 注:v-text解决差值表达式闪烁问题,因为他是属性不是差值表达式 1.1.v-text: 是没有闪烁问题的,会覆盖标签的元素中原本的内容 ...

  9. 【Linux】shell 脚本 (.sh) 编写及执行

    shell脚本 shell脚本就是一些命令的集合 #!/bin/bash echo "文件开头代表:该文件使用的是bash语法" 一.运行.sh文件 方法一:当前文件执行.sh 文 ...

  10. Python 生成器说明

    生成器 python 生成器 常规形态 # list def square_numbers(nums: list): squared_nums = [] for i in nums: squared_ ...