Skip to content

Latest commit

 

History

History
114 lines (78 loc) · 4.96 KB

README.md

File metadata and controls

114 lines (78 loc) · 4.96 KB

e-mission-common library

Part of the greater e-mission project.

This library contains code used by multiple e-mission components including e-mission-server, e-mission-phone, em-public-dashboard, and op-admin-dashboard.

This repository uses the Transcrypt library to compile Python code to JavaScript. This allows us to write and maintain code in 1 language and import it for use in all 4 of the above repos.

Setup

. bin/setup.sh

This will set up a venv, and install dependencies from both pip and npm. Re-run this if you change the dependencies in requirements.txt or package.json.

To contribute

  1. Make your changes to Python code under the src directory.
  2. Run bash bin/compile_to_js.sh to build the JavaScript. This will produce output JS files in the emcommon_js directory.
  3. Commit changes from both the src and emcommon_js directories to your branch.

Tips for writing code to work in both Python and JavaScript

  1. Use the emcommon.logger module for all logs; it is set up to work in both languages.

    import emcommon.logger as Log
  2. When you need to run a snippet in Python but not in JavaScript, you can use the skip pragma to skip one line or multiple lines.

    Log.info("This line executes in both Python and JavaScript")
    
    Log.info("But this line only executes in in Python") # __: skip
    
    # __pragma__('skip')
    Log.info("This whole block of lines only executes in Python")
    Log.info("JS ignores anything between the skip and noskip pragma comments")
    # __pragma__('noskip')
    
    Log.info("This line is back to executing in both Python and JavaScript")
  3. When you need to run something in JavaScript but not in Python, you can use an "executable comment", which starts with '''? and ends with ?'''.

    Log.info("This line executes in both Python and JavaScript")
    
    '''?
    Log.info("This only executes in JavaScript")
    Log.info("The Transcrypt compiler will convert this to JavaScript code")
    Log.info("But regular Python will just see it as a comment and ignore it")
    ?'''
  4. If you need to insert raw JavaScript code, you can use the js pragma.

    Log.info("This is Python code that executes in both Python and JavaScript")
    # __pragma__('js', '{}', 'alert("This is raw JavaScript code that executes in JavaScript")')

    If you need multiple lines of raw JavaScript, you can wrap the js pragma in an "executable comment".

    '''?
    __pragma__('js', '{}', """
      let msg = "This is raw JavaScript code that executes in JavaScript";
      msg += ", and it can be multiple lines";
      alert(msg);
    """)
    ?'''

For more detail, refer to the Transcrypt docs on the many kinds of pragmas available.

Dev workflow to test local changes in other repos

For Python:

  1. Make your local changes in the 'src' directory.
  2. From your other repo, run pip install -e <path_to_this_repo> to use the local version of e-mission-common.

Alternatively, you can use pip install git+https://github.com/JGreenlee/e-mission-common@master to test the master branch or any other branch or tag.

For JavaScript:

  1. Make your local changes in the 'src' directory.
  2. Run bash bin/compile_to_js.sh to build the JavaScript.
  3. From this repo, run npm link to establish a symlink to your local version of e-mission-common.
  4. From the other repo, run npm link e-mission-common to use the symlinked version of this repo.

Unit testing

Due to the nature of this library, it is critical to test both the Python source and the compiled JavaScript. Ideally, we can write a test suite in a .py file and have it run in both Python and JavaScript environments.

This is possible, to a degree, using transcrypt to compile test files to JavaScript, and then running jest on the compiled JavaScript.

pytest was chosen over unittest because it is more flexible, allowing tests to be written in a way that can be compiled to Jest-compatible JavaScript. (It is also more concise and friendly to write.)

See the tests directory for .py files that are not accompanied by a .js file. These files are intended to be run in both Python and JavaScript environments.

Dedicated JS test files

There may be testing scenarios that must significantly diverge between Python and JavaScript versions. In these cases, we can write a separate .js file next to the .py file of the same name. In this case, no compilation is necessary; pytest will run the .py file and jest will run the .js.

Running the tests

. bin/run_pytest.sh
. bin/run_jest.sh