Skip to main content

Decorate class methods with shell commands

Project description

qwikfire

Release Build status codecov Commit activity License

Decorate class methods with one or more shell commands

QwikFire allows users to annotate class methods with a list of shell commands. The decorator does not execute these commands but injects an extra (hidden) QwikFire argument into the annotated method's argument list right after self. The method implementation can use the qf handle to execute all commands in the annotation using a one-liner: qf.run().

An brief example should clearly demonstrate the benefits of using the @qwikfire annotation to keep your python code clean and easy to read. The annotation executes two echo commands one after the other using Jinja style variable substitution. If an exception were to occur it will be caught, wrapped within a user defined, domain specific WrappingException which extends QwikFireException, and re-raised:

  class WrappingException(QwikFireException):
    pass

  class MyAnnotatedClass:
    ...
    @qwikfire(WrappingException, "echo {{hello_var}}", "echo {{world_var}}")
    def many_twovars(self, qf: QwikFire) -> str:
      return qf.run(self, hello_var="hello", world_var="world").stripped

  ...
  # invoking the method without the (hidden) QwikFire argument
  instance.many_twovars()

Callers do not include the injected hidden QwikFire argument intended for the implementation to use. Notice that no warnings arise with callers missing the QwikFire argument: i.e. instance.many_twovars().

The example above, demonstrates how almost all the boilerplate for try / except blocks, conditional checks, logging, etc, is gone. Reading and understanding what commands the method executes and what exceptions it raises makes the annotation self documenting.

QwikFire and its annotation is properly typed preventing Python typing tools (i.e pyright, pyre, mypy) from needlessly littering your code with complaint's. This is a common problem with decorators that inject additional parameters since the signature of the definition differs from the signature of callers. Overall the code is much more readable while the pattern results in pythonic OO code when chaining method outputs to other method inputs.

sh package kwargs (see https://sh.readthedocs.io/en/latest/sections/special_arguments.html#) can be used in the run() method. They're prefixed with _, and are passed through to the sh.Command. If the class whose methods are annotated, exposes a dictionary accessor method called sh_defaults(self, method: Caller[..., Any]), the values of the dictionary it returns are used for defaults both for variable substitutions and for pass-through arguments to the sh.Command. NOTE: the method is provided to, if needed, tailor defaults to specific class methods. Method specific kwarg key pairs provided to the run method override these defaults.

Why?

Even with the glorious sh package, I still find myself writing boilerplate code, logging, error handling and raising higher level (wrapper) exceptions specific to the domain of the package or application using sh. Even though sh does a great job minimizing the boilerplate, it still clutters my code, and reduces its readability. Other code maintainers will still need to know about sh and how I used it.

NOTE: Piped commands do NOT work. Use output chaining of one annotated method as input into another, if needed as a one off. Multiple semi-colon separated commands in the same string, i.e. "echo {{hello_var}}; echo {{world_var}}" will NOT work, just use separate commands in the variadic string commands array.

If you find you need these or other shell features you're probably over doing it. Write an actual shell script and execute that instead, or directly use the sh package in your code. QwikFire is purposefully meant to be simple and there to prevent the occasional shell command from cluttering up your code.


Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

qwikfire-0.1.2a1.tar.gz (8.3 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

qwikfire-0.1.2a1-py3-none-any.whl (9.5 kB view details)

Uploaded Python 3

File details

Details for the file qwikfire-0.1.2a1.tar.gz.

File metadata

  • Download URL: qwikfire-0.1.2a1.tar.gz
  • Upload date:
  • Size: 8.3 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: poetry/1.8.4 CPython/3.10.12 Linux/6.8.0-49-generic

File hashes

Hashes for qwikfire-0.1.2a1.tar.gz
Algorithm Hash digest
SHA256 a25eea44534cdd153d02a9da54dd297f9bb90cd974560e31de62abcb92c853a4
MD5 f908cb9509e3e5c4b1583a70d8b0c5a7
BLAKE2b-256 7dd1f15d5579465f22ee28b4ba822390f4a45d24d946a87d75a0d8d5ae61655e

See more details on using hashes here.

File details

Details for the file qwikfire-0.1.2a1-py3-none-any.whl.

File metadata

  • Download URL: qwikfire-0.1.2a1-py3-none-any.whl
  • Upload date:
  • Size: 9.5 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: poetry/1.8.4 CPython/3.10.12 Linux/6.8.0-49-generic

File hashes

Hashes for qwikfire-0.1.2a1-py3-none-any.whl
Algorithm Hash digest
SHA256 3f6fa83528a00bd9f4cc8d17420a383f7e411d81bbf9bd588c9712b09e8363fd
MD5 d14a13c99a5fac845d63f608b892c132
BLAKE2b-256 8d7bbef3fc77aada02fc8680288afe6cf847973d6cb8e609a81ca3d5dd3ca92f

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page