2020-07-27 18:57:36 +00:00
|
|
|
"""
|
|
|
|
This module defines what sandbox functions are made available to the Node controller,
|
|
|
|
and starts the grist sandbox. See engine.py for the API documentation.
|
|
|
|
"""
|
(core) get all tests working under python3/gvisor
Summary:
This verifies that all existing tests are capable of running under python3/gvisor, and fixes the small issues that came up. It does not yet activate python3 tests on all diffs, only diffs that specifically request them.
* Adds a suffix in test names and output directories for tests run with PYTHON_VERSION=3, so that results of the same test run with and without the flag can be aggregated cleanly.
* Adds support for checkpointing to the gvisor sandbox adapter.
* Prepares a checkpoint made after grist python code has loaded in the gvisor sandbox.
* Changes how `DOC_URL` is passed to the sandbox, since it can no longer be passed in as an environment variable when using checkpoints.
* Uses the checkpoint to speed up tests using the gvisor sandbox, otherwise a lot of tests need more time (especially on mac under docker).
* Directs jenkins to run all tests with python2 and python3 when a new file `buildtools/changelogs/python.txt` is touched (this diff counts as touching that file).
* Tweaks miscellaneous tests
- some needed fixes exposed by slightly different timing
- a small number actually give different results in py3 (removal of `u` prefixes).
- some needed a little more time
The DOC_URL change is not the ultimate solution we want for DOC_URL. Eventually it should be a variable that gets updated, like the date perhaps. This is just a small pragmatic change to preserve existing behavior.
Tests are run mindlessly as py3, and for some tests it won't change anything (e.g. if they do not use NSandbox). Tests are not run in parallel, doubling overall test time.
Checkpoints could be useful in deployment, though this diff doesn't use them there.
The application of checkpoints doesn't check for other configuration like 3-versus-5-pipe that we don't actually use.
Python2 tests run using pynbox as always for now.
The diff got sufficiently bulky that I didn't tackle running py3 on "regular" diffs in it. My preference, given that most tests don't appear to stress the python side of things, would be to make a selection of the tests that do and a few wild cards, and run those tests on both pythons rather then all of them. For diffs making a significant python change, I'd propose touching buildtools/changelogs/python.txt for full tests. But this is a conversation in progress.
A total of 6886 tests ran on this diff.
Test Plan: this is a step in preparing tests for py3 transition
Reviewers: dsagal
Reviewed By: dsagal
Subscribers: dsagal
Differential Revision: https://phab.getgrist.com/D3066
2021-10-18 17:37:51 +00:00
|
|
|
import os
|
2022-07-27 18:19:41 +00:00
|
|
|
import random
|
2020-07-27 18:57:36 +00:00
|
|
|
import sys
|
2024-04-18 12:13:16 +00:00
|
|
|
|
|
|
|
from timing import DummyTiming, Timing
|
2020-07-27 18:57:36 +00:00
|
|
|
sys.path.append('thirdparty')
|
|
|
|
# pylint: disable=wrong-import-position
|
|
|
|
|
2022-04-26 05:50:57 +00:00
|
|
|
import logging
|
2020-07-27 18:57:36 +00:00
|
|
|
import marshal
|
|
|
|
import functools
|
|
|
|
|
2021-06-22 15:12:25 +00:00
|
|
|
import six
|
|
|
|
|
2020-07-27 18:57:36 +00:00
|
|
|
import actions
|
|
|
|
import engine
|
2023-02-08 15:46:34 +00:00
|
|
|
import formula_prompt
|
2020-07-27 18:57:36 +00:00
|
|
|
import migrations
|
|
|
|
import schema
|
|
|
|
import useractions
|
|
|
|
import objtypes
|
2024-04-26 20:34:16 +00:00
|
|
|
from predicate_formula import parse_predicate_formula
|
2021-08-09 14:51:43 +00:00
|
|
|
from sandbox import get_default_sandbox
|
|
|
|
from imports.register import register_import_parsers
|
2020-07-27 18:57:36 +00:00
|
|
|
|
2023-07-18 15:20:02 +00:00
|
|
|
# Handler for logging, which flushes each message.
|
|
|
|
class FlushingStreamHandler(logging.StreamHandler):
|
|
|
|
def emit(self, record):
|
|
|
|
super(FlushingStreamHandler, self).emit(record)
|
|
|
|
self.flush()
|
2020-07-27 18:57:36 +00:00
|
|
|
|
2023-07-18 15:20:02 +00:00
|
|
|
# Configure logging module to produce messages with log level and logger name.
|
|
|
|
logging.basicConfig(format="[%(levelname)s] [%(name)s] %(message)s",
|
|
|
|
handlers=[FlushingStreamHandler(sys.stderr)],
|
|
|
|
level=logging.INFO)
|
|
|
|
|
|
|
|
# The default level is INFO. If a different level is desired, add a call like this:
|
|
|
|
# log.setLevel(logging.WARNING)
|
|
|
|
log = logging.getLogger(__name__)
|
2022-04-26 05:50:57 +00:00
|
|
|
|
2020-07-27 18:57:36 +00:00
|
|
|
def table_data_from_db(table_name, table_data_repr):
|
|
|
|
if table_data_repr is None:
|
|
|
|
return actions.TableData(table_name, [], {})
|
|
|
|
table_data_parsed = marshal.loads(table_data_repr)
|
2021-06-24 12:23:33 +00:00
|
|
|
table_data_parsed = {key.decode("utf8"): value for key, value in table_data_parsed.items()}
|
2020-07-27 18:57:36 +00:00
|
|
|
id_col = table_data_parsed.pop("id")
|
|
|
|
return actions.TableData(table_name, id_col,
|
|
|
|
actions.decode_bulk_values(table_data_parsed, _decode_db_value))
|
|
|
|
|
|
|
|
def _decode_db_value(value):
|
|
|
|
# Decode database values received from SQLite's allMarshal() call. These are encoded by
|
|
|
|
# marshalling certain types and storing as BLOBs (received in Python as binary strings, as
|
|
|
|
# opposed to text which is received as unicode). See also encodeValue() in DocStorage.js
|
|
|
|
t = type(value)
|
2021-06-24 12:23:33 +00:00
|
|
|
if t == six.binary_type:
|
2020-07-27 18:57:36 +00:00
|
|
|
return objtypes.decode_object(marshal.loads(value))
|
|
|
|
else:
|
|
|
|
return value
|
|
|
|
|
2021-06-30 13:35:05 +00:00
|
|
|
def run(sandbox):
|
2020-07-27 18:57:36 +00:00
|
|
|
eng = engine.Engine()
|
|
|
|
|
2021-06-30 13:35:05 +00:00
|
|
|
def export(method):
|
|
|
|
# Wrap each method so that it logs a message that it's being called.
|
|
|
|
@functools.wraps(method)
|
|
|
|
def wrapper(*args, **kwargs):
|
2023-07-18 15:20:02 +00:00
|
|
|
log.debug("calling %s", method.__name__)
|
2021-06-30 13:35:05 +00:00
|
|
|
return method(*args, **kwargs)
|
|
|
|
|
|
|
|
sandbox.register(method.__name__, wrapper)
|
|
|
|
|
(core) Log statistics about table sizes
Summary:
Record numbers of rows, columns, cells, and bytes of marshalled data for most calls to table_data_from_db
Export new function get_table_stats in the sandbox, which gives the raw numbers and totals.
Get and log these stats in ActiveDoc right after loading tables, before Calculate, so they are logged even in case of errors.
Tweak logging about number of tables, especially number of on-demand tables, to not only show in debug logging.
Test Plan: Updated doc regression tests, that shows what the data looks like nicely.
Reviewers: dsagal, paulfitz
Reviewed By: dsagal
Differential Revision: https://phab.getgrist.com/D3081
2021-10-21 13:03:37 +00:00
|
|
|
def load_and_record_table_data(table_name, table_data_repr):
|
|
|
|
result = table_data_from_db(table_name, table_data_repr)
|
|
|
|
eng.record_table_stats(result, table_data_repr)
|
|
|
|
return result
|
|
|
|
|
2020-07-27 18:57:36 +00:00
|
|
|
@export
|
2021-07-15 00:45:53 +00:00
|
|
|
def apply_user_actions(action_reprs, user=None):
|
|
|
|
action_group = eng.apply_user_actions([useractions.from_repr(u) for u in action_reprs], user)
|
2022-06-17 18:49:18 +00:00
|
|
|
result = dict(
|
2022-02-21 14:19:11 +00:00
|
|
|
rowCount=eng.count_rows(),
|
|
|
|
**eng.acl_split(action_group).to_json_obj()
|
|
|
|
)
|
2022-06-17 18:49:18 +00:00
|
|
|
if action_group.requests:
|
|
|
|
result["requests"] = action_group.requests
|
|
|
|
return result
|
2020-07-27 18:57:36 +00:00
|
|
|
|
|
|
|
@export
|
|
|
|
def fetch_table(table_id, formulas=True, query=None):
|
|
|
|
return actions.get_action_repr(eng.fetch_table(table_id, formulas=formulas, query=query))
|
|
|
|
|
|
|
|
@export
|
|
|
|
def fetch_table_schema():
|
|
|
|
return eng.fetch_table_schema()
|
|
|
|
|
|
|
|
@export
|
(core) Show example values in formula autocomplete
Summary:
This diff adds a preview of the value of certain autocomplete suggestions, especially of the form `$foo.bar` or `user.email`. The main initial motivation was to show the difference between `$Ref` and `$Ref.DisplayCol`, but the feature is more general.
The client now sends the row ID of the row being edited (along with the table and column IDs which were already sent) to the server to fetch autocomplete suggestions. The returned suggestions are now tuples `(suggestion, example_value)` where `example_value` is a string or null. The example value is simply obtained by evaluating (in a controlled way) the suggestion in the context of the given record and the current user. The string representation is similar to the standard `repr` but dates and datetimes are formatted, and the whole thing is truncated for efficiency.
The example values are shown in the autocomplete popup separated from the actual suggestion by a number of spaces calculated to:
1. Clearly separate the suggestion from the values
2. Left-align the example values in most cases
3. Avoid having so much space such that connecting suggestions and values becomes visually difficult.
The tokenization of the row is then tweaked to show the example in light grey to deemphasise it.
Main discussion where the above was decided: https://grist.slack.com/archives/CDHABLZJT/p1661795588100009
The diff also includes various other small improvements and fixes:
- The autocomplete popup is much wider to make room for long suggestions, particularly lookups, as pointed out in https://phab.getgrist.com/D3580#inline-41007. The wide popup is the reason a fancy solution was needed to position the example values. I didn't see a way to dynamically resize the popup based on suggestions, and it didn't seem like a good idea to try.
- The `grist` and `python` labels previously shown on the right are removed. They were not helpful (https://grist.slack.com/archives/CDHABLZJT/p1659697086155179) and would get in the way of the example values.
- Fixed a bug in our custom tokenization that caused function arguments to be weirdly truncated in the middle: https://grist.slack.com/archives/CDHABLZJT/p1661956353699169?thread_ts=1661953258.342739&cid=CDHABLZJT and https://grist.slack.com/archives/C069RUP71/p1659696778991339
- Hide suggestions involving helper columns like `$gristHelper_Display` or `Table.lookupRecords(gristHelper_Display=` (https://grist.slack.com/archives/CDHABLZJT/p1661953258342739). The former has been around for a while and seems to be a mistake. The fix is simply to use `is_visible_column` instead of `is_user_column`. Since the latter is not used anywhere else, and using it in the first place seems like a mistake more than anything else, I've also removed the function to prevent similar mistakes in the future.
- Don't suggest private columns as lookup arguments: https://grist.slack.com/archives/CDHABLZJT/p1662133416652499?thread_ts=1661795588.100009&cid=CDHABLZJT
- Only fetch fresh suggestions specifically after typing `lookupRecords(` or `lookupOne(` rather than just `(`, as this would needlessly hide function suggestions which could still be useful to see the arguments. However this only makes a difference when there are still multiple matching suggestions, otherwise Ace hides them anyway.
Test Plan: Extended and updated several Python and browser tests.
Reviewers: paulfitz
Reviewed By: paulfitz
Differential Revision: https://phab.getgrist.com/D3611
2022-09-28 14:47:55 +00:00
|
|
|
def autocomplete(txt, table_id, column_id, row_id, user):
|
|
|
|
return eng.autocomplete(txt, table_id, column_id, row_id, user)
|
2020-07-27 18:57:36 +00:00
|
|
|
|
|
|
|
@export
|
|
|
|
def find_col_from_values(values, n, opt_table_id):
|
|
|
|
return eng.find_col_from_values(values, n, opt_table_id)
|
|
|
|
|
|
|
|
@export
|
|
|
|
def fetch_meta_tables(formulas=True):
|
|
|
|
return {table_id: actions.get_action_repr(table_data)
|
2021-06-22 15:12:25 +00:00
|
|
|
for (table_id, table_data) in six.iteritems(eng.fetch_meta_tables(formulas))}
|
2020-07-27 18:57:36 +00:00
|
|
|
|
|
|
|
@export
|
|
|
|
def load_meta_tables(meta_tables, meta_columns):
|
(core) Log statistics about table sizes
Summary:
Record numbers of rows, columns, cells, and bytes of marshalled data for most calls to table_data_from_db
Export new function get_table_stats in the sandbox, which gives the raw numbers and totals.
Get and log these stats in ActiveDoc right after loading tables, before Calculate, so they are logged even in case of errors.
Tweak logging about number of tables, especially number of on-demand tables, to not only show in debug logging.
Test Plan: Updated doc regression tests, that shows what the data looks like nicely.
Reviewers: dsagal, paulfitz
Reviewed By: dsagal
Differential Revision: https://phab.getgrist.com/D3081
2021-10-21 13:03:37 +00:00
|
|
|
return eng.load_meta_tables(load_and_record_table_data("_grist_Tables", meta_tables),
|
|
|
|
load_and_record_table_data("_grist_Tables_column", meta_columns))
|
2020-07-27 18:57:36 +00:00
|
|
|
|
|
|
|
@export
|
|
|
|
def load_table(table_name, table_data):
|
(core) Log statistics about table sizes
Summary:
Record numbers of rows, columns, cells, and bytes of marshalled data for most calls to table_data_from_db
Export new function get_table_stats in the sandbox, which gives the raw numbers and totals.
Get and log these stats in ActiveDoc right after loading tables, before Calculate, so they are logged even in case of errors.
Tweak logging about number of tables, especially number of on-demand tables, to not only show in debug logging.
Test Plan: Updated doc regression tests, that shows what the data looks like nicely.
Reviewers: dsagal, paulfitz
Reviewed By: dsagal
Differential Revision: https://phab.getgrist.com/D3081
2021-10-21 13:03:37 +00:00
|
|
|
return eng.load_table(load_and_record_table_data(table_name, table_data))
|
|
|
|
|
|
|
|
@export
|
|
|
|
def get_table_stats():
|
|
|
|
return eng.get_table_stats()
|
2020-07-27 18:57:36 +00:00
|
|
|
|
|
|
|
@export
|
2020-11-11 21:29:11 +00:00
|
|
|
def create_migrations(all_tables, metadata_only=False):
|
2020-07-27 18:57:36 +00:00
|
|
|
doc_actions = migrations.create_migrations(
|
2021-06-22 15:12:25 +00:00
|
|
|
{t: table_data_from_db(t, data) for t, data in six.iteritems(all_tables)}, metadata_only)
|
|
|
|
return [actions.get_action_repr(action) for action in doc_actions]
|
2020-07-27 18:57:36 +00:00
|
|
|
|
|
|
|
@export
|
|
|
|
def get_version():
|
|
|
|
return schema.SCHEMA_VERSION
|
|
|
|
|
(core) get all tests working under python3/gvisor
Summary:
This verifies that all existing tests are capable of running under python3/gvisor, and fixes the small issues that came up. It does not yet activate python3 tests on all diffs, only diffs that specifically request them.
* Adds a suffix in test names and output directories for tests run with PYTHON_VERSION=3, so that results of the same test run with and without the flag can be aggregated cleanly.
* Adds support for checkpointing to the gvisor sandbox adapter.
* Prepares a checkpoint made after grist python code has loaded in the gvisor sandbox.
* Changes how `DOC_URL` is passed to the sandbox, since it can no longer be passed in as an environment variable when using checkpoints.
* Uses the checkpoint to speed up tests using the gvisor sandbox, otherwise a lot of tests need more time (especially on mac under docker).
* Directs jenkins to run all tests with python2 and python3 when a new file `buildtools/changelogs/python.txt` is touched (this diff counts as touching that file).
* Tweaks miscellaneous tests
- some needed fixes exposed by slightly different timing
- a small number actually give different results in py3 (removal of `u` prefixes).
- some needed a little more time
The DOC_URL change is not the ultimate solution we want for DOC_URL. Eventually it should be a variable that gets updated, like the date perhaps. This is just a small pragmatic change to preserve existing behavior.
Tests are run mindlessly as py3, and for some tests it won't change anything (e.g. if they do not use NSandbox). Tests are not run in parallel, doubling overall test time.
Checkpoints could be useful in deployment, though this diff doesn't use them there.
The application of checkpoints doesn't check for other configuration like 3-versus-5-pipe that we don't actually use.
Python2 tests run using pynbox as always for now.
The diff got sufficiently bulky that I didn't tackle running py3 on "regular" diffs in it. My preference, given that most tests don't appear to stress the python side of things, would be to make a selection of the tests that do and a few wild cards, and run those tests on both pythons rather then all of them. For diffs making a significant python change, I'd propose touching buildtools/changelogs/python.txt for full tests. But this is a conversation in progress.
A total of 6886 tests ran on this diff.
Test Plan: this is a step in preparing tests for py3 transition
Reviewers: dsagal
Reviewed By: dsagal
Subscribers: dsagal
Differential Revision: https://phab.getgrist.com/D3066
2021-10-18 17:37:51 +00:00
|
|
|
@export
|
2022-07-27 18:19:41 +00:00
|
|
|
def initialize(doc_url):
|
|
|
|
if os.environ.get("DETERMINISTIC_MODE"):
|
|
|
|
random.seed(1)
|
|
|
|
else:
|
|
|
|
# Make sure we have randomness, even if we are being cloned from a checkpoint
|
|
|
|
random.seed()
|
|
|
|
if doc_url:
|
|
|
|
os.environ['DOC_URL'] = doc_url
|
(core) get all tests working under python3/gvisor
Summary:
This verifies that all existing tests are capable of running under python3/gvisor, and fixes the small issues that came up. It does not yet activate python3 tests on all diffs, only diffs that specifically request them.
* Adds a suffix in test names and output directories for tests run with PYTHON_VERSION=3, so that results of the same test run with and without the flag can be aggregated cleanly.
* Adds support for checkpointing to the gvisor sandbox adapter.
* Prepares a checkpoint made after grist python code has loaded in the gvisor sandbox.
* Changes how `DOC_URL` is passed to the sandbox, since it can no longer be passed in as an environment variable when using checkpoints.
* Uses the checkpoint to speed up tests using the gvisor sandbox, otherwise a lot of tests need more time (especially on mac under docker).
* Directs jenkins to run all tests with python2 and python3 when a new file `buildtools/changelogs/python.txt` is touched (this diff counts as touching that file).
* Tweaks miscellaneous tests
- some needed fixes exposed by slightly different timing
- a small number actually give different results in py3 (removal of `u` prefixes).
- some needed a little more time
The DOC_URL change is not the ultimate solution we want for DOC_URL. Eventually it should be a variable that gets updated, like the date perhaps. This is just a small pragmatic change to preserve existing behavior.
Tests are run mindlessly as py3, and for some tests it won't change anything (e.g. if they do not use NSandbox). Tests are not run in parallel, doubling overall test time.
Checkpoints could be useful in deployment, though this diff doesn't use them there.
The application of checkpoints doesn't check for other configuration like 3-versus-5-pipe that we don't actually use.
Python2 tests run using pynbox as always for now.
The diff got sufficiently bulky that I didn't tackle running py3 on "regular" diffs in it. My preference, given that most tests don't appear to stress the python side of things, would be to make a selection of the tests that do and a few wild cards, and run those tests on both pythons rather then all of them. For diffs making a significant python change, I'd propose touching buildtools/changelogs/python.txt for full tests. But this is a conversation in progress.
A total of 6886 tests ran on this diff.
Test Plan: this is a step in preparing tests for py3 transition
Reviewers: dsagal
Reviewed By: dsagal
Subscribers: dsagal
Differential Revision: https://phab.getgrist.com/D3066
2021-10-18 17:37:51 +00:00
|
|
|
|
2020-07-27 18:57:36 +00:00
|
|
|
@export
|
|
|
|
def get_formula_error(table_id, col_id, row_id):
|
|
|
|
return objtypes.encode_object(eng.get_formula_error(table_id, col_id, row_id))
|
|
|
|
|
2023-02-08 15:46:34 +00:00
|
|
|
@export
|
2024-02-12 03:11:06 +00:00
|
|
|
def get_formula_prompt(table_id, col_id, description, include_all_tables=True, lookups=True):
|
|
|
|
return formula_prompt.get_formula_prompt(eng, table_id, col_id, description,
|
|
|
|
include_all_tables, lookups)
|
2023-02-08 15:46:34 +00:00
|
|
|
|
|
|
|
@export
|
|
|
|
def convert_formula_completion(completion):
|
|
|
|
return formula_prompt.convert_completion(completion)
|
|
|
|
|
2023-07-24 18:56:38 +00:00
|
|
|
@export
|
|
|
|
def evaluate_formula(table_id, col_id, row_id):
|
|
|
|
return formula_prompt.evaluate_formula(eng, table_id, col_id, row_id)
|
|
|
|
|
2024-04-18 12:13:16 +00:00
|
|
|
@export
|
|
|
|
def start_timing():
|
|
|
|
eng._timing = Timing()
|
|
|
|
|
|
|
|
@export
|
|
|
|
def stop_timing():
|
|
|
|
stats = eng._timing.get()
|
|
|
|
eng._timing = DummyTiming()
|
|
|
|
return stats
|
|
|
|
|
|
|
|
@export
|
|
|
|
def get_timings():
|
|
|
|
return eng._timing.get()
|
|
|
|
|
2024-04-26 20:34:16 +00:00
|
|
|
export(parse_predicate_formula)
|
2020-07-27 18:57:36 +00:00
|
|
|
export(eng.load_empty)
|
|
|
|
export(eng.load_done)
|
|
|
|
|
2021-08-09 14:51:43 +00:00
|
|
|
register_import_parsers(sandbox)
|
|
|
|
|
(core) get all tests working under python3/gvisor
Summary:
This verifies that all existing tests are capable of running under python3/gvisor, and fixes the small issues that came up. It does not yet activate python3 tests on all diffs, only diffs that specifically request them.
* Adds a suffix in test names and output directories for tests run with PYTHON_VERSION=3, so that results of the same test run with and without the flag can be aggregated cleanly.
* Adds support for checkpointing to the gvisor sandbox adapter.
* Prepares a checkpoint made after grist python code has loaded in the gvisor sandbox.
* Changes how `DOC_URL` is passed to the sandbox, since it can no longer be passed in as an environment variable when using checkpoints.
* Uses the checkpoint to speed up tests using the gvisor sandbox, otherwise a lot of tests need more time (especially on mac under docker).
* Directs jenkins to run all tests with python2 and python3 when a new file `buildtools/changelogs/python.txt` is touched (this diff counts as touching that file).
* Tweaks miscellaneous tests
- some needed fixes exposed by slightly different timing
- a small number actually give different results in py3 (removal of `u` prefixes).
- some needed a little more time
The DOC_URL change is not the ultimate solution we want for DOC_URL. Eventually it should be a variable that gets updated, like the date perhaps. This is just a small pragmatic change to preserve existing behavior.
Tests are run mindlessly as py3, and for some tests it won't change anything (e.g. if they do not use NSandbox). Tests are not run in parallel, doubling overall test time.
Checkpoints could be useful in deployment, though this diff doesn't use them there.
The application of checkpoints doesn't check for other configuration like 3-versus-5-pipe that we don't actually use.
Python2 tests run using pynbox as always for now.
The diff got sufficiently bulky that I didn't tackle running py3 on "regular" diffs in it. My preference, given that most tests don't appear to stress the python side of things, would be to make a selection of the tests that do and a few wild cards, and run those tests on both pythons rather then all of them. For diffs making a significant python change, I'd propose touching buildtools/changelogs/python.txt for full tests. But this is a conversation in progress.
A total of 6886 tests ran on this diff.
Test Plan: this is a step in preparing tests for py3 transition
Reviewers: dsagal
Reviewed By: dsagal
Subscribers: dsagal
Differential Revision: https://phab.getgrist.com/D3066
2021-10-18 17:37:51 +00:00
|
|
|
log.info("Ready") # This log message is significant for checkpointing.
|
2020-07-27 18:57:36 +00:00
|
|
|
sandbox.run()
|
|
|
|
|
2021-06-30 13:35:05 +00:00
|
|
|
def main():
|
2021-07-08 21:52:00 +00:00
|
|
|
run(get_default_sandbox())
|
2021-06-30 13:35:05 +00:00
|
|
|
|
2020-07-27 18:57:36 +00:00
|
|
|
if __name__ == "__main__":
|
|
|
|
main()
|