Как я могу скопировать базу данных SQLite в памяти в другую базу данных SQLite в памяти на Python? - PullRequest
4 голосов
/ 08 ноября 2011

Я пишу набор тестов для Django, который запускает тесты в виде дерева.Например, у Testcase A может быть 2 результата, а у Testcase B может быть 1, а у Testcase C может быть 3. Дерево выглядит так

      X
     /
A-B-C-X
 \   \
  B   X
   \   X
    \ /
     C-X
      \
       X

Для каждого пути в дереве выше содержимое базы данныхбудь другим.Так что на каждой развилке я думаю о создании в памяти копии текущего состояния базы данных, а затем о передаче этого параметра в следующий тест.

У любого есть идея о том, как по существу скопироватьв базу данных в памяти к другой, а затем получить ссылку для передачи этой базы данных?

Спасибо!

Ответы [ 2 ]

4 голосов
/ 09 ноября 2011

Хорошо, после веселого приключения я понял это.

from django.db import connections
import sqlite3

# Create a Django database connection for our test database
connections.databases['test'] = {'NAME': ":memory:", 'ENGINE': "django.db.backends.sqlite3"}

# We assume that the database under the source_wrapper hasn't been created
source_wrapper = connections['default']  # put alias of source db here
target_wrapper = connections['test'] 

# Create the tables for the source database
source_wrapper.creation.create_test_db()

# Dump the database into a single text query
query = "".join(line for line in source_wrapper.connection.iterdump())

# Generate an in-memory sqlite connection
target_wrapper.connection = sqlite3.connect(":memory:")
target_wrapper.connection.executescript(query)

И теперь база данных с именем test будет точной копией базы данных default. Используйте target_wrapper.connection в качестве ссылки на вновь созданную базу данных.

2 голосов
/ 03 июля 2015

Вот функция, которая копирует базы данных.И источник, и место назначения могут быть в памяти или на диске (местом назначения по умолчанию является копия в памяти):

import sqlite3

def copy_database(source_connection, dest_dbname=':memory:'):
    '''Return a connection to a new copy of an existing database.                        
       Raises an sqlite3.OperationalError if the destination already exists.             
    '''
    script = ''.join(source_connection.iterdump())
    dest_conn = sqlite3.connect(dest_dbname)
    dest_conn.executescript(script)
    return dest_conn

И вот пример того, как это применяется к вашему варианту использования:1004 *

from contextlib import closing

with closing(sqlite3.connect('root_physical.db')) as on_disk_start:
    in_mem_start = copy_database(on_disk_start)

a1 = testcase_a_outcome1(copy_database(in_mem_start))
a2 = testcase_a_outcome1(copy_database(in_mem_start))
a1b = test_case_b(a1)
a2b = test_case_b(a2)
a1bc1 = test_case_c_outcome1(copy_database(a1b))
a1bc2 = test_case_c_outcome2(copy_database(a1b))
a1bc3 = test_case_c_outcome3(copy_database(a1b))
a2bc1 = test_case_c_outcome1(copy_database(a2b))
a2bc2 = test_case_c_outcome2(copy_database(a2b))
a2bc3 = test_case_c_outcome3(copy_database(a2b))
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...