2016-02-12 46 views
0

我想將舊的sqlite3代碼轉換爲SQL鍊金術。我試圖理解如何最好地處理我的用例。我是數據庫訪問的ORM方法的新手。動態表名與SQLalchemy

我想動態生成基於通用定義的唯一表名。我已閱讀mixins guide以及如何使用typeto dynamically declare classes的帖子,但我仍然不確定如何去做這件事。以下是我迄今爲止:

from sqlalchemy.ext.declarative import declarative_base 
from sqlalchemy import Table, Column, Integer, String, MetaData, ForeignKey 
from sqlalchemy.ext.declarative import declared_attr 

Base = declarative_base() 


class DynamicName(object): 
    @declared_attr 
    def __tablename__(cls): 
     return cls.__name__.lower() 

class Genome(DynamicName, Base): 
    __tablename__ = 'AbstractGenome' 
    AlignmentId = Column(Integer, primary_key=True) 
    StartOutOfFrame = Column(Integer) 
    BadFrame = Column(Integer) 

def build_genome_table(genome): 
    d = {'__tablename__': genome} 
    table = type(genome, (Genome,), d) 
    return table 

如果我嘗試使用這個,這是行不通的:

>>> from sqlalchemy import create_engine 
>>> engine = create_engine('sqlite:///:memory:', echo=True) 
>>> genomes = ["A", "B"] 
>>> tables = {x: build_genome_table(x) for x in genomes} 
Traceback (most recent call last): 
    File "<stdin>", line 1, in <module> 
    File "<stdin>", line 1, in <dictcomp> 
    File "<stdin>", line 3, in build_genome_table 
    File "/cluster/home/ifiddes/python2.7/lib/python2.7/site-packages/sqlalchemy/ext/declarative/api.py", line 55, in __init__ 
    _as_declarative(cls, classname, cls.__dict__) 
    File "/cluster/home/ifiddes/python2.7/lib/python2.7/site-packages/sqlalchemy/ext/declarative/base.py", line 88, in _as_declarative 
    _MapperConfig.setup_mapping(cls, classname, dict_) 
    File "/cluster/home/ifiddes/python2.7/lib/python2.7/site-packages/sqlalchemy/ext/declarative/base.py", line 103, in setup_mapping 
    cfg_cls(cls_, classname, dict_) 
    File "/cluster/home/ifiddes/python2.7/lib/python2.7/site-packages/sqlalchemy/ext/declarative/base.py", line 135, in __init__ 
    self._early_mapping() 
    File "/cluster/home/ifiddes/python2.7/lib/python2.7/site-packages/sqlalchemy/ext/declarative/base.py", line 138, in _early_mapping 
    self.map() 
    File "/cluster/home/ifiddes/python2.7/lib/python2.7/site-packages/sqlalchemy/ext/declarative/base.py", line 529, in map 
    **self.mapper_args 
    File "<string>", line 2, in mapper 
    File "/cluster/home/ifiddes/python2.7/lib/python2.7/site-packages/sqlalchemy/orm/mapper.py", line 623, in __init__ 
    self._configure_inheritance() 
    File "/cluster/home/ifiddes/python2.7/lib/python2.7/site-packages/sqlalchemy/orm/mapper.py", line 930, in _configure_inheritance 
    self.local_table) 
    File "<string>", line 2, in join_condition 
    File "/cluster/home/ifiddes/python2.7/lib/python2.7/site-packages/sqlalchemy/sql/selectable.py", line 839, in _join_condition 
    (a.description, b.description, hint)) 
sqlalchemy.exc.NoForeignKeysError: Can't find any foreign key relationships between 'AbstractGenome' and 'A'. 

我如何去根據傳遞的名稱有關動態生成Genome表?我最好也喜歡一個可以有層次繼承的設置,以便我可以聲明不同的子類,如ReferenceGenomeTargetGenome,它們具有附加列,但也可以具有動態名稱。

+1

SQLAlchemy試圖設置繼承和失敗,因爲'A'表和'AbstractGenome'表之間沒有外鍵。你真的打算創建'A'表和'AbstractGenome'表嗎?如果沒有,你應該在''Genome'定義中加入'__abstract__ = True'。 – univerio

回答

1

參見:sqlalchemy.orm.mapperhttp://docs.sqlalchemy.org/en/latest/orm/mapping_api.html#class-mapping-api。我的理解(因爲我以前只使用這個函數修改過代碼),它直接將模型類映射到Table對象,該對象本身連接到數據庫表。

這個用例實際上聽起來非常類似於配方history_meta:http://docs.sqlalchemy.org/en/latest/_modules/examples/versioned_history/history_meta.html。它可能需要一些時間才能排序,但是基於現有模型(任何版本的子類)動態創建Table對象,然後在創建類時直接映射到數據庫表。

雖然這是問題:您確實需要一個實際的數據庫表來映射到。畢竟它是一個ORM。你有幾個選擇這裏:

  1. 如果要動態創建一個表,將數據庫中的堅持,你可以使用Table.create()按這裏:http://docs.sqlalchemy.org/en/latest/core/metadata.html#creating-and-dropping-database-tables

  2. 如果你只需要時不時地創建表,你可以用蒸餾器集成:https://pypi.python.org/pypi/alembic

  3. 如果你只是需要一個過程,也再沒有,你可以創建臨時表,雖然我不知道,如果SQLAlchemy的直接支持它。我檢查的少量資源似乎是使用create()和drop()。我沒有使用SQLAlchemy 1.0+,所以它可能有一些我沒有見過的支持。

讓我知道,如果有什麼在這裏不明確。自從我玩過history_meta.py之後已經有一段時間了,所以我可能會生鏽。