-
Notifications
You must be signed in to change notification settings - Fork 199
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
goal: results should not (never? in weak small cache?) be stored in an in-memory memo table. so that memo table should be not present in this implementation. instead all memo questions go to the sqlite3 database. this drives some blurring between in-memory caching and disk-based checkpointing: the previous disk based checkpointed model relied on repopulating the in-memory memo table cache... i hit some thread problems when using one sqlite3 connection across threads and the docs are unclear about what I can/cannot do, so i made this open the sqlite3 database on every access. that's probably got quite a performance hit, but its probably enough for basically validating the idea.
- Loading branch information
1 parent
9ff13d7
commit 4726c81
Showing
4 changed files
with
174 additions
and
12 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,118 @@ | ||
import logging | ||
import pickle | ||
import sqlite3 | ||
from concurrent.futures import Future | ||
from pathlib import Path | ||
from typing import Optional, Sequence | ||
|
||
from parsl.dataflow.dflow import DataFlowKernel | ||
from parsl.dataflow.memoization import Memoizer, make_hash | ||
from parsl.dataflow.taskrecord import TaskRecord | ||
|
||
logger = logging.getLogger(__name__) | ||
|
||
|
||
class SQLiteMemoizer(Memoizer): | ||
"""Memoize out of memory into an sqlite3 database. | ||
TODO: probably going to need some kind of shutdown now, to close | ||
the sqlite3 connection. | ||
which might also be useful for driving final checkpoints in the | ||
original impl? | ||
""" | ||
|
||
def start(self, *, dfk: DataFlowKernel, memoize: bool = True, checkpoint_files: Sequence[str], run_dir: str) -> None: | ||
"""TODO: run_dir is the per-workflow run dir, but we need a broader checkpoint context... one level up | ||
by default... get_all_checkpoints uses "runinfo/" as a relative path for that by default so replicating | ||
that choice would do here. likewise I think for monitoring.""" | ||
|
||
self.db_path = Path(dfk.config.run_dir) / "checkpoint.sqlite3" | ||
logger.debug("starting with db_path %r", self.db_path) | ||
|
||
# TODO: api wart... turning memoization on or off should not be part of the plugin API | ||
self.memoize = memoize | ||
|
||
connection = sqlite3.connect(self.db_path) | ||
cursor = connection.cursor() | ||
|
||
cursor.execute("CREATE TABLE IF NOT EXISTS checkpoints(key, result)") | ||
# probably want some index on key because that's what we're doing all the access via. | ||
|
||
connection.commit() | ||
connection.close() | ||
logger.debug("checkpoint table created") | ||
|
||
def close(self): | ||
pass | ||
|
||
def checkpoint(self, tasks: Sequence[TaskRecord]) -> None: | ||
"""All the behaviour for this memoizer is in check_memo and update_memo. | ||
""" | ||
logger.debug("Explicit checkpoint call is a no-op with this memoizer") | ||
|
||
def check_memo(self, task: TaskRecord) -> Optional[Future]: | ||
"""TODO: document this: check_memo is required to set the task hashsum, | ||
if that's how we're going to key checkpoints in update_memo. (that's not | ||
a requirement though: other equalities are available.""" | ||
task_id = task['id'] | ||
|
||
if not self.memoize or not task['memoize']: | ||
task['hashsum'] = None | ||
logger.debug("Task %s will not be memoized", task_id) | ||
return None | ||
|
||
hashsum = make_hash(task) | ||
logger.debug("Task {} has memoization hash {}".format(task_id, hashsum)) | ||
task['hashsum'] = hashsum | ||
|
||
connection = sqlite3.connect(self.db_path) | ||
cursor = connection.cursor() | ||
cursor.execute("SELECT result FROM checkpoints WHERE key = ?", (hashsum, )) | ||
r = cursor.fetchone() | ||
|
||
if r is None: | ||
connection.close() | ||
return None | ||
else: | ||
data = pickle.loads(r[0]) | ||
connection.close() | ||
|
||
memo_fu: Future = Future() | ||
|
||
if data['exception'] is None: | ||
memo_fu.set_result(data['result']) | ||
else: | ||
assert data['result'] is None | ||
memo_fu.set_exception(data['exception']) | ||
|
||
return memo_fu | ||
|
||
def update_memo(self, task: TaskRecord, r: Future) -> None: | ||
logger.debug("updating memo") | ||
|
||
if not self.memoize or not task['memoize'] or 'hashsum' not in task: | ||
logger.debug("preconditions for memo not satisfied") | ||
return | ||
|
||
if not isinstance(task['hashsum'], str): | ||
logger.error(f"Attempting to update app cache entry but hashsum is not a string key: {task['hashsum']}") | ||
return | ||
|
||
app_fu = task['app_fu'] | ||
hashsum = task['hashsum'] | ||
|
||
# this comes from the original concatenation-based checkpoint code: | ||
if app_fu.exception() is None: | ||
t = {'hash': hashsum, 'exception': None, 'result': app_fu.result()} | ||
else: | ||
t = {'hash': hashsum, 'exception': app_fu.exception(), 'result': None} | ||
|
||
value = pickle.dumps(t) | ||
|
||
connection = sqlite3.connect(self.db_path) | ||
cursor = connection.cursor() | ||
|
||
cursor.execute("INSERT INTO checkpoints VALUES(?, ?)", (hashsum, value)) | ||
|
||
connection.commit() | ||
connection.close() |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
44 changes: 44 additions & 0 deletions
44
parsl/tests/test_checkpointing/test_python_checkpoint_2_sqlite.py
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,44 @@ | ||
import contextlib | ||
import os | ||
|
||
import pytest | ||
|
||
import parsl | ||
from parsl import python_app | ||
from parsl.dataflow.memosql import SQLiteMemoizer | ||
from parsl.tests.configs.local_threads_checkpoint import fresh_config | ||
|
||
|
||
@contextlib.contextmanager | ||
def parsl_configured(run_dir, **kw): | ||
c = fresh_config() | ||
c.memoizer = SQLiteMemoizer() | ||
c.run_dir = run_dir | ||
for config_attr, config_val in kw.items(): | ||
setattr(c, config_attr, config_val) | ||
dfk = parsl.load(c) | ||
for ex in dfk.executors.values(): | ||
ex.working_dir = run_dir | ||
yield dfk | ||
|
||
parsl.dfk().cleanup() | ||
|
||
|
||
@python_app(cache=True) | ||
def uuid_app(): | ||
import uuid | ||
return uuid.uuid4() | ||
|
||
|
||
@pytest.mark.local | ||
def test_loading_checkpoint(tmpd_cwd): | ||
"""Load memoization table from previous checkpoint | ||
""" | ||
with parsl_configured(tmpd_cwd, checkpoint_mode="task_exit"): | ||
checkpoint_files = [os.path.join(parsl.dfk().run_dir, "checkpoint")] | ||
result = uuid_app().result() | ||
|
||
with parsl_configured(tmpd_cwd, checkpoint_files=checkpoint_files): | ||
relaunched = uuid_app().result() | ||
|
||
assert result == relaunched, "Expected following call to uuid_app to return cached uuid" |