ivoryos 1.2.3__py3-none-any.whl → 1.2.4__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of ivoryos might be problematic. Click here for more details.

ivoryos/__init__.py CHANGED
@@ -6,7 +6,7 @@ from typing import Union
6
6
  from flask import Flask, redirect, url_for, g, Blueprint, session
7
7
  from flask_login import AnonymousUserMixin
8
8
 
9
- sys.path.insert(0, os.path.abspath(os.path.join(os.path.dirname(__file__), '..')))
9
+ # sys.path.insert(0, os.path.abspath(os.path.join(os.path.dirname(__file__), '..')))
10
10
 
11
11
  from ivoryos.config import Config, get_config
12
12
  from ivoryos.routes.auth.auth import auth, login_manager
@@ -25,7 +25,7 @@ from ivoryos.utils.global_config import GlobalConfig
25
25
  from ivoryos.optimizer.registry import OPTIMIZER_REGISTRY
26
26
  from ivoryos.utils.script_runner import ScriptRunner
27
27
  from ivoryos.version import __version__ as ivoryos_version
28
- from importlib.metadata import entry_points
28
+ # from importlib.metadata import entry_points
29
29
 
30
30
  global_config = GlobalConfig()
31
31
  from sqlalchemy import event
@@ -210,22 +210,22 @@ def run(module=None, host="0.0.0.0", port=None, debug=None, llm_server=None, mod
210
210
  # return app
211
211
 
212
212
 
213
- def load_installed_plugins(app, socketio):
214
- """
215
- Dynamically load installed plugins and attach Flask-SocketIO.
216
- """
217
- plugin_names = []
218
- for entry_point in entry_points().get("ivoryos.plugins", []):
219
- plugin = entry_point.load()
220
-
221
- # If the plugin has an `init_socketio()` function, pass socketio
222
- if hasattr(plugin, 'init_socketio'):
223
- plugin.init_socketio(socketio)
224
-
225
- plugin_names.append(entry_point.name)
226
- app.register_blueprint(getattr(plugin, entry_point.name), url_prefix=f"{url_prefix}/{entry_point.name}")
227
-
228
- return plugin_names
213
+ # def load_installed_plugins(app, socketio):
214
+ # """
215
+ # Dynamically load installed plugins and attach Flask-SocketIO.
216
+ # """
217
+ # plugin_names = []
218
+ # for entry_point in entry_points().get("ivoryos.plugins", []):
219
+ # plugin = entry_point.load()
220
+ #
221
+ # # If the plugin has an `init_socketio()` function, pass socketio
222
+ # if hasattr(plugin, 'init_socketio'):
223
+ # plugin.init_socketio(socketio)
224
+ #
225
+ # plugin_names.append(entry_point.name)
226
+ # app.register_blueprint(getattr(plugin, entry_point.name), url_prefix=f"{url_prefix}/{entry_point.name}")
227
+ #
228
+ # return plugin_names
229
229
 
230
230
 
231
231
  def load_plugins(blueprints: Union[list, Blueprint], app, socketio):
@@ -173,7 +173,7 @@ def update_ui_state():
173
173
  autofill = data.get("autofill", False)
174
174
  session['autofill'] = autofill
175
175
  _, forms = _create_forms(instrument, script, autofill)
176
- rendered_html = render_template("components/methods_panel.html", forms=forms, script=script, instrument=instrument)
176
+ rendered_html = render_template("components/actions_panel.html", forms=forms, script=script, instrument=instrument)
177
177
  return jsonify({"html": rendered_html})
178
178
 
179
179
  if "deck_name" in data:
@@ -94,6 +94,7 @@ function toggleAutoFill() {
94
94
  .then(data => {
95
95
  if (data.html) {
96
96
  document.getElementById("instrument-panel").innerHTML = data.html;
97
+ initializeDragHandlers()
97
98
  }
98
99
  })
99
100
  }
ivoryos/utils/form.py CHANGED
@@ -1,5 +1,10 @@
1
1
  from enum import Enum
2
- from typing import get_origin, get_args, Union, Any
2
+ from typing import Union, Any
3
+ try:
4
+ from typing import get_origin, get_args
5
+ except ImportError:
6
+ # For Python versions = 3.7, use typing_extensions
7
+ from typing_extensions import get_origin, get_args
3
8
 
4
9
  from wtforms.fields.choices import SelectField
5
10
  from wtforms.fields.core import Field
@@ -188,44 +188,52 @@ class ScriptRunner:
188
188
  # _func_str = script.compile()
189
189
  # step_list_dict: dict = script.convert_to_lines(_func_str)
190
190
  self._emit_progress(socketio, 1)
191
-
192
- # Run "prep" section once
193
- script_dict = script.script_dict
194
- with current_app.app_context():
195
-
196
- run = WorkflowRun(name=script.name or "untitled", platform=script.deck or "deck",start_time=datetime.now())
197
- db.session.add(run)
198
- db.session.commit()
199
- run_id = run.id # Save the ID
200
- global_config.runner_status = {"id":run_id, "type": "workflow"}
201
- self._run_actions(script, section_name="prep", logger=logger, socketio=socketio, run_id=run_id)
202
- output_list = []
203
- _, arg_type = script.config("script")
204
- _, return_list = script.config_return()
205
-
206
- # Run "script" section multiple times
207
- if repeat_count:
208
- self._run_repeat_section(repeat_count, arg_type, bo_args, output_list, script,
209
- run_name, return_list, compiled, logger, socketio,
210
- history, output_path, run_id=run_id, optimizer=optimizer)
211
- elif config:
212
- self._run_config_section(config, arg_type, output_list, script, run_name, logger,
213
- socketio, run_id=run_id, compiled=compiled)
214
-
215
- # Run "cleanup" section once
216
- self._run_actions(script, section_name="cleanup", logger=logger, socketio=socketio,run_id=run_id)
217
- # Reset the running flag when done
191
+ filename = None
192
+ error_flag = False
193
+ # create a new run entry in the database
194
+ try:
195
+ with current_app.app_context():
196
+ run = WorkflowRun(name=script.name or "untitled", platform=script.deck or "deck",start_time=datetime.now())
197
+ db.session.add(run)
198
+ db.session.commit()
199
+ run_id = run.id # Save the ID
200
+ global_config.runner_status = {"id":run_id, "type": "workflow"}
201
+
202
+ # Run "prep" section once
203
+ self._run_actions(script, section_name="prep", logger=logger, socketio=socketio, run_id=run_id)
204
+ output_list = []
205
+ _, arg_type = script.config("script")
206
+ _, return_list = script.config_return()
207
+ # Run "script" section multiple times
208
+ if repeat_count:
209
+ self._run_repeat_section(repeat_count, arg_type, bo_args, output_list, script,
210
+ run_name, return_list, compiled, logger, socketio,
211
+ history, output_path, run_id=run_id, optimizer=optimizer)
212
+ elif config:
213
+ self._run_config_section(config, arg_type, output_list, script, run_name, logger,
214
+ socketio, run_id=run_id, compiled=compiled)
215
+ # Run "cleanup" section once
216
+ self._run_actions(script, section_name="cleanup", logger=logger, socketio=socketio,run_id=run_id)
217
+ # Reset the running flag when done
218
+
219
+ # Save results if necessary
220
+
221
+ if not script.python_script and output_list:
222
+ filename = self._save_results(run_name, arg_type, return_list, output_list, logger, output_path)
223
+ self._emit_progress(socketio, 100)
224
+
225
+ except Exception as e:
226
+ logger.error(f"Error during script execution: {e.__str__()}")
227
+ error_flag = True
228
+ finally:
218
229
  self.lock.release()
219
- # Save results if necessary
220
- filename = None
221
- if not script.python_script and output_list:
222
- filename = self._save_results(run_name, arg_type, return_list, output_list, logger, output_path)
223
- self._emit_progress(socketio, 100)
224
- with current_app.app_context():
225
- run = db.session.get(WorkflowRun, run_id) # SQLAlchemy 1.4+ recommended method
226
- run.end_time = datetime.now()
227
- run.data_path = filename
228
- db.session.commit()
230
+ with current_app.app_context():
231
+ run = db.session.get(WorkflowRun, run_id)
232
+ run.end_time = datetime.now()
233
+ run.output_file = filename
234
+ run.run_error = error_flag
235
+ db.session.commit()
236
+
229
237
 
230
238
  def _run_actions(self, script, section_name="", logger=None, socketio=None, run_id=None):
231
239
  _func_str = script.python_script or script.compile()
@@ -254,7 +262,7 @@ class ScriptRunner:
254
262
  logger.info(f'Stopping execution during {run_name}: {i + 1}/{len(config)}')
255
263
  break
256
264
  logger.info(f'Executing {i + 1} of {len(config)} with kwargs = {kwargs}')
257
- progress = (i + 1) * 100 / len(config)
265
+ progress = ((i + 1) * 100 / len(config)) - 0.1
258
266
  self._emit_progress(socketio, progress)
259
267
  # fname = f"{run_name}_script"
260
268
  # function = self.globals_dict[fname]
ivoryos/version.py CHANGED
@@ -1 +1 @@
1
- __version__ = "1.2.3"
1
+ __version__ = "1.2.4"
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: ivoryos
3
- Version: 1.2.3
3
+ Version: 1.2.4
4
4
  Summary: an open-source Python package enabling Self-Driving Labs (SDLs) interoperability
5
5
  Author-email: Ivory Zhang <ivoryzhang@chem.ubc.ca>
6
6
  License: MIT
@@ -25,142 +25,121 @@ Dynamic: license-file
25
25
  [![Documentation Status](https://readthedocs.org/projects/ivoryos/badge/?version=latest)](https://ivoryos.readthedocs.io/en/latest/?badge=latest)
26
26
  [![PyPI version](https://img.shields.io/pypi/v/ivoryos)](https://pypi.org/project/ivoryos/)
27
27
  ![License](https://img.shields.io/pypi/l/ivoryos)
28
- [![YouTube](https://img.shields.io/badge/YouTube-video-red?logo=youtube)](https://youtu.be/dFfJv9I2-1g)
28
+ [![YouTube](https://img.shields.io/badge/YouTube-tutorial-red?logo=youtube)](https://youtu.be/dFfJv9I2-1g)
29
+ [![YouTube](https://img.shields.io/badge/YouTube-demo-red?logo=youtube)](https://youtu.be/flr5ydiE96s)
29
30
  [![Published](https://img.shields.io/badge/Nature_Comm.-paper-blue)](https://www.nature.com/articles/s41467-025-60514-w)
30
31
  [![Discord](https://img.shields.io/discord/1313641159356059770?label=Discord&logo=discord&color=5865F2)](https://discord.gg/AX5P9EdGVX)
31
32
 
32
33
  ![](https://gitlab.com/heingroup/ivoryos/raw/main/docs/source/_static/ivoryos.png)
33
34
  # ivoryOS: interoperable Web UI for self-driving laboratories (SDLs)
34
- "plug and play" web UI extension for flexible SDLs.
35
+ A **plug-and-play** web interface for flexible SDLs
36
+
37
+ ---
35
38
 
36
39
  ## Table of Contents
37
40
  - [Description](#description)
38
41
  - [System requirements](#system-requirements)
39
42
  - [Installation](#installation)
40
- - [Instructions for use](#instructions-for-use)
43
+ - [Quick Start](#quick-start)
44
+ - [Features](#features)
41
45
  - [Demo](#demo)
42
46
  - [Roadmap](#roadmap)
47
+ - [Acknowledgements](#acknowledgements)
43
48
 
44
-
49
+ ---
45
50
  ## Description
46
- Granting SDLs flexibility and modularity makes it almost impossible to design a UI, yet it's a necessity for allowing more people to interact with it (democratisation).
47
- This web UI aims to ease up the control of any Python-based SDLs by displaying functions and parameters for initialized modules dynamically.
48
- The modules can be hardware API, high-level functions, or experiment workflow.
49
- With the least modification of the current workflow, user can design, manage and execute their experimental designs and monitor the execution process.
50
-
51
- ## System requirements
52
- This software is developed and tested using Windows. This software and its dependencies are compatible across major platforms: Linux, macOS, and Windows. Some dependencies (Flask-SQLAlchemy) may require additional setup.
53
-
54
- ### Python Version
55
- Python >=3.10 for the best compatibility. Python >=3.7 without Ax.
56
- ### Python dependencies
57
- This software is compatible with the latest versions of all dependencies.
58
- - bcrypt~=4.0
59
- - Flask-Login~=0.6
60
- - Flask-Session~=0.8
61
- - Flask-SocketIO~=5.3
62
- - Flask-SQLAlchemy~=3.1
63
- - SQLAlchemy-Utils~=0.41
64
- - Flask-WTF~=1.2
65
- - python-dotenv==1.0.1
66
- - ax-platform (optional 1.0 for Python>=3.10)
67
- - baybe (optional)
51
+ Building UIs for SDLs is challenging because flexibility and modularity make them unpredictable yet accessibility is essential for **democratisation** of AI-driven scientific discovery.
52
+
53
+ **IvoryOS** bridges the gap by:
54
+ - Dynamically inspecting initialized Python modules (hardware APIs, high-level functions, or workflows)
55
+ - Automatically displaying functions and parameters in a web UI
56
+ - Allowing users to **design**, **manage**, and **execute** experimental workflows with minimal changes to existing scripts
57
+ - Providing natural language support for workflow design and execution, check [IvoryOS MCP](https://gitlab.com/heingroup/ivoryos-suite/ivoryos-mcp) for more details.
58
+
59
+ ----
60
+ ## System Requirements
61
+
62
+ **Platforms:** Compatible with Linux, macOS, and Windows (developed/tested on Windows).
63
+ **Python:**
64
+ - Recommended: Python ≥3.10
65
+ - Minimum: Python ≥3.7 (without Ax optimizer support)
66
+
67
+ **Core Dependencies:**
68
+ <details>
69
+ <summary>Click to expand</summary>
70
+
71
+ - bcrypt~=4.0
72
+ - Flask-Login~=0.6
73
+ - Flask-Session~=0.8
74
+ - Flask-SocketIO~=5.3
75
+ - Flask-SQLAlchemy~=3.1
76
+ - SQLAlchemy-Utils~=0.41
77
+ - Flask-WTF~=1.2
78
+ - python-dotenv==1.0.1
79
+
80
+ **Optional:**
81
+ - ax-platform (≥1.0, Python≥3.10)
82
+ - baybe
83
+ </details>
84
+
85
+ ---
68
86
 
69
87
 
70
88
  ## Installation
89
+ From PyPI:
71
90
  ```bash
72
91
  pip install ivoryos
73
92
  ```
74
- or
93
+ From source:
75
94
  ```bash
76
95
  git clone https://gitlab.com/heingroup/ivoryos.git
77
96
  cd ivoryos
78
- pip install .
97
+ pip install -e .
79
98
  ```
80
99
 
81
- The installation may take 10 to 30 seconds to install. The installation time may vary and take up to several minutes, depending on the network speed, computer performance, and virtual environment settings.
82
100
 
83
- ## Instructions for use
84
- ### Quick start
85
- In your SDL script, use `ivoryos(__name__)`.
101
+ ## Quick start
102
+ In your SDL script,
86
103
  ```python
87
104
  import ivoryos
88
105
 
89
106
  ivoryos.run(__name__)
90
107
  ```
91
- ### Login
92
- Create an account and login (local database with bcrypt password)
93
- ### Features
94
- - **Direct control**: direct function calling _Devices_ tab
95
- - **Workflows**:
108
+ Login: Create an account (local DB, bcrypt password)
109
+
110
+ ----
111
+ ## Features
112
+ ### Direct control:
113
+ direct function calling _Devices_ tab
114
+ ### Workflows
96
115
  - **Design Editor**: drag/add function to canvas in _Design_ tab. click `Compile and Run` button to go to the execution configuration page
97
116
  - **Execution Config**: configure iteration methods and parameters in _Compile/Run_ tab.
98
117
  - **Design Library**: manage workflow scripts in _Library_ tab.
99
118
  - **Workflow Data**: Execution records are in _Data_ tab.
119
+ ### Offline mode
120
+ after one successful connection, a blueprint will be automatically saved and made accessible without hardware connection. In a new Python script in the same directory, use `ivoryos.run()` to start offline mode.
100
121
 
101
- [//]: # (![Discord]&#40;https://img.shields.io/discord/1313641159356059770&#41;)
102
-
103
- [//]: # (![PyPI - Downloads]&#40;https://img.shields.io/pypi/dm/ivoryos&#41;)
104
-
105
-
106
- ### Additional settings
107
- [//]: # (#### AI assistant)
108
-
109
- [//]: # (To streamline the experimental design on SDLs, we also integrate Large Language Models &#40;LLMs&#41; to interpret the inspected functions and generate code according to task descriptions.)
110
-
111
- [//]: # ()
112
- [//]: # (#### Enable LLMs with [OpenAI API]&#40;https://github.com/openai/openai-python&#41;)
113
-
114
- [//]: # (1. Create a `.env` file for `OPENAI_API_KEY`)
115
-
116
- [//]: # (```)
117
-
118
- [//]: # (OPENAI_API_KEY="Your API Key")
119
-
120
- [//]: # (```)
121
-
122
- [//]: # (2. In your SDL script, define model, you can use any GPT models.)
123
-
124
- [//]: # ()
125
- [//]: # (```python)
126
-
127
- [//]: # (ivoryos.run&#40;__name__, model="gpt-3.5-turbo"&#41;)
128
122
 
129
- [//]: # (```)
130
123
 
131
- [//]: # ()
132
- [//]: # (#### Enable local LLMs with [Ollama]&#40;https://ollama.com/&#41;)
133
-
134
- [//]: # (1. Download Ollama.)
135
-
136
- [//]: # (2. pull models from Ollama)
137
-
138
- [//]: # (3. In your SDL script, define LLM server and model, you can use any models available on Ollama.)
139
-
140
- [//]: # ()
141
- [//]: # (```python)
142
-
143
- [//]: # (ivoryos.run&#40;__name__, llm_server="localhost", model="llama3.1"&#41;)
144
-
145
- [//]: # (```)
146
-
147
- #### Add additional logger(s)
124
+ ### Logging
125
+ Add single or multiple loggers:
148
126
  ```python
149
127
  ivoryos.run(__name__, logger="logger name")
150
- ```
151
- or
152
- ```python
153
128
  ivoryos.run(__name__, logger=["logger 1", "logger 2"])
154
129
  ```
155
- #### Offline (design without hardware connection)
156
- After one successful connection, a blueprint will be automatically saved and made accessible without hardware connection. In a new Python script in the same directory, use `ivoryos.run()` to start offline mode.
157
-
158
- ```python
159
- ivoryos.run()
160
- ```
130
+ ### Directory Structure
131
+
132
+ Created automatically on first run:
133
+ - **`ivoryos_data/`**:
134
+ - **`ivoryos_data/config_csv/`**: Batch configuration `csv`
135
+ - **`ivoryos_data/pseudo_deck/`**: Offline deck `.pkl`
136
+ - **`ivoryos_data/results/`**: Execution results
137
+ - **`ivoryos_data/scripts/`**: Compiled workflows Python scripts
138
+ - **`default.log`**: Application logs
139
+ - **`ivoryos.db`**: Local database
140
+ ---
161
141
  ## Demo
162
- In the [abstract_sdl.py](https://gitlab.com/heingroup/ivoryos/-/blob/main/example/abstract_sdl_example/abstract_sdl.py), where instances of `AbstractSDL` is created as `sdl`,
163
- addresses will be available on terminal.
142
+ In the [abstract_sdl.py](https://gitlab.com/heingroup/ivoryos/-/blob/main/example/abstract_sdl_example/abstract_sdl.py)
164
143
  ```Python
165
144
  ivoryos.run(__name__)
166
145
  ```
@@ -169,35 +148,19 @@ ivoryos.run(__name__)
169
148
  * Running on http://127.0.0.1:8000
170
149
  * Running on http://0.0.0.0:8000
171
150
 
172
- ### Deck function and web form
173
- ![](https://gitlab.com/heingroup/ivoryos/raw/main/docs/source/_static/demo.gif)
174
-
175
-
176
- ### Directory structure
177
-
178
- When you run the application for the first time, it will automatically create the following folders and files in the same directory:
179
-
180
- - **`ivoryos_data/`**: Main directory for application-related data.
181
- - **`ivoryos_data/config_csv/`**: Contains iteration configuration files in CSV format.
182
- - **`ivoryos_data/llm_output/`**: Stores raw prompt generated for the large language model.
183
- - **`ivoryos_data/pseudo_deck/`**: Contains pseudo-deck `.pkl` files for offline access.
184
- - **`ivoryos_data/results/`**: Used for storing results or outputs during workflow execution.
185
- - **`ivoryos_data/scripts/`**: Holds Python scripts compiled from the visual programming script design.
186
-
187
- - **`default.log`**: Log file that captures application logs.
188
- - **`ivoryos.db`**: Database file that stores application data locally.
189
-
151
+ ---
190
152
 
191
153
  ## Roadmap
192
154
 
193
155
  - [x] Allow plugin pages ✅
194
156
  - [x] pause, resume, abort current and pending workflows ✅
195
- - [x] dropdown input
196
- - [x] show line number option ✅
157
+ - [ ] dropdown input
197
158
  - [ ] snapshot version control
198
159
  - [ ] optimizer-agnostic
199
160
  - [ ] check batch-config file compatibility
200
161
 
162
+ ---
163
+
201
164
  ## Citing
202
165
 
203
166
  If you find this project useful, please consider citing the following manuscript:
@@ -233,8 +196,6 @@ For an additional perspective related to the development of the tool, please see
233
196
  url = {https://communities.springernature.com/posts/behind-ivoryos-empowering-scientists-to-harness-self-driving-labs-for-accelerated-discovery}
234
197
  }
235
198
  ```
236
-
237
- ## Authors and Acknowledgement
238
- Ivory Zhang, Lucy Hao
239
-
240
- Authors acknowledge Telescope Innovations, Hein Lab members for their valuable suggestions and contributions.
199
+ ---
200
+ ## Acknowledgements
201
+ Authors acknowledge Telescope Innovations Corp., Hein Lab members for their valuable suggestions and contributions.
@@ -1,7 +1,7 @@
1
- ivoryos/__init__.py,sha256=x4PnDTbhx1ZTugfuclGQ03GO3yidAfiUFU7vuGyKD8M,9830
1
+ ivoryos/__init__.py,sha256=BAA7OPl3h_QdSc4na-7OWTKhaOq1LtHOrJHX3gGITrc,9863
2
2
  ivoryos/config.py,sha256=y3RxNjiIola9tK7jg-mHM8EzLMwiLwOzoisXkDvj0gA,2174
3
3
  ivoryos/socket_handlers.py,sha256=VWVWiIdm4jYAutwGu6R0t1nK5MuMyOCL0xAnFn06jWQ,1302
4
- ivoryos/version.py,sha256=C-D_WWrVkBDmQmApLcm0sWNh2CgIrwWfc8_sB5vvU-Q,22
4
+ ivoryos/version.py,sha256=XBKH8E1LmDxv06U39yqMBbXZapOERFgICEDYZs_kRso,22
5
5
  ivoryos/optimizer/ax_optimizer.py,sha256=PoSu8hrDFFpqyhRBnaSMswIUsDfEX6sPWt8NEZ_sobs,7112
6
6
  ivoryos/optimizer/base_optimizer.py,sha256=JTbUharZKn0t8_BDbAFuwZIbT1VOnX1Xuog1pJuU8hY,1992
7
7
  ivoryos/optimizer/baybe_optimizer.py,sha256=EdrrRiYO-IOx610cPXiQhH4qG8knUP0uiZ0YoyaGIU8,7954
@@ -25,7 +25,7 @@ ivoryos/routes/data/templates/workflow_database.html,sha256=ofvHcovpwmJXo1SFiSrL
25
25
  ivoryos/routes/data/templates/workflow_view.html,sha256=72xKreX9WhYx-0n0cFf-CL-fJIWXPCIaTi_Aa8Tq3xg,3651
26
26
  ivoryos/routes/data/templates/components/step_card.html,sha256=9lKR4NCgU2v5Nbdv2uaJ-9aKibtiB_2-Y_kyHX6Ka1k,730
27
27
  ivoryos/routes/design/__init__.py,sha256=zS3HXKaw0ALL5n6t_W1rUz5Uj5_tTQ-Y1VMXyzewvR0,113
28
- ivoryos/routes/design/design.py,sha256=F5X0Wa1sXSdolQ8w87wQBv6bN7jMwCnQ4jX8r4SkO24,17763
28
+ ivoryos/routes/design/design.py,sha256=BxCSyb9NHlm6SLi7iZITS-g4IyPyBK0ZVAeadPGG-Cw,17763
29
29
  ivoryos/routes/design/design_file.py,sha256=m4yku8fkpLUs4XvLJBqR5V-kyaGKbGB6ZoRxGbjEU5Q,2140
30
30
  ivoryos/routes/design/design_step.py,sha256=l8U3-FuXmap__sYm51AueKdbTaLCFaKjAz-j02b4g-E,5200
31
31
  ivoryos/routes/design/templates/experiment_builder.html,sha256=hh-d2tOc_40gww5WfUYIf8sM3qBaALZnR8Sx7Ja4tpU,1623
@@ -79,22 +79,22 @@ ivoryos/static/js/script_metadata.js,sha256=m8VYZ8OGT2oTx1kXMXq60bKQI9WCbJNkzcFD
79
79
  ivoryos/static/js/socket_handler.js,sha256=2Iyv_3METjhSlSavs_L9FE3PKY4xDEpfzJpd2FywY9o,5300
80
80
  ivoryos/static/js/sortable_card.js,sha256=ifmlGe3yy0U_KzMphV4ClRhK2DLOvkELYMlq1vECuac,807
81
81
  ivoryos/static/js/sortable_design.js,sha256=QqYyk385JNm6zCgZK_Oa-cJEP1uPtZ_tVz27x4hyx5A,4790
82
- ivoryos/static/js/ui_state.js,sha256=5EuqwIDndjZTqj9dsAR_IA91iGMcodY_TuQo17LLL1w,3331
82
+ ivoryos/static/js/ui_state.js,sha256=XYsOcfGlduqLlqHySvPrRrR50CiAsml51duqneigsRY,3368
83
83
  ivoryos/templates/base.html,sha256=cl5w6E8yskbUzdiJFal6fZjnPuFNKEzc7BrrbRd6bMI,8581
84
84
  ivoryos/utils/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
85
85
  ivoryos/utils/bo_campaign.py,sha256=Fil-zT7JexL_p9XqyWByjAk42XB1R9XUKN8CdV5bi6c,9714
86
86
  ivoryos/utils/client_proxy.py,sha256=0OT2xTMkqh_2ybgCxMV_71ZVUThWwrsnAhTIBY5vDR8,2095
87
87
  ivoryos/utils/db_models.py,sha256=baE4LJcSGUj10Tj6imfINXi4JQX_4oLv_kb9bd0rp-M,27920
88
- ivoryos/utils/form.py,sha256=eIk1N77Ynxc4Omww5ZYlmpOIJfQPWto2qfiU6nzIIeQ,21755
88
+ ivoryos/utils/form.py,sha256=8GSEhfY3cE-q9QxGSW_u-9dFNO72N6QOSGwfYyPTfM0,21912
89
89
  ivoryos/utils/global_config.py,sha256=zNO9GYhGn7El3msWoxJIm3S4Mzb3VMh2i5ZEsVtvb2Q,2463
90
90
  ivoryos/utils/llm_agent.py,sha256=-lVCkjPlpLues9sNTmaT7bT4sdhWvV2DiojNwzB2Lcw,6422
91
91
  ivoryos/utils/py_to_json.py,sha256=fyqjaxDHPh-sahgT6IHSn34ktwf6y51_x1qvhbNlH-U,7314
92
- ivoryos/utils/script_runner.py,sha256=g3_pLYcu6gF9sPjhW9WRlwMH7ScDpz_MqMJzxNayfyg,16725
92
+ ivoryos/utils/script_runner.py,sha256=MdLMSAeaVXxnbcQfzHJximeJ6W7uOCxqTQhFSpsEieg,17065
93
93
  ivoryos/utils/serilize.py,sha256=lkBhkz8r2bLmz2_xOb0c4ptSSOqjIu6krj5YYK4Nvj8,6784
94
94
  ivoryos/utils/task_runner.py,sha256=cDIcmDaqYh0vXoYaL_kO877pluAo2tyfsHl9OgZqJJE,3029
95
95
  ivoryos/utils/utils.py,sha256=-WiU0_brszB9yDsiQepf_7SzNgPTSpul2RSKDOY3pqo,13921
96
- ivoryos-1.2.3.dist-info/licenses/LICENSE,sha256=p2c8S8i-8YqMpZCJnadLz1-ofxnRMILzz6NCMIypRag,1084
97
- ivoryos-1.2.3.dist-info/METADATA,sha256=HZPZvy7QwKlAtnKsh41WPjbl7UK7hVbOkteIWuNmZn8,9416
98
- ivoryos-1.2.3.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91
99
- ivoryos-1.2.3.dist-info/top_level.txt,sha256=FRIWWdiEvRKqw-XfF_UK3XV0CrnNb6EmVbEgjaVazRM,8
100
- ivoryos-1.2.3.dist-info/RECORD,,
96
+ ivoryos-1.2.4.dist-info/licenses/LICENSE,sha256=p2c8S8i-8YqMpZCJnadLz1-ofxnRMILzz6NCMIypRag,1084
97
+ ivoryos-1.2.4.dist-info/METADATA,sha256=V4lFZwkePsSRrYNShj0i2KM6u8_Z5F2omuHbYeAW5FU,7351
98
+ ivoryos-1.2.4.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91
99
+ ivoryos-1.2.4.dist-info/top_level.txt,sha256=FRIWWdiEvRKqw-XfF_UK3XV0CrnNb6EmVbEgjaVazRM,8
100
+ ivoryos-1.2.4.dist-info/RECORD,,