# Overview
This is a Python application that allows you to create/maintain/manage study configurations away from your implementations. `experiment-server` has several different interfaces (see below) to allow using it in a range of different scenarios. I've used it with Python, js and [Unity projects](https://github.com/ahmed-shariff/experiment_server/wiki/Using-with-Unity). See the [wiki](https://github.com/ahmed-shariff/experiment_server/wiki) for examples.
Documentation is available at [https://shariff-faleel.com/experiment_server/](https://shariff-faleel.com/experiment_server/)
# Content
- [Overview](#overview)
- [Installation](#installation)
- [Usage](#usage)
- [Configuration of an experiment](#configuration-of-an-experiment)
- [Verify config](#verify-config)
- [Loading experiment through server](#loading-experiment-through-server)
- [Loading experiment through API](#loading-experiment-through-api)
- [Generate expanded config files](#generate-expanded-config-files)
- [Function calls in config](#function-calls-in-config)
- [Supported functions](#supported-functions)
- [Example function calls](#example-function-calls)
# Installation
Install it directly into an activated virtual environment:
```text
$ pip install experiment-server
```
or add it to your [Poetry](https://poetry.eustace.io/) project:
```text
$ poetry add experiment-server
```
# Usage
## Configuration of an experiment
The configuration is defined in a [toml](https://toml.io/en/) file.
A config file can be generated as follows
```sh
$ experiment-server new-config-file new_config.toml
```
See example `.toml` below for how the configuration can be defined.
```toml
# The `configuration` table contains the settings of the study/experiment itself
[configuration]
# The `order` is an array of block names or an array of array of block names.
order = [["conditionA", "conditionB", "conditionA", "conditionB"]]
# The `groups` and `within_groups` are optional keys that allows you to define how the
# conditions specified in `order` will be managed. `groups` would dictate how the top
# level array of `order` will be handled. `within_groups` would dictate how the conditions
# in the nested arrays (if specified) would be managed. These keys can have one
# of the following values.
# - "latin_square": Apply latin square to balance the values.
# - "randomize": For each participant randomize the order of the values in the array.
# - "as_is": Use the order of the values as specified.
# When not specified, the default value is "as_is" for both keys.
groups = "latin_square"
within_groups= "randomize"
# The random seed to use for any randomization. Default seed is 0. The seed will be
# the value of random_seed + participant_index
random_seed = 0
# The subtable `variabels` are values that can be used anywhere when defining the blocks.
# Any variable can be used by appending "$" before the variable name in the blocks. See
# below for an exmaple of how variables can be used
[configuration.variables]
TRIALS_PER_ITEM = 3
# Blocks are defined as an array of tables. Each block must contain `name` and the
# subtable `config`. Optionally, a block can also specify `extends`, whish is a `name` of
# another block. See below for more explanation on how `extends` works
# Block: Condition A
[[blocks]]
name = "conditionA"
# The `config` subtable can have any key-values. Note that `name` and `participant_index`
# will be added to the `config` when this file is being processed. Hence, those keys
# will be overwritten if used in this subtable.
[blocks.config]
trialsPerItem = "$TRIALS_PER_ITEM"
param1 = 1
# The value can also be a function call. A function call is represented as a table
# The following function call will be replaced with a call to
# [random.choices](https://docs.python.org/3/library/random.html#random.choices)
# See `# Function calls` in README for more information.
param2 = { function_name = "choices", args = { population = [1 , 2 , 3 ], k = 2}}
param3 = { function_name = "choices", args = [[1 , 2 , 3 ]], params = { unique = true } }
# Block: Condition B
[[blocks]]
name = "conditionB"
extends = "conditionA"
# Since "conditionB" is extending "conditionA", the keys in the `config` subtable of
# the block "conditionA" not defined in the `config` subtable of "conditionB" will be copied
# to the `config` subtable of "conditionB". In this example, `param1`, `param2` and
# `trialsPerItem` will be copied over here.
[blocks.config]
param3 = [2]
```
See [toml spec](https://toml.io/en/v1.0.0) for more information on the format of a toml file.
The above config file, after being processed, would result in the following list of blocks for participant number 1:
```json
[
{
"name": "conditionB",
"extends": "conditionA",
"config": {
"param3": [
2
],
"trialsPerItem": 3,
"param1": 1,
"param2": [
1,
2
],
"participant_index": 1,
"name": "conditionB",
"block_id": 0
}
},
{
"name": "conditionA",
"config": {
"trialsPerItem": 3,
"param1": 1,
"param2": [
2,
2
],
"param3": [
3
],
"participant_index": 1,
"name": "conditionA",
"block_id": 1
}
},
{
"name": "conditionA",
"config": {
"trialsPerItem": 3,
"param1": 1,
"param2": [
1,
1
],
"param3": [
2
],
"participant_index": 1,
"name": "conditionA",
"block_id": 2
}
},
{
"name": "conditionB",
"extends": "conditionA",
"config": {
"param3": [
2
],
"trialsPerItem": 3,
"param1": 1,
"param2": [
3,
1
],
"participant_index": 1,
"name": "conditionB",
"block_id": 3
}
}
]
```
## Verify config
A config file can be validated by running:
```sh
$ experiment-server verify-config-file sample_config.toml
```
This will show how the expanded config looks like for the first 5 participants.
## Loading experiment through server
After installation, the server can used as:
```sh
$ experiment-server run sample_config.toml
```
See more options with `--help`
The server exposes the following REST API:
- [GET] `/api/blocks-count` / `/api/blocks-count/:participant-id` - Return the number of blocks in the configuration loaded. For a given config, the `blocks-count` will be the same for all participants.
- [GET] `/api/block-id` / `/api/block-id/:participant-id` - Returns the current block-id. If `participant-id` is provided, the blcok-id of the participant will be returned, if not the default participant's block-id will be returned. Note that the block-id is 0 indexed. i.e., the first block's block-id is 0.
- [GET] `/api/active` / `/api/active/:participant-id` - Returns the status for `participant-id`, if `participant-id` is not provided, will return the status of the default participant. Will be `false` if the participant was just initialized or the participant has gone through all blocks. To initialize the participant's status (or move to a given block), use the `move-to-next` or `move-to-block` endpoints.
- [GET] `/api/config` / `api/config/:participant-id` - Return the config for `participant-id`, if `participant-id` is not provided, will return the config for the default participant.
- [GET] `/api/summary-data` / `/api/summary-data/:participant-id` - Returns the summary of the configs for `participant-id`, if `participant-id` is not provided, returns the summary of the configs for the default participant. Currently, the summary is a JSON with the following keys
- "participant_index"
- "config_length"
- [GET] `/api/all-configs` / `/api/all-configs/:participant-id` - Returns all the configs as a list for the `participant-id`, if `participant-id` is not provided, returns the configs for the default participant.This is akin having all the results from calling the `config` endpoint for each block in one list.
- [GET] `/api/status-string` / `/api/status-string/:participant-id` - Returns status string for `participant-id`, if `participant-id` is not provided, returns statu string the default participant.
- [POST] `/api/move-to-next` / `/api/move-to-next/:participant-id` - Move `participant-id` to the next block, if `participant-id` is not provided, move the default participant to the next block. If the participant was not initialized (`active` is false), will make be marked as active (`active` will be set to true). If the block the participant was in was the last block, they will be marked as not active (`active` will be set to false).
- [POST] `/api/move-to-block/:block-id` / `/api/move-to-block/:participant-id/:block-id` - Move `participant-id` to the block number indicated by `block-id`, if `participant-id` is not provided, move the default participant to the block number indicated by `block-id`. If the participant was not initialized (`active` is false), will make be marked as active (`active` will be set to true). Will fail if the `block-id` is below 0 or above the length of the config.
- [POST] `/api/move-all-to-block/:block-id` - Move all active participants (`active` returns true) to the block number indicated by `block-id`.
- [POST] `/api/shutdown` - Shuts-down the server.
- [PUT] `/api/new-participant` - Adds a new participant and returns the new participant-id. The new participant-id will be the largest current participant-id +1.
- [PUT] `/api/add-participant/:participant-id` - Add a new participant with `participant-id`. If there is already a participant with the `participant-id`, this will fail.
For a Python application, `experiment_server.Client` can be used to access configs from the server. Also, the server can be launched programmatically using `experiment_server.server_process` which returns a [`Process`](https://docs.python.org/3/library/multiprocessing.html#multiprocessing.Process) object.
**NOTE**: If the config file served is changed, the new config will be loaded, but the state of the participants will be maintained. i.e., the added participants and the block id they are at will not change. To move the block ids for all active participants, you would have to call the `move-all-to-block` endpoint.
The server also provides a simple web interface, which can be accessed at `/` or `/index`. This interface allows to manage and monitor the flow of the experiment:
![web UI screenshot](https://raw.githubusercontent.com/ahmed-shariff/experiment_server/master/media/screenshot.png)
## Loading experiment through API
A configuration can be loaded and managed by importing `experiment_server.Experiment`.
## Generate expanded configs
A config file (i.e. `.toml` file), can be expanded to JSON with the following command
```sh
$ experiment-server generate-config-json sample_config.toml --participant-range 5
```
The above will generate the expanded configs for participant indices 1 to 5 as JSON output on stdout. This result can be written out to individual JSON files by setting the `--out-dir`/`-d` to a directory. See more options with `--help`
## Function calls in config
A function call in the config is represented by a table, with the following keys
- `function_name`: This should be one of the names in the supported functions list below.
- `args`: The arguments to be passed to the function represented by `function_name`. This can be a list or a table/dict. They should unpack with `*` or `**` respectively when called with the corresponding function.
- (optional) `params`: function-specific configurations to apply with the function calls.
- (optional) `id`: A unique identifier to group function calls.
A table that has keys other than the above keys would not be treated as a function call. Any function calls in different places of the config with the same `id` would be treated as a single group. Tables without an `id` are grouped based on their key-value pairs. Groups are used to identify how some parameters affect the results (e.g., `unique` for `choices`). Function calls can also be in `configurations.variabels`. Note that all function calls are made after the `extends` are resolved and variables from `configurations.variabels` are replaced.
### Supported functions
- `choices`: Calls [random.choices](https://docs.python.org/3/library/random.html#random.choices). `params` can be a table/dictionary which can have the key `unique`. The value of `unique` must be `true` or `false`. By default `unique` is `false`. If it's `true`, within a group of function calls, no value from the population passed to `random.choices` is repeated for a given participant.
### Example function calls
```toml
param = { function_name = "choices", args = [[1 , 2 , 3 , 4]], params = { unique = true } }
```
```toml
param = { foo = "test", bar = { function_name = "choices", args = { population = ["w", "x", "y", "z"], k = 1 } } }
```
For more on the `experiemnt-server` and how it can be used see the [wiki](https://github.com/ahmed-shariff/experiment_server/wiki)
# Wishlist (todo list?)
- Improved docs
- Add the option of using dict values in order
Raw data
{
"_id": null,
"home_page": "https://shariff-faleel.com/experiment_server/",
"name": "experiment_server",
"maintainer": null,
"docs_url": null,
"requires_python": ">=3.8",
"maintainer_email": null,
"keywords": "experiment, study-design",
"author": "Ahmed Shariff",
"author_email": "shariff.mfa@outlook.com",
"download_url": "https://files.pythonhosted.org/packages/fd/80/579dfb10d75c5cddd7ce1a72a1a6803fd225ae6f6fee52c9757f8ac70d32/experiment_server-0.3.3.tar.gz",
"platform": null,
"description": "# Overview\n\nThis is a Python application that allows you to create/maintain/manage study configurations away from your implementations. `experiment-server` has several different interfaces (see below) to allow using it in a range of different scenarios. I've used it with Python, js and [Unity projects](https://github.com/ahmed-shariff/experiment_server/wiki/Using-with-Unity). See the [wiki](https://github.com/ahmed-shariff/experiment_server/wiki) for examples.\n\nDocumentation is available at [https://shariff-faleel.com/experiment_server/](https://shariff-faleel.com/experiment_server/)\n\n# Content\n\n- [Overview](#overview)\n- [Installation](#installation)\n- [Usage](#usage)\n - [Configuration of an experiment](#configuration-of-an-experiment)\n - [Verify config](#verify-config)\n - [Loading experiment through server](#loading-experiment-through-server)\n - [Loading experiment through API](#loading-experiment-through-api)\n - [Generate expanded config files](#generate-expanded-config-files)\n - [Function calls in config](#function-calls-in-config)\n - [Supported functions](#supported-functions)\n - [Example function calls](#example-function-calls)\n\n# Installation\n\nInstall it directly into an activated virtual environment:\n\n```text\n$ pip install experiment-server\n```\n\nor add it to your [Poetry](https://poetry.eustace.io/) project:\n\n```text\n$ poetry add experiment-server\n```\n\n# Usage\n## Configuration of an experiment\nThe configuration is defined in a [toml](https://toml.io/en/) file. \n\nA config file can be generated as follows\n```sh\n$ experiment-server new-config-file new_config.toml\n```\n\nSee example `.toml` below for how the configuration can be defined.\n\n```toml\n# The `configuration` table contains the settings of the study/experiment itself\n[configuration]\n# The `order` is an array of block names or an array of array of block names.\norder = [[\"conditionA\", \"conditionB\", \"conditionA\", \"conditionB\"]]\n# The `groups` and `within_groups` are optional keys that allows you to define how the\n# conditions specified in `order` will be managed. `groups` would dictate how the top \n# level array of `order` will be handled. `within_groups` would dictate how the conditions\n# in the nested arrays (if specified) would be managed. These keys can have one \n# of the following values.\n# - \"latin_square\": Apply latin square to balance the values.\n# - \"randomize\": For each participant randomize the order of the values in the array.\n# - \"as_is\": Use the order of the values as specified.\n# When not specified, the default value is \"as_is\" for both keys.\ngroups = \"latin_square\"\nwithin_groups= \"randomize\"\n# The random seed to use for any randomization. Default seed is 0. The seed will be\n# the value of random_seed + participant_index\nrandom_seed = 0\n\n# The subtable `variabels` are values that can be used anywhere when defining the blocks.\n# Any variable can be used by appending \"$\" before the variable name in the blocks. See \n# below for an exmaple of how variables can be used\n[configuration.variables]\nTRIALS_PER_ITEM = 3\n\n# Blocks are defined as an array of tables. Each block must contain `name` and the \n# subtable `config`. Optionally, a block can also specify `extends`, whish is a `name` of\n# another block. See below for more explanation on how `extends` works\n\n# Block: Condition A\n[[blocks]]\nname = \"conditionA\"\n\n# The `config` subtable can have any key-values. Note that `name` and `participant_index`\n# will be added to the `config` when this file is being processed. Hence, those keys \n# will be overwritten if used in this subtable.\n[blocks.config]\ntrialsPerItem = \"$TRIALS_PER_ITEM\"\nparam1 = 1\n# The value can also be a function call. A function call is represented as a table\n# The following function call will be replaced with a call to \n# [random.choices](https://docs.python.org/3/library/random.html#random.choices)\n# See `# Function calls` in README for more information.\nparam2 = { function_name = \"choices\", args = { population = [1 , 2 , 3 ], k = 2}}\nparam3 = { function_name = \"choices\", args = [[1 , 2 , 3 ]], params = { unique = true } }\n\n# Block: Condition B\n[[blocks]]\nname = \"conditionB\"\nextends = \"conditionA\"\n\n# Since \"conditionB\" is extending \"conditionA\", the keys in the `config` subtable of \n# the block \"conditionA\" not defined in the `config` subtable of \"conditionB\" will be copied\n# to the `config` subtable of \"conditionB\". In this example, `param1`, `param2` and \n# `trialsPerItem` will be copied over here.\n[blocks.config]\nparam3 = [2]\n```\n\nSee [toml spec](https://toml.io/en/v1.0.0) for more information on the format of a toml file.\n\nThe above config file, after being processed, would result in the following list of blocks for participant number 1:\n```json\n[\n {\n \"name\": \"conditionB\",\n \"extends\": \"conditionA\",\n \"config\": {\n \"param3\": [\n 2\n ],\n \"trialsPerItem\": 3,\n \"param1\": 1,\n \"param2\": [\n 1,\n 2\n ],\n \"participant_index\": 1,\n \"name\": \"conditionB\",\n \"block_id\": 0\n }\n },\n {\n \"name\": \"conditionA\",\n \"config\": {\n \"trialsPerItem\": 3,\n \"param1\": 1,\n \"param2\": [\n 2,\n 2\n ],\n \"param3\": [\n 3\n ],\n \"participant_index\": 1,\n \"name\": \"conditionA\",\n \"block_id\": 1\n }\n },\n {\n \"name\": \"conditionA\",\n \"config\": {\n \"trialsPerItem\": 3,\n \"param1\": 1,\n \"param2\": [\n 1,\n 1\n ],\n \"param3\": [\n 2\n ],\n \"participant_index\": 1,\n \"name\": \"conditionA\",\n \"block_id\": 2\n }\n },\n {\n \"name\": \"conditionB\",\n \"extends\": \"conditionA\",\n \"config\": {\n \"param3\": [\n 2\n ],\n \"trialsPerItem\": 3,\n \"param1\": 1,\n \"param2\": [\n 3,\n 1\n ],\n \"participant_index\": 1,\n \"name\": \"conditionB\",\n \"block_id\": 3\n }\n }\n]\n```\n\n## Verify config\nA config file can be validated by running:\n```sh\n$ experiment-server verify-config-file sample_config.toml\n```\nThis will show how the expanded config looks like for the first 5 participants.\n\n## Loading experiment through server\nAfter installation, the server can used as:\n\n```sh\n$ experiment-server run sample_config.toml\n```\n\nSee more options with `--help`\n\nThe server exposes the following REST API:\n\n- [GET] `/api/blocks-count` / `/api/blocks-count/:participant-id` - Return the number of blocks in the configuration loaded. For a given config, the `blocks-count` will be the same for all participants. \n\n- [GET] `/api/block-id` / `/api/block-id/:participant-id` - Returns the current block-id. If `participant-id` is provided, the blcok-id of the participant will be returned, if not the default participant's block-id will be returned. Note that the block-id is 0 indexed. i.e., the first block's block-id is 0. \n\n- [GET] `/api/active` / `/api/active/:participant-id` - Returns the status for `participant-id`, if `participant-id` is not provided, will return the status of the default participant. Will be `false` if the participant was just initialized or the participant has gone through all blocks. To initialize the participant's status (or move to a given block), use the `move-to-next` or `move-to-block` endpoints.\n\n- [GET] `/api/config` / `api/config/:participant-id` - Return the config for `participant-id`, if `participant-id` is not provided, will return the config for the default participant.\n\n- [GET] `/api/summary-data` / `/api/summary-data/:participant-id` - Returns the summary of the configs for `participant-id`, if `participant-id` is not provided, returns the summary of the configs for the default participant. Currently, the summary is a JSON with the following keys \n\n - \"participant_index\"\n\n - \"config_length\"\n\n- [GET] `/api/all-configs` / `/api/all-configs/:participant-id` - Returns all the configs as a list for the `participant-id`, if `participant-id` is not provided, returns the configs for the default participant.This is akin having all the results from calling the `config` endpoint for each block in one list.\n\n- [GET] `/api/status-string` / `/api/status-string/:participant-id` - Returns status string for `participant-id`, if `participant-id` is not provided, returns statu string the default participant.\n\n- [POST] `/api/move-to-next` / `/api/move-to-next/:participant-id` - Move `participant-id` to the next block, if `participant-id` is not provided, move the default participant to the next block. If the participant was not initialized (`active` is false), will make be marked as active (`active` will be set to true). If the block the participant was in was the last block, they will be marked as not active (`active` will be set to false).\n\n- [POST] `/api/move-to-block/:block-id` / `/api/move-to-block/:participant-id/:block-id` - Move `participant-id` to the block number indicated by `block-id`, if `participant-id` is not provided, move the default participant to the block number indicated by `block-id`. If the participant was not initialized (`active` is false), will make be marked as active (`active` will be set to true). Will fail if the `block-id` is below 0 or above the length of the config.\n\n- [POST] `/api/move-all-to-block/:block-id` - Move all active participants (`active` returns true) to the block number indicated by `block-id`.\n\n- [POST] `/api/shutdown` - Shuts-down the server.\n\n- [PUT] `/api/new-participant` - Adds a new participant and returns the new participant-id. The new participant-id will be the largest current participant-id +1.\n\n- [PUT] `/api/add-participant/:participant-id` - Add a new participant with `participant-id`. If there is already a participant with the `participant-id`, this will fail. \n\nFor a Python application, `experiment_server.Client` can be used to access configs from the server. Also, the server can be launched programmatically using `experiment_server.server_process` which returns a [`Process`](https://docs.python.org/3/library/multiprocessing.html#multiprocessing.Process) object.\n\n**NOTE**: If the config file served is changed, the new config will be loaded, but the state of the participants will be maintained. i.e., the added participants and the block id they are at will not change. To move the block ids for all active participants, you would have to call the `move-all-to-block` endpoint.\n\nThe server also provides a simple web interface, which can be accessed at `/` or `/index`. This interface allows to manage and monitor the flow of the experiment:\n\n![web UI screenshot](https://raw.githubusercontent.com/ahmed-shariff/experiment_server/master/media/screenshot.png)\n\n\n## Loading experiment through API\nA configuration can be loaded and managed by importing `experiment_server.Experiment`.\n\n## Generate expanded configs\nA config file (i.e. `.toml` file), can be expanded to JSON with the following command\n\n```sh\n$ experiment-server generate-config-json sample_config.toml --participant-range 5\n```\n\nThe above will generate the expanded configs for participant indices 1 to 5 as JSON output on stdout. This result can be written out to individual JSON files by setting the `--out-dir`/`-d` to a directory. See more options with `--help`\n\n## Function calls in config\nA function call in the config is represented by a table, with the following keys \n- `function_name`: This should be one of the names in the supported functions list below.\n- `args`: The arguments to be passed to the function represented by `function_name`. This can be a list or a table/dict. They should unpack with `*` or `**` respectively when called with the corresponding function.\n- (optional) `params`: function-specific configurations to apply with the function calls.\n- (optional) `id`: A unique identifier to group function calls.\n\nA table that has keys other than the above keys would not be treated as a function call. Any function calls in different places of the config with the same `id` would be treated as a single group. Tables without an `id` are grouped based on their key-value pairs. Groups are used to identify how some parameters affect the results (e.g., `unique` for `choices`). Function calls can also be in `configurations.variabels`. Note that all function calls are made after the `extends` are resolved and variables from `configurations.variabels` are replaced.\n\n### Supported functions\n- `choices`: Calls [random.choices](https://docs.python.org/3/library/random.html#random.choices). `params` can be a table/dictionary which can have the key `unique`. The value of `unique` must be `true` or `false`. By default `unique` is `false`. If it's `true`, within a group of function calls, no value from the population passed to `random.choices` is repeated for a given participant.\n\n### Example function calls\n```toml\nparam = { function_name = \"choices\", args = [[1 , 2 , 3 , 4]], params = { unique = true } }\n```\n```toml\nparam = { foo = \"test\", bar = { function_name = \"choices\", args = { population = [\"w\", \"x\", \"y\", \"z\"], k = 1 } } }\n```\n\nFor more on the `experiemnt-server` and how it can be used see the [wiki](https://github.com/ahmed-shariff/experiment_server/wiki)\n\n# Wishlist (todo list?)\n- Improved docs\n - Add the option of using dict values in order\n",
"bugtrack_url": null,
"license": "MIT",
"summary": "Server for experiments to get configuarations from",
"version": "0.3.3",
"project_urls": {
"Documentation": "https://shariff-faleel.com/experiment_server/documentation/",
"Homepage": "https://shariff-faleel.com/experiment_server/",
"Repository": "https://github.com/ahmed-shariff/experiment_server"
},
"split_keywords": [
"experiment",
" study-design"
],
"urls": [
{
"comment_text": "",
"digests": {
"blake2b_256": "22a5f902f5bd1853e8b50d8743f540816199fedeece3ef35b2715f47cf139b05",
"md5": "02371a510494736f8962e52f85021daf",
"sha256": "faac32a7795c0e11fe20eba6ee0b567aed3c835b9416808d5ee7775f876d39d6"
},
"downloads": -1,
"filename": "experiment_server-0.3.3-py3-none-any.whl",
"has_sig": false,
"md5_digest": "02371a510494736f8962e52f85021daf",
"packagetype": "bdist_wheel",
"python_version": "py3",
"requires_python": ">=3.8",
"size": 167569,
"upload_time": "2024-08-23T22:26:26",
"upload_time_iso_8601": "2024-08-23T22:26:26.012577Z",
"url": "https://files.pythonhosted.org/packages/22/a5/f902f5bd1853e8b50d8743f540816199fedeece3ef35b2715f47cf139b05/experiment_server-0.3.3-py3-none-any.whl",
"yanked": false,
"yanked_reason": null
},
{
"comment_text": "",
"digests": {
"blake2b_256": "fd80579dfb10d75c5cddd7ce1a72a1a6803fd225ae6f6fee52c9757f8ac70d32",
"md5": "0a397f4883dba01233569d533f0cf3ad",
"sha256": "b6acdd1e52e398b1cc31a043048971e88a7b364461b820442dbed42caf7dbd10"
},
"downloads": -1,
"filename": "experiment_server-0.3.3.tar.gz",
"has_sig": false,
"md5_digest": "0a397f4883dba01233569d533f0cf3ad",
"packagetype": "sdist",
"python_version": "source",
"requires_python": ">=3.8",
"size": 159573,
"upload_time": "2024-08-23T22:26:27",
"upload_time_iso_8601": "2024-08-23T22:26:27.769398Z",
"url": "https://files.pythonhosted.org/packages/fd/80/579dfb10d75c5cddd7ce1a72a1a6803fd225ae6f6fee52c9757f8ac70d32/experiment_server-0.3.3.tar.gz",
"yanked": false,
"yanked_reason": null
}
],
"upload_time": "2024-08-23 22:26:27",
"github": true,
"gitlab": false,
"bitbucket": false,
"codeberg": false,
"github_user": "ahmed-shariff",
"github_project": "experiment_server",
"travis_ci": false,
"coveralls": false,
"github_actions": true,
"lcname": "experiment_server"
}