{ "info": { "author": "GE Flight Analytics", "author_email": "AviationAdiSupport@ge.com", "bugtrack_url": null, "classifiers": [ "License :: OSI Approved :: MIT License", "Programming Language :: Python :: 2.7", "Programming Language :: Python :: 3" ], "description": "# emsPy\nA Python Wrapper of EMS API. There is also a R wrapper for EMS API. If you are interest in the R version, please visit . The goal of this project is provide a way to bring EMS data in Python environment via the EMS's RESTful API.\n\n## Branches\n\n* New work is completed in the `master` branch of this repository and may contain breaking changes. Pull requests should be made to this branch.\n* Old stable functionality exists in the `v0.2.1` branch and should never change except for minor fixes.\n* Releases are publicly available on `PyPi` using semantic versioning.\n\n## Installation\n\nInstall from the package index:\n\n```\npip install emspy\n```\n\nAlternatively, the package can be installed from the git repository or a zip package:\n\n1. Download or clone emsPy. If downloaded, unzip the compressed file.\n2. Go to the folder that you unzipped or git-cloned, where you can find `setup.py` file.\n3. At the folder, open the command prompt window and run `pip install .` For dev-mode, run `pip install -e .`\n\n## Make an EMS API Connection\n\nThe optional proxy setting can be passed to the EMS connection object with the following format:\n```python\nproxies = {\n 'http': 'http://{prxy_usrname}:{prxy_password}@{proxy_server_address}:{port}',\n 'https': 'https://{prxy_usrname}:{prxy_password}@{proxy_server_address}:{port}'\n}\n```\n\n```python\nfrom emspy import Connection\n\nc = Connection(\"efoqa_usrname\", \"efoqa_password\", proxies = proxies, server = \"prod\")\n\n```\nWith optional `server` argument, you can select one of the currently available EMS API servers, which are:\n* \"prod\" (default)\n* \"cluster\" (clustered production version)\n* \"stable\" (stable test version)\n* \"beta\" \n* \"nightly\"\n\nFor servers hosted locally or in Azure, the server_url argument should be used instead of the server argument. This argument should be of the format \"/api\". For example, if the server hosting the API is http://abc-api.us.efoqa.com, then the connection object would look like this\n\n```python\nfrom emspy import Connection\n\nc = Connection(\"usrname\", \"password\", proxies=proxies, server_url=\"http://abc-api.us.efoqa.com/api\")\n\n```\n\n## Fight Querying\n\n### Instantiate Query \n\nThe following example instantiates a flight-specific query object that will send queries to the EMS 9 system. \n```python\nfrom emspy.query import FltQuery\n\nquery = FltQuery(c, 'ems9', data_file = 'metadata.db')\n```\nwhere optional `data_file` input specifies the SQLite file that will be used to read/write the meta data in the local machine. If there is no file with a specified file name, a new db file will be created. If no file name is passed, it will generate a db file in the default location (emspy/data). If None is specified, no db file will be created. \n\n\n\n### EMS Database Setup\nThe FDW Flights database is one of the frequently used databases. In order to select it as your database for querying, you can simply run the following line.\n\n```python\nquery.set_database(\"fdw flights\")\n```\n\nIn EMS system, all databases & data fields are organized in hierarchical tree structures. In order to use a database that is not the FDW Flights, you need to tell the query object where in the EMS DB tree your database is at. You can explore the EMS database tree from [EMS Online](http://fas.efoqa.com/Docs/Rest/Demos/DataSources). The following example specifies the location of one of the Event databases in the DB tree and then set the Event database that you want to use:\n\n```python\nquery.update_dbtree(\"fdw\", \"events\", \"standard\", \"p0\")\nquery.set_database(\"p0: library flight safety events\")\n```\n\nThese code lines first send queries to find the database-groups path, **FDW → APM Events → Standard Library Profiles → P0: Library Flight Safety Events**, and then select the \"P0: Library Flight Safety Events\" database that is located at the specified path. For a complete example, please check on [this Gist](https://gist.github.com/KMoon01/d82324594c975104c763140a54133565).\n\n### Data Fields\nSimilar to the databases, the EMS data fields are organized in a tree structure so the steps are almost identical except that you use `update_fieldtree(...)` method in order to march through the tree branches.\n\nBefore calling the `update_fieldtree(...)`, you can call `update_preset_fieldtree()` method to load a basic tree with fields belonging to the following field groups:\n* Flight Information\n* Aircraft Information\n* Navigation Information\n\nLet say you have selected the FDW Flights database. The following code lines will query for the meta-data of basic data fields, and then some of the data fields in the Profile 301 in EMS9. \n\n```python\n# Let the query object load preset data fields that are frequently used\nquery.generate_preset_fieldtree()\n\n# Load other data fields that you want to use\nquery.update_fieldtree(\"profiles\", \"standard\", \"block-cost\", \"p301\",\n \"measured\", \"ground operations (before takeoff)\")\n```\nThe `update_fieldtree(...)` above queries the meta-data of all measurements located at the path, **Profiles → Standard Library Profiles → Block-Cost Model → P301: Block-Cost Model Planned Fuel Setup and Tests → Measured Items →Ground Operations (before takeoff)** in EMS Explorer.\n\n**Caution**: the process of adding a subtree usually requires a very large number of recursive RESTful API calls which take quite a long time. Please try to specify the subtree to as low level as possible to avoid a long processing time.\n\nAs you may noticed in the example codes, you can specify a data entity by the string fraction of its full name. The \"key words\" of the entity name follows this rule:\n* Case insensitive\n* Keyword can be a single word or multiple consecutive words that are found in the full name string\n* Keyword should uniquely specify a single data entity among all children under their parent database group\n* Regular expression is not supported\n\n\n### Saving Meta-Data\nFinally, you can save your the meta-data of the database/data trees for later uses. Once you save it, you can go directly call `set_database(...)` without querying the same meta-data for later executions. However, you will have to update trees again if any of the data entities are modified at the EMS-system side.\n\n```python\n# This will save the meta-data into demo.db file, in SQLite format\nquery.save_metadata()\n```\n\n### Select\nAs a next step, you will start make an actual query. The `select(...)` method is used to select what will be the columns of the returned data for your query. Following is an example:\n\n```python\nquery.select(\"flight date\", \n \"customer id\", \n \"takeoff valid\", \n \"takeoff airport iata code\")\n```\n\nThe passed data fields must be part of the data fields in your data tree. \n\nYou need to make a separate select call if you want to add a field with aggregation applied.\n\n```python\nquery.select(\"P301: duration from first indication of engines running to start\", \n aggregate=\"avg\")\n```\nSupported aggregation functions are:\n* avg\n* count\n* max\n* min\n* stdev\n* sum\n* var\n\nYou may want to define grouping, which is described in the next section, when you want to apply an aggregation function.\n\n`select(...)` method accepts the keywords too, and even a combination of keywords to specify the parent directories of the fields in the data tree. For example, the following keywords are all valid to select \"Flight Date (Exact)\" for query:\n- Search by a consecutive substring. The method returns a match with the shortest field name if there are multiple match.\n - Ex) \"flight date\"\n- Search by exact name. \n - Ex) \"flight date (exact)\"\n- Field name keyword along with multiple keywords for the names of upstream field groups (i.e., directories). \n - Ex) (\"flight info\", \"date (exact)\")\n\n\n### Group by & Order by\nSimilarly, you can pass the grouping and ordering condition:\n\n\n```python\nquery.group_by(\"flight date\",\n \"customer id\",\n \"takeoff valid\",\n \"takeoff airport iata code\")\n\nquery.order_by(\"flight date\")\n# the ascending order is default. You can pass a descending order by optional input:\n# query.order_by(\"flight date\", order=\"desc\")\n```\n\n### Filtering\nCurrently the following conditional operators are supported with respect to the data field types:\n- Number: \"==\", \"!=\", \"<\", \"<=\", \">\", \">=\"\n- Discrete: \"==\", \"!=\", \"in\", \"not in\" (Filtering condition made with value, not discrete integer key)\n- Boolean: \"==\", \"!=\"\n- String: \"==\", \"!=\", \"in\", \"not in\"\n- Datetime: \">=\", \"<\"\n\nFollowing is the example:\n\n\n```python\nquery.filter(\"'flight date' >= '2016-1-1'\")\nquery.filter(\"'takeoff valid' == True\")\n# Discrete field filtering is pretty much the same as string filtering.\nquery.filter(\"'customer id' in ['CQH','EVA']\") \nquery.filter(\"'takeoff airport iata code' == 'KUL'\")\n```\n\nThe current filter method has the following limitation:\n- Single filtering condition for each filter method call\n- Filtering conditions are combined only by \"AND\" relationship\n- The field keyword must be at left-hand side of a conditional expression\n- No support of NULL value filtering, which is being worked on now\n- The datetime condition should be only with the ISO8601 format\n\n### ETC.\nYou can pass additional attributes supported by EMS query:\n\n\n```python\n# Returns only the distinct rows. Turned on as default\nquery.distinct(True)\n\n# If you want get top N the rows of the output data in response to the query, \nquery.get_top(5000)\n\n# This is optional. If you don't set this value, all output data will be returned.\n```\n\n### Viewing JSON Translation of Your Query\nYou can check on the resulting JSON string of the translated query using the following method calls.\n\n\n```python\n# Returns JSON string\n# print query.in_json()\n\n# View in Python's native Dictionary form \nfrom pprint import pprint # This gives you a prettier print\n\nprint(\"\\n\")\npprint(query.in_dict())\n```\n\n\n\n {'distinct': True,\n 'filter': {'args': [{'type': 'filter',\n 'value': {'args': [{'type': 'field',\n 'value': u'[-hub-][field][[[ems-core][entity-type][foqa-flights]][[ems-core][base-field][flight.exact-date]]]'},\n {'type': 'constant',\n 'value': '2016-1-1'},\n {'type': 'constant',\n 'value': 'Utc'}],\n 'operator': 'dateTimeOnAfter'}},\n {'type': 'filter',\n 'value': {'args': [{'type': 'field',\n 'value': u'[-hub-][field][[[ems-core][entity-type][foqa-flights]][[ems-core][base-field][flight.exist-takeoff]]]'}],\n 'operator': 'isTrue'}},\n {'type': 'filter',\n 'value': {'args': [{'type': 'field',\n 'value': u'[-hub-][field][[[ems-core][entity-type][foqa-flights]][[ems-fcs][base-field][fdw-flight-extra.customer]]]'},\n {'type': 'constant',\n 'value': 18},\n {'type': 'constant',\n 'value': 11}],\n 'operator': 'in'}},\n {'type': 'filter',\n 'value': {'args': [{'type': 'field',\n 'value': u'[-hub-][field][[[ems-core][entity-type][foqa-flights]][[[nav][type-link][airport-takeoff * foqa-flights]]][[nav][base-field][nav-airport.iata-code]]]'},\n {'type': 'constant',\n 'value': 'KUL'}],\n 'operator': 'equal'}}],\n 'operator': 'and'},\n 'format': 'display',\n 'groupBy': [{'fieldId': u'[-hub-][field][[[ems-core][entity-type][foqa-flights]][[ems-core][base-field][flight.exact-date]]]'},\n {'fieldId': u'[-hub-][field][[[ems-core][entity-type][foqa-flights]][[ems-fcs][base-field][fdw-flight-extra.customer]]]'},\n {'fieldId': u'[-hub-][field][[[ems-core][entity-type][foqa-flights]][[ems-core][base-field][flight.exist-takeoff]]]'},\n {'fieldId': u'[-hub-][field][[[ems-core][entity-type][foqa-flights]][[[nav][type-link][airport-takeoff * foqa-flights]]][[nav][base-field][nav-airport.iata-code]]]'}],\n 'orderBy': [{'aggregate': 'none',\n 'fieldId': u'[-hub-][field][[[ems-core][entity-type][foqa-flights]][[ems-core][base-field][flight.exact-date]]]',\n 'order': 'asc'}],\n 'select': [{'aggregate': 'none',\n 'fieldId': u'[-hub-][field][[[ems-core][entity-type][foqa-flights]][[ems-core][base-field][flight.exact-date]]]'},\n {'aggregate': 'none',\n 'fieldId': u'[-hub-][field][[[ems-core][entity-type][foqa-flights]][[ems-fcs][base-field][fdw-flight-extra.customer]]]'},\n {'aggregate': 'none',\n 'fieldId': u'[-hub-][field][[[ems-core][entity-type][foqa-flights]][[ems-core][base-field][flight.exist-takeoff]]]'},\n {'aggregate': 'none',\n 'fieldId': u'[-hub-][field][[[ems-core][entity-type][foqa-flights]][[[nav][type-link][airport-takeoff * foqa-flights]]][[nav][base-field][nav-airport.iata-code]]]'},\n {'aggregate': 'avg',\n 'fieldId': u'[-hub-][field][[[ems-core][entity-type][foqa-flights]][[ems-apm][flight-field][msmt:profile-cbaa5341ca674914a6ceccd6f498bffc:msmt-0d7fe63d6863451a9c663a09fd780985]]]'}],\n 'top': 5000}\n\n\n### Run Query and Retrieve Data\nYou can finally send the query to the EMS system and get the data. The output data is returned in Pandas' DataFrame object.\n\n\n\n```python\ndf = query.run()\n\n# This will return your data in Pandas dataframe format\n```\n\nEMS API supports two different query executions which are regular and async queries. The regular query has a data size limit for the output data, which is 25000 rows. On the other hand, the async query is able to handle large output data by letting you send repeated requests for mini batches of the large output data.\n\nThe `run()` method takes care of the repeated async requests for a query whose returning data is expected to be large.\n\nThe batch data size for the async request is set 25,000 rows as default (which is the maximum). If you want to change this size,\n```python\n# Set the batch size as 20,000 rows per request\ndf = query.run(n_row = 20000)\n``` \n\n## Querying Time-Series Data\nYou can query data of time-series parameters with respect to individual flight records. Below is a simple example code that sends a flight query first in order to retrieve a set of flights and then sends queries to get some of the time-series parameters for each of these flights.\n\n```python\n# Flight query with an APM profile. It will return data for 10 flights\nfq = FltQuery(c, \"ems9\", data_file = \"demo.db\")\nfq.set_database(\"fdw flights\")\n# If you reuse the meta-data, you don't need to update db/field trees.\n\nfq.select(\n \"customer id\", \"flight record\", \"airframe\", \"flight date (exact)\",\n \"takeoff airport code\", \"takeoff airport icao code\", \"takeoff runway id\",\n \"takeoff airport longitude\", \"takeoff airport latitude\",\n \"p185: processed date\", \"p185: oooi pushback hour gmt\",\n \"p185: oooi pushback hour solar local\",\n \"p185: total fuel burned from first indication of engines running to start of takeoff (kg)\")\nfq.order_by(\"flight record\", order='desc')\nfq.get_top(10)\nfq.filter(\"'p185: processing state' == 'Succeeded'\")\nflt = fq.run()\n\n# === Run time series query for flights ===\n\ntsq = TSeriesQuery(c, \"ems9\", data_file = \"demo.db\")\ntsq.select(\n \"baro-corrected altitude\", \n \"airspeed (calibrated; 1 or only)\", \n \"ground speed (best avail)\",\n \"egt (left inbd eng)\", \n \"egt (right inbd eng)\", \n \"N1 (left inbd eng) (%)\", \n \"N1 (right inbd eng) (%)\")\n\n# Run querying multiple flights at once. Start time = 0, end time = 15 mins (900 secs) for all flights. \n# A better use case is that those start/end times are fed by timepoint measurements of your APM profile.\nres_dat = tsq.multi_run(flt, start = [0]*flt.shape[0], end = [15*60]*flt.shape[0])\n```\n\nThe inputs to function `multi_run(...)` are:\n* flt : a vector of Flight Records or flight data in Pandas DataFrame format. The dataframe should have a column of flight records with its column name \"Flight Record\"\n* start: a list-like object defining the starting times (secs) of the timepoints for individual flights. The vector length must be the same as the number of flight records\n* end : a list-like object defining the end times (secs) of the timepoints for individual flights. The vector length must be the same as the number of flight records\n* timestep: a list-like object defining the size of timesteps in seconds for individual flights. Default is set 1 second. If you set \"None\", it will use the parameters' own default timesteps. The vector length must be the same as the number of flight records\n\nThe output will be Python dictionary object which contains the following data:\n* flt_data : Dictionary. Copy of the flight data for each flight\n* ts_data : Pandas DataFrame. the time series data for each flight\n\nIn case you just want to query for a single flight, `run(...)` function will be better suited. Below is an example of time-series querying for a single flight.\n\n```python\nres_dat = tsq.run(1901112, start=0, end=900)\n```\nThis function will return a Pandas DataFrame that contains timepoints from 0 to 900 secs and corresponding values for selected parameters. You can also pass a timestep as an optional argument. Default timestep is set 1.0 sec.\n\n### Querying Analytics\n\nYou can retrieve a list of physical parameters for a flight by utilizing methods in the Analytic class. \n\nFirst, instantiate an analytic_query object with your connection (`c`) and a system id (`1`):\n\n```python\nanalytic_query = Analytic(c, 1)\n```\n\nThen, call `analytic_query.get_physical_parameter_list(fr)` with a valid Flight Record:\n\n```python\nphysicals = analytic_query.get_physical_parameter_list(flight_id = flight_id)\n```\n\n`physicals.sample(3)` looks like:\n\n| | id | name | description | units |\n|----:|:-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|:--------------------------|:------------------------------------------|:--------|\n| 277 | foobar123 | PARAMETER 1 | Uid: P1\\nName: PARAMETER 1. | DEG |\n| 696 | foobar124 | PARAMETER 2 | Uid: P2\\nName: PARAMETER 2. | YR |\n| 48 | foobar125 | PARAMETER 3 | Uid: P3\\nName: PARAMETER 3. | DEG |\n\nYou can also retrieve analytic metadata for a Flight (including for physical parameters):\n\n```python\nanalytic_id = physicals['id'].iloc[0]\nmetadata = analytic_query.get_flight_analytic_metadata(analytic_id=analytic_id, flight_id=flight_id)\n```\n\n`metadata` looks like:\n\n```python\n{\n 'Display\\\\Leading Zero': True,\n 'Parameter\\\\Name': Foo\n}\n```\n\n", "description_content_type": "text/markdown", "docs_url": null, "download_url": "", "downloads": { "last_day": -1, "last_month": -1, "last_week": -1 }, "home_page": "https://github.com/ge-flight-analytics/emspy", "keywords": "", "license": "MIT", "maintainer": "", "maintainer_email": "", "name": "emspy", "package_url": "https://pypi.org/project/emspy/", "platform": "", "project_url": "https://pypi.org/project/emspy/", "project_urls": { "Homepage": "https://github.com/ge-flight-analytics/emspy" }, "release_url": "https://pypi.org/project/emspy/0.5.0/", "requires_dist": [ "numpy (>=1.13.3)", "pandas", "future", "networkx" ], "requires_python": "", "summary": "A Python EMS RESTful API Client/Wrapper", "version": "0.5.0", "yanked": false, "yanked_reason": null }, "last_serial": 10702357, "releases": { "0.2.1": [ { "comment_text": "", "digests": { "md5": "64e069858c479a56aeb3b4cc1a175249", "sha256": "668496d9c5545820256e08f0ebf40d379458ac9486dfe0398b74a6553e606698" }, "downloads": -1, "filename": "emspy-0.2.1-py3-none-any.whl", "has_sig": false, "md5_digest": "64e069858c479a56aeb3b4cc1a175249", "packagetype": "bdist_wheel", "python_version": "py3", "requires_python": null, "size": 34153, "upload_time": "2019-10-30T00:34:49", "upload_time_iso_8601": "2019-10-30T00:34:49.142714Z", "url": "https://files.pythonhosted.org/packages/a6/1a/0aef98715a8f44d7a98c3463b4714e021ef27ed32ddbae6280c22b074de8/emspy-0.2.1-py3-none-any.whl", "yanked": false, "yanked_reason": null }, { "comment_text": "", "digests": { "md5": "c8dce09f772118eb33e977415707cab6", "sha256": "3e0162c10601ea949224c502148b384c95f6b485197b008d5ae7225977045467" }, "downloads": -1, "filename": "emspy-0.2.1.tar.gz", "has_sig": false, "md5_digest": "c8dce09f772118eb33e977415707cab6", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 34139, "upload_time": "2019-10-30T00:34:51", "upload_time_iso_8601": "2019-10-30T00:34:51.750789Z", "url": "https://files.pythonhosted.org/packages/24/3f/22a11e1c17080233ebdc19866f6cf3aafdc9a4c595c5c5cd2ab08def220c/emspy-0.2.1.tar.gz", "yanked": false, "yanked_reason": null } ], "0.2.2": [ { "comment_text": "", "digests": { "md5": "6b85d83b05afcd0201353bc0459c3932", "sha256": "6bb647c125547e1003e3ff663afb65720222bcbaaa8449cb55abc5deadf764bc" }, "downloads": -1, "filename": "emspy-0.2.2-py3-none-any.whl", "has_sig": false, "md5_digest": "6b85d83b05afcd0201353bc0459c3932", "packagetype": "bdist_wheel", "python_version": "py3", "requires_python": null, "size": 34243, "upload_time": "2019-11-04T22:18:23", "upload_time_iso_8601": "2019-11-04T22:18:23.193461Z", "url": "https://files.pythonhosted.org/packages/fc/fb/483213bec56e2bc5d1bb3bf209964cf9735a41ecb613ead574cd1f87c086/emspy-0.2.2-py3-none-any.whl", "yanked": false, "yanked_reason": null }, { "comment_text": "", "digests": { "md5": "3c6737ef33b9c0ed43bb14c04cdeb497", "sha256": "4e2171fa730930739c3faf9154b5006f00d1f9fa497fb2cac08f1a259c02c57e" }, "downloads": -1, "filename": "emspy-0.2.2.tar.gz", "has_sig": false, "md5_digest": "3c6737ef33b9c0ed43bb14c04cdeb497", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 34308, "upload_time": "2019-11-04T22:18:24", "upload_time_iso_8601": "2019-11-04T22:18:24.874484Z", "url": "https://files.pythonhosted.org/packages/b6/09/bc6a96178ac53dd8b42c0e3830408c55d9898b4a0625c2aa71f80a130856/emspy-0.2.2.tar.gz", "yanked": false, "yanked_reason": null } ], "0.2.3": [ { "comment_text": "", "digests": { "md5": "9852dd42ec5c657842ef3daa6e364e09", "sha256": "4f40571affd683f88e35f5d3d60fb79828f4ce1a6004beaf1195345b1fb8a59b" }, "downloads": -1, "filename": "emspy-0.2.3-py3-none-any.whl", "has_sig": false, "md5_digest": "9852dd42ec5c657842ef3daa6e364e09", "packagetype": "bdist_wheel", "python_version": "py3", "requires_python": null, "size": 34270, "upload_time": "2019-11-04T23:48:38", "upload_time_iso_8601": "2019-11-04T23:48:38.740058Z", "url": "https://files.pythonhosted.org/packages/58/df/57b5be08014b7948151588bd2a89b9849c31acc0013c93c6efedcc7097ce/emspy-0.2.3-py3-none-any.whl", "yanked": false, "yanked_reason": null }, { "comment_text": "", "digests": { "md5": "0139c4b52ff41e4595c47fc4168db7bb", "sha256": "7c8bddb8e22451786bbf722031651697668b24d0b3a060a7b59687fa5f45fdb4" }, "downloads": -1, "filename": "emspy-0.2.3.tar.gz", "has_sig": false, "md5_digest": "0139c4b52ff41e4595c47fc4168db7bb", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 34315, "upload_time": "2019-11-04T23:48:41", "upload_time_iso_8601": "2019-11-04T23:48:41.601896Z", "url": "https://files.pythonhosted.org/packages/45/63/597ce1e8dc3c99cd4e731925da689320c9a23473de671ea5de0c411a69a3/emspy-0.2.3.tar.gz", "yanked": false, "yanked_reason": null } ], "0.2.4": [ { "comment_text": "", "digests": { "md5": "a5883304c1f1364db02a3104c372fba2", "sha256": "6e1fe89cdb3f7e061c7016d5b8ae061691b81cbb12cd2836ab4395fab8d4c3fb" }, "downloads": -1, "filename": "emspy-0.2.4-py3-none-any.whl", "has_sig": false, "md5_digest": "a5883304c1f1364db02a3104c372fba2", "packagetype": "bdist_wheel", "python_version": "py3", "requires_python": null, "size": 34269, "upload_time": "2019-12-17T17:58:52", "upload_time_iso_8601": "2019-12-17T17:58:52.526895Z", "url": "https://files.pythonhosted.org/packages/f7/e0/b1214dbbf8242e8847a05ab9e54ff6cd03286088a30da685bdc28e4dd319/emspy-0.2.4-py3-none-any.whl", "yanked": false, "yanked_reason": null }, { "comment_text": "", "digests": { "md5": "4fa363d88295f03030597fcee31f7440", "sha256": "7e2580ff70080952106bdb73b46bb25f7b13f6c2fe1db157e35453a6cf39a32f" }, "downloads": -1, "filename": "emspy-0.2.4.tar.gz", "has_sig": false, "md5_digest": "4fa363d88295f03030597fcee31f7440", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 34314, "upload_time": "2019-12-17T17:58:53", "upload_time_iso_8601": "2019-12-17T17:58:53.963433Z", "url": "https://files.pythonhosted.org/packages/5f/c4/57a2fed2d2cd5dec87afec57595af256048fdf65f2a2ead1c90678edede3/emspy-0.2.4.tar.gz", "yanked": false, "yanked_reason": null } ], "0.2.5": [ { "comment_text": "", "digests": { "md5": "5830f4abe9979aefb40a0243821e1f64", "sha256": "e1face17973d91947f0079696e1ebbcf4388646cbb9bd6c0c16ed015c17d3af4" }, "downloads": -1, "filename": "emspy-0.2.5-py3-none-any.whl", "has_sig": false, "md5_digest": "5830f4abe9979aefb40a0243821e1f64", "packagetype": "bdist_wheel", "python_version": "py3", "requires_python": null, "size": 34279, "upload_time": "2020-03-02T16:59:13", "upload_time_iso_8601": "2020-03-02T16:59:13.324422Z", "url": "https://files.pythonhosted.org/packages/55/81/33bdf59fcb1309174ddbf461779fb1059ad75602f6d85d764fb3a2d60fa0/emspy-0.2.5-py3-none-any.whl", "yanked": false, "yanked_reason": null }, { "comment_text": "", "digests": { "md5": "434c6e1ba59e83a78c12fe98f05f758f", "sha256": "1927a75be7f5e89d0b0a8f601dc7ff712afc916545f341c9992a6210a1eab351" }, "downloads": -1, "filename": "emspy-0.2.5.tar.gz", "has_sig": false, "md5_digest": "434c6e1ba59e83a78c12fe98f05f758f", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 34465, "upload_time": "2020-03-02T16:59:15", "upload_time_iso_8601": "2020-03-02T16:59:15.049968Z", "url": "https://files.pythonhosted.org/packages/ef/29/01c5c8313316d6eabe2e68ae075cad2bef329afa26a357b194e3f91c6bc9/emspy-0.2.5.tar.gz", "yanked": false, "yanked_reason": null } ], "0.2.6": [ { "comment_text": "", "digests": { "md5": "429be2608fde060ccb42d391bb7dc16c", "sha256": "e874457bf7367265d14189e040088b26f4fa7f15d5cee78c782985682a5d84b2" }, "downloads": -1, "filename": "emspy-0.2.6-py3-none-any.whl", "has_sig": false, "md5_digest": "429be2608fde060ccb42d391bb7dc16c", "packagetype": "bdist_wheel", "python_version": "py3", "requires_python": null, "size": 36120, "upload_time": "2020-05-26T15:17:34", "upload_time_iso_8601": "2020-05-26T15:17:34.507529Z", "url": "https://files.pythonhosted.org/packages/6c/80/70a9d7814199deb8155e18be69cf0c04739239c1c3c8071ba77ae508c43d/emspy-0.2.6-py3-none-any.whl", "yanked": false, "yanked_reason": null }, { "comment_text": "", "digests": { "md5": "d24045bf1e67aa2beb197698c904c9c6", "sha256": "3c1d219eefa962ecaf7f0ae5a6c2a66da421a95abad92a4d90bfd7066ea8cc2e" }, "downloads": -1, "filename": "emspy-0.2.6.tar.gz", "has_sig": false, "md5_digest": "d24045bf1e67aa2beb197698c904c9c6", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 35830, "upload_time": "2020-05-26T15:17:35", "upload_time_iso_8601": "2020-05-26T15:17:35.942530Z", "url": "https://files.pythonhosted.org/packages/05/6f/23a1cb4936b53ed807fb44fe8aae375db406ba9683fb61573ff93719e490/emspy-0.2.6.tar.gz", "yanked": false, "yanked_reason": null } ], "0.3.0": [ { "comment_text": "", "digests": { "md5": "654efa7b782952e90b3b1b24a8636571", "sha256": "d5d31a796ec5afb72ad7802aaa8b9c80e9a13443e41f99c0a7690e5dc9006e46" }, "downloads": -1, "filename": "emspy-0.3.0-py3-none-any.whl", "has_sig": false, "md5_digest": "654efa7b782952e90b3b1b24a8636571", "packagetype": "bdist_wheel", "python_version": "py3", "requires_python": null, "size": 37004, "upload_time": "2020-07-24T20:23:10", "upload_time_iso_8601": "2020-07-24T20:23:10.862844Z", "url": "https://files.pythonhosted.org/packages/ef/be/0323594447a131455945df4c4c21ae5fa4705bf67773c2682c093482d32d/emspy-0.3.0-py3-none-any.whl", "yanked": false, "yanked_reason": null }, { "comment_text": "", "digests": { "md5": "bab7a9dd9684a9a23c7370df81c84899", "sha256": "ab8d010374788ec0445bc45de6719c5a7d7fa1b710994338fd35da3fb218d0af" }, "downloads": -1, "filename": "emspy-0.3.0.tar.gz", "has_sig": false, "md5_digest": "bab7a9dd9684a9a23c7370df81c84899", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 36810, "upload_time": "2020-07-24T20:23:12", "upload_time_iso_8601": "2020-07-24T20:23:12.286972Z", "url": "https://files.pythonhosted.org/packages/40/80/47b830b8fac6c43094103b5a0c9b67afcd74873a9b9716f40b66f1d78952/emspy-0.3.0.tar.gz", "yanked": false, "yanked_reason": null } ], "0.4.0": [ { "comment_text": "", "digests": { "md5": "744bbeab02535fe51359c471e46b2107", "sha256": "bb32dd459ebb42a588aedf943fcf65d45b5a91a28fc2142b42ae815e06877bd9" }, "downloads": -1, "filename": "emspy-0.4.0-py3-none-any.whl", "has_sig": false, "md5_digest": "744bbeab02535fe51359c471e46b2107", "packagetype": "bdist_wheel", "python_version": "py3", "requires_python": null, "size": 44108, "upload_time": "2020-09-01T18:17:13", "upload_time_iso_8601": "2020-09-01T18:17:13.806776Z", "url": "https://files.pythonhosted.org/packages/25/1f/78c6b66e25c059efb8899667ece5d1ee50697e395c9d9bc20c44f585d04f/emspy-0.4.0-py3-none-any.whl", "yanked": true, "yanked_reason": "Released prematurely, prefer 0.4.1" }, { "comment_text": "", "digests": { "md5": "84390cee23f7595d8b63965840433607", "sha256": "4ff5189bbcaf324e32858dcfd9bed3b417696b0ecd2b421092a90ebf35d4debe" }, "downloads": -1, "filename": "emspy-0.4.0.tar.gz", "has_sig": false, "md5_digest": "84390cee23f7595d8b63965840433607", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 41997, "upload_time": "2020-09-01T18:17:15", "upload_time_iso_8601": "2020-09-01T18:17:15.862980Z", "url": "https://files.pythonhosted.org/packages/b2/2d/4fc8f935b7d1b4e06cca3ca86b341a0e4f4a7f719b3c531e99bb089cb737/emspy-0.4.0.tar.gz", "yanked": true, "yanked_reason": "Released prematurely, prefer 0.4.1" } ], "0.4.1": [ { "comment_text": "", "digests": { "md5": "3b149ba931b1834f4550134527f22e2b", "sha256": "e8a66968d06f1eff393ea5ea2b498b080b6855c4f5a1e828241553d5e55f30d3" }, "downloads": -1, "filename": "emspy-0.4.1-py3-none-any.whl", "has_sig": false, "md5_digest": "3b149ba931b1834f4550134527f22e2b", "packagetype": "bdist_wheel", "python_version": "py3", "requires_python": null, "size": 44129, "upload_time": "2020-09-02T20:58:29", "upload_time_iso_8601": "2020-09-02T20:58:29.572665Z", "url": "https://files.pythonhosted.org/packages/72/14/30e47b4cd864c35ae78886939d7c96e06b77c8ed18b265a64b32834153be/emspy-0.4.1-py3-none-any.whl", "yanked": false, "yanked_reason": null }, { "comment_text": "", "digests": { "md5": "b63242007b1752ff6bf7999779946e97", "sha256": "9a0ba1d6c8d3e1c86350bb549e2f0c630f29987ba2b7d1db468389d1523ff841" }, "downloads": -1, "filename": "emspy-0.4.1.tar.gz", "has_sig": false, "md5_digest": "b63242007b1752ff6bf7999779946e97", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 42039, "upload_time": "2020-09-02T20:58:32", "upload_time_iso_8601": "2020-09-02T20:58:32.077630Z", "url": "https://files.pythonhosted.org/packages/cf/97/d410005c737dbb54f161a22d6b1411c48737977afc99dfc0eb5f702ceeb6/emspy-0.4.1.tar.gz", "yanked": false, "yanked_reason": null } ], "0.5.0": [ { "comment_text": "", "digests": { "md5": "ff19312d80bc841277fcc6e5c52d4a2a", "sha256": "8a126195739324afb2e4bd9e4f23af7b63ec5f8addbfaf93228ff651c5126832" }, "downloads": -1, "filename": "emspy-0.5.0-py3-none-any.whl", "has_sig": false, "md5_digest": "ff19312d80bc841277fcc6e5c52d4a2a", "packagetype": "bdist_wheel", "python_version": "py3", "requires_python": null, "size": 45222, "upload_time": "2021-06-21T14:14:18", "upload_time_iso_8601": "2021-06-21T14:14:18.332198Z", "url": "https://files.pythonhosted.org/packages/76/21/fcaa9ffc913a367c40fc88615a7cacea5ec6c2367f15fec1e306ad61c5a6/emspy-0.5.0-py3-none-any.whl", "yanked": false, "yanked_reason": null }, { "comment_text": "", "digests": { "md5": "8ccf200c9e58dd85190ce4e54b40d66a", "sha256": "b24e8625d0304910f831dddfe514d66b21d1100cc0046b6e2b8c9f378aa00fba" }, "downloads": -1, "filename": "emspy-0.5.0.tar.gz", "has_sig": false, "md5_digest": "8ccf200c9e58dd85190ce4e54b40d66a", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 43553, "upload_time": "2021-06-21T14:14:20", "upload_time_iso_8601": "2021-06-21T14:14:20.182576Z", "url": "https://files.pythonhosted.org/packages/de/ca/44a563e517d5d3e3db901c7bf41ec18d3e517a4a10705df2e5d9d3703d6a/emspy-0.5.0.tar.gz", "yanked": false, "yanked_reason": null } ] }, "urls": [ { "comment_text": "", "digests": { "md5": "ff19312d80bc841277fcc6e5c52d4a2a", "sha256": "8a126195739324afb2e4bd9e4f23af7b63ec5f8addbfaf93228ff651c5126832" }, "downloads": -1, "filename": "emspy-0.5.0-py3-none-any.whl", "has_sig": false, "md5_digest": "ff19312d80bc841277fcc6e5c52d4a2a", "packagetype": "bdist_wheel", "python_version": "py3", "requires_python": null, "size": 45222, "upload_time": "2021-06-21T14:14:18", "upload_time_iso_8601": "2021-06-21T14:14:18.332198Z", "url": "https://files.pythonhosted.org/packages/76/21/fcaa9ffc913a367c40fc88615a7cacea5ec6c2367f15fec1e306ad61c5a6/emspy-0.5.0-py3-none-any.whl", "yanked": false, "yanked_reason": null }, { "comment_text": "", "digests": { "md5": "8ccf200c9e58dd85190ce4e54b40d66a", "sha256": "b24e8625d0304910f831dddfe514d66b21d1100cc0046b6e2b8c9f378aa00fba" }, "downloads": -1, "filename": "emspy-0.5.0.tar.gz", "has_sig": false, "md5_digest": "8ccf200c9e58dd85190ce4e54b40d66a", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 43553, "upload_time": "2021-06-21T14:14:20", "upload_time_iso_8601": "2021-06-21T14:14:20.182576Z", "url": "https://files.pythonhosted.org/packages/de/ca/44a563e517d5d3e3db901c7bf41ec18d3e517a4a10705df2e5d9d3703d6a/emspy-0.5.0.tar.gz", "yanked": false, "yanked_reason": null } ], "vulnerabilities": [] }