lms-waltz


Namelms-waltz JSON
Version 0.2.7 PyPI version JSON
download
home_pagehttps://github.com/acbart/waltz
SummaryCoordinate resources between an LMS like Canvas and a local directory
upload_time2023-01-02 16:55:59
maintainer
docs_urlNone
authorAustin Cory Bart
requires_python>=3.6
license
keywords lms learning management system curriculum curricular resources
VCS
bugtrack_url
requirements tqdm requests ruamel.yaml jinja2 markdown python-frontmatter requests_cache python-dateutil html2text canvasapi tabulate natsort pygments beautifulsoup4
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # waltz

A software system for synchronizing curricular materials between a
Learning Management Systems (LMS) and your local filesystem.

Major features include:
* Pull and push data between an LMS and the file system
* Lightweight, friendly Markdown-based format for local resources
* Diffing of local and remote content
* List and search for remote resources

We aim to support the following LMSes:
* Canvas (pages, assignments, quizzes)
* BlockPy (assignment, groups) <- In-progress

# Installation

You can install Waltz from PyPi (the package name is `lms-waltz` even though the module and command line script is `waltz`):

```console
foo@bar:~$ pip install lms-waltz
```

You can also install our dev version from GitHub:

```console
foo@bar:~$ pip install git+https://github.com/acbart/waltz.git
```

## Setup Waltz

Waltz can synchronize content between a local directory and a remote server.
You'll need to initialize the local directory, whether it is empty or already has your learning materials.

```console
foo@bar:~$ waltz init
```

This will create two new files which should both be included in your `.gitignore`:

* `.waltz` is a plain-text YAML file with settings for the current course repository.
* `.waltz.db` is a SQLite database used in file uploading/downloading.

## Setup a Service

Before you can start interacting with an LMS, you'll need to configure an
instance of a service. For example, to configure a new Canvas service, you'll need the following:

1. A short, easy-to-type name for the service instance (e.g., `ud_canvas` or `cs1014_canvas`)
2. [API token](https://community.canvaslms.com/docs/DOC-10806-4214724194)
3. The URL base for your Canvas site (e.g., `https://canvas.instructure.com/`)
4. The Course ID that you want to connect to (usually a large number)

Then, you can use the command below:

```console
foo@bar:~$ waltz configure canvas <name> --base <url> --course <id> --token <token>
```

A more concrete example:

```console
foo@bar:~$ waltz configure canvas ud_canvas --base https://udel.instructure.com/ --course 4234343432343 --token 1081~zJIOJSDFBBSDFIJAIJ==>...
```

If things went well, you can list the available services:

```console
foo@bar:~$ waltz list
The following services are available:
         local:
                 ./
         canvas:
                 ud_canvas
         blockpy: (none configured)
```

By default, there's a Local service for the current directory (representing the connection to the filesystem).
You can have more than one instance of a service, which can allow you to
access multiple data sources from one course (e.g., to transfer resources between semesters).
For convenience, you can refer to the first instance of a service by the service's name.
In our case, anywhere that we use `ud_canvas` we could use `canvas` instead.

## Identifying Resources

You can list available resources for a service:

```console
foo@bar:~$ waltz list <service> <category>
```

So the following code checks the Pages for the Canvas course. Note that
you could use the specific instance name instead of `canvas`, and either
the singular or plural form of `pages`.

```console
foo@bar:~$ waltz list canvas pages
Remote    Local    Title                      Path
--------  -------  -------------------------  --------------------
Yes       Yes      Abstraction                pages\Abstraction.md
Yes       Yes      NetLogo-2                  NetLogo-2.md
Yes       No       Overview of Functions
Yes       No       Syllabus
Yes       No       Themes
Yes       Yes      Turtles                    pages\Turtles.md
```

The `local` service allows you to omit the category (but you can filter by category if you want):

```console
foo@bar:~$ waltz list local

Resource         Title                        Path
---------------  ---------------------------  ---------------------------------------------------------
[unknown]        NetLogo-2                    NetLogo-2.md
[unknown]                                     pages\Abstraction.md
[Page]           Turtles                      pages\Turtles.md
[quiz]           Backup Quiz                  quizzes\Backup Quiz.md
[unknown]                                     quizzes\Mangled Resource.md
[quiz]           New Quiz                     quizzes\New Quiz.md
[assignment]     Video Upload Test            quizzes\Video Upload Test.md
[quiz question]  Complex Matching Question    quizzes\New Quiz Questions\Complex Matching Question.md
[quiz question]  Essay Question               quizzes\New Quiz Questions\Essay Question.md
```

The local service only knows the resource category and title if that information
is available in the front-matter of the Markdown file. Otherwise, it can
only report the path of potential resource files.

Similar to Git, you can use `waltz` commands from child folders and the
system will search up for the configuration and use it appropriately.

## Managing Resources

The two basic commands for moving resources are `pull` and `push`, moving resources between
the local service and a remote one. If the local version does not already exist,
it will be created in the current directory with a filename based on the title of
the resource from the LMS.

```console
foo@bar:~$ waltz pull <service> <category> <title>
foo@bar:~$ waltz push <service> <category> <title>
```

Generally, you want to fully specify the remote service, resource category,
and title of the resource you want to use. This avoids ambiguity.

```
foo@bar:~$ waltz pull canvas assignment "Project 3"
foo@bar:~$ ls
Project 3.md
foo@bar:~$ waltz push canvas assignment "Project 3"
```

However, the commands attempt to work intelligently to find the appropriate file given minimal amounts
of information, particularly for pushing. If the given title matches the title of a file
in the current directory (either based on the filename or the front-matter of the file),
and that file has a resource specified, then the appropriate service can automatically be inferred.

```console
foo@bar:~$ waltz push "Turtles"
```

If the desired files are not in the present folder, it will search subfolders.

TOOD: We are working on ways to further disambiguate resources that have
the same name. This will be accomplished using their IDs or some other attribute.
In general, however, you can avoid ambiguity by organizing the file system
and your LMS resources appropriately.

## Diffing Resources

Pushing and pulling resources can get quite complex, so it can be
helpful to find out how the local version of a resource is different
from the remote. The `diff` command has similar syntax to the other
actions, but does not modify the filesystem when it is run.

```console
foo@bar:~$ waltz diff canvas page "Turtles"
```

![Screenshot showing web-based output of the Waltz Diff command, with the local version of a file on the left and the remote version on the right. Changes are highlighted using red, green, and yellow highlighting.](docs/example-diff-1.png)

By default, `diff`ing will create a new HTML file adjacent to the
resource and automatically open that file in the browser. There is also
simplified console output available.

## Download, Decode, Encode, and Upload

`Pull`ing is actually a two-step process: first you `download` a resource, and then you `decode` it.
Similarly, `Push`ing requires you to `encode` a resource and then `upload` it.

The `upload` and `download` commands (which are available in Waltz) move data between Canvas and the
local Waltz SQLite database. Generally, we download raw data from the remote service (e.g., a JSON blob)
and upload using whatever API is available for the service.

The `encode` and `decode` commands (also available) move data between the local filesystem and the
local Waltz SQLite database. Generally, this means we are decoding the raw data into a Markdown file
or potentially even multiple files.

## Templates

**TODO: This feature is still in progress**

The HTML body of a Resource can be built using a Template. We use the
Jinja2 format to embed data.

## Previewing Resources

**TODO: This feature is still in progress**

If you want to generate a web-friendly local version of a resource, you
can use the `preview` command.

## Building Public Versions

**TODO: This feature is still in progress**

Frequently, you want to create public versions of assignments that are
viewable by folks who are not authorized to see the answers or other
components of a resource. We distinguish between *public* (viewable by
anyone), *private* (viewable by other teachers), and *secret* (viewable
only by local course staff).

You can use the `--hide_answers` flag to remove answers and other
private information from a resource, generating a new file in a parallel
repository.

Secret information usually includes data covered by FERPA, such as
students' grades or assignment submissions. Secret information is harder
to secure in a VCS, but can be done through encryption. In general,
however, data that is meant to be secret should be kept in a different
repository. If you want it alongside the data, though, perhaps we could
encrypt/decrypt the file, suspending git commands until all relevant
files are re-encrypted?

# Terminology

* Resource: A curricular material to be transferred around. Resources can be represented in different styles. Some resources can be composed of other resources.
* Category: Resources of the same type. The "Pages" category is distinct from the "Assignment" category. Categories do not belong to a particular Service, meaning their names must be meaningfully distinct if they are different kinds of things.
* Title: A name that, idealistically, is globally unique within a resource.
* Markdown: A plain-text format that we use to store files locally. Meant to be easily edited. Can have Front-matter.
* Front-matter: At the top of Markdown files, you can put "---" above and below YAML-encoded data to store extra data. We particularly look for a `waltz` attribute with additional Waltz information.

# Service Specific Information

This section details information specific to individual services.

## Local

Waltz is not opinionated about file system organization. You are free to
arrange your data into directories as you see fit. By default, new files
are stored in the current working directory. However, existing files
will be matched as best they can.

Largely, Waltz encourages resources to decode themselves as Markdown
files. In particular, Markdown files with front-matter can be parsed and
have special Waltz-specific information. The only requirement is to have
an attribute named `waltz`. We also strongly encourage a `title` and
`resource` attribute within.

```markdown
---
waltz:
  title: The Resource's Full Proper Name
  resource: Page
---
The body of the file
```

Additional attributes can be stored in the Markdown, and services
attempt to leave them unmangled when downloading updated versions.

## Canvas

Instructure's Canvas has a rich API that allows us to control resources
relatively well.

Code in Markdown files will be syntax-highlighted.

Uploading data into Canvas is somewhat limited. To get around the
limitations, we store additional front-matter as HTML content hidden
inside of an invisible DIV tag at the start of the resource's body. If
that information is deleted on the server, however, there isn't much to
be done to reclaim it. Be careful when deleting large swathes of data if
you are relying on this feature! Also note that we do not currently
encode any extra `waltz` front-matter, so you should be sure not to rely
on extra data stored in that attribute.

You can use the `--all` switch to bulk download resources.

You can use the `--destination <path>` parameter to set the location
where files will be decoded to.

### Canvas Pages

A Canvas Page is decoded into a Markdown file with very minimal
front-matter.

### Canvas Assignments

A Canvas Page is decoded into a Markdown file with very minimal
front-matter.

### Canvas Quizzes

By default, quizzes are pulled as multiple files. The core file will
hold the quiz itself, with the questions stored as a list of their
names. Those questions' content will be then be stored as separate files
in an adjacent folder named `"<Quiz Title> Questions/"`. All question
types are supported.

Groups are also supported, and will be represented inside of the core
Markdown file alongside other questions. However, due to a limitation of
the Canvas API, we are not able to keep the order of grouped questions -
further, we cannot guarantee that question groups will be properly
disposed of. When using quizzes, keep an eye on the Canvas side to make
sure that the content is appropriately transferred.

A Canvas Quiz can be pulled as a combined single file.

```console
foo@bar:~$ waltz pull canvas quiz "My Quiz" --combine 
```

You can specify a question bank to reuse questions between quizzes.

### Canvas Files

**TODO: This feature is still in progress**

You can have either a folder dedicated to being a file system, or
identify an individual file with an adjacent Markdown file.

### Linking Resources

**TODO: This feature is still in progress**

We attempt to decode URLs to images, files, and other resources as
intelligently as possible so that they make sense locally and remotely.

# Full Command List

```console
foo@bar:~$ waltz reset
foo@bar:~$ waltz init
foo@bar:~$ waltz configure <service> <name>
foo@bar:~$ waltz list <service> <category>
foo@bar:~$ waltz pull <service> <category> <title>
foo@bar:~$ waltz push <service> <category> <title>
foo@bar:~$ waltz download <service> <category> <title>
foo@bar:~$ waltz upload <service> <category> <title>
foo@bar:~$ waltz encode <service> <category> <title>
foo@bar:~$ waltz decode <service> <category> <title>
foo@bar:~$ waltz diff <service> <category> <title>
```



            

Raw data

            {
    "_id": null,
    "home_page": "https://github.com/acbart/waltz",
    "name": "lms-waltz",
    "maintainer": "",
    "docs_url": null,
    "requires_python": ">=3.6",
    "maintainer_email": "",
    "keywords": "lms learning management system curriculum curricular resources",
    "author": "Austin Cory Bart",
    "author_email": "acbart@udel.edu",
    "download_url": "https://files.pythonhosted.org/packages/eb/ef/c6db818b6a18b104517ace0ae2fb7a51a0790514f94e66841305cc041b33/lms-waltz-0.2.7.tar.gz",
    "platform": null,
    "description": "# waltz\n\nA software system for synchronizing curricular materials between a\nLearning Management Systems (LMS) and your local filesystem.\n\nMajor features include:\n* Pull and push data between an LMS and the file system\n* Lightweight, friendly Markdown-based format for local resources\n* Diffing of local and remote content\n* List and search for remote resources\n\nWe aim to support the following LMSes:\n* Canvas (pages, assignments, quizzes)\n* BlockPy (assignment, groups) <- In-progress\n\n# Installation\n\nYou can install Waltz from PyPi (the package name is `lms-waltz` even though the module and command line script is `waltz`):\n\n```console\nfoo@bar:~$ pip install lms-waltz\n```\n\nYou can also install our dev version from GitHub:\n\n```console\nfoo@bar:~$ pip install git+https://github.com/acbart/waltz.git\n```\n\n## Setup Waltz\n\nWaltz can synchronize content between a local directory and a remote server.\nYou'll need to initialize the local directory, whether it is empty or already has your learning materials.\n\n```console\nfoo@bar:~$ waltz init\n```\n\nThis will create two new files which should both be included in your `.gitignore`:\n\n* `.waltz` is a plain-text YAML file with settings for the current course repository.\n* `.waltz.db` is a SQLite database used in file uploading/downloading.\n\n## Setup a Service\n\nBefore you can start interacting with an LMS, you'll need to configure an\ninstance of a service. For example, to configure a new Canvas service, you'll need the following:\n\n1. A short, easy-to-type name for the service instance (e.g., `ud_canvas` or `cs1014_canvas`)\n2. [API token](https://community.canvaslms.com/docs/DOC-10806-4214724194)\n3. The URL base for your Canvas site (e.g., `https://canvas.instructure.com/`)\n4. The Course ID that you want to connect to (usually a large number)\n\nThen, you can use the command below:\n\n```console\nfoo@bar:~$ waltz configure canvas <name> --base <url> --course <id> --token <token>\n```\n\nA more concrete example:\n\n```console\nfoo@bar:~$ waltz configure canvas ud_canvas --base https://udel.instructure.com/ --course 4234343432343 --token 1081~zJIOJSDFBBSDFIJAIJ==>...\n```\n\nIf things went well, you can list the available services:\n\n```console\nfoo@bar:~$ waltz list\nThe following services are available:\n         local:\n                 ./\n         canvas:\n                 ud_canvas\n         blockpy: (none configured)\n```\n\nBy default, there's a Local service for the current directory (representing the connection to the filesystem).\nYou can have more than one instance of a service, which can allow you to\naccess multiple data sources from one course (e.g., to transfer resources between semesters).\nFor convenience, you can refer to the first instance of a service by the service's name.\nIn our case, anywhere that we use `ud_canvas` we could use `canvas` instead.\n\n## Identifying Resources\n\nYou can list available resources for a service:\n\n```console\nfoo@bar:~$ waltz list <service> <category>\n```\n\nSo the following code checks the Pages for the Canvas course. Note that\nyou could use the specific instance name instead of `canvas`, and either\nthe singular or plural form of `pages`.\n\n```console\nfoo@bar:~$ waltz list canvas pages\nRemote    Local    Title                      Path\n--------  -------  -------------------------  --------------------\nYes       Yes      Abstraction                pages\\Abstraction.md\nYes       Yes      NetLogo-2                  NetLogo-2.md\nYes       No       Overview of Functions\nYes       No       Syllabus\nYes       No       Themes\nYes       Yes      Turtles                    pages\\Turtles.md\n```\n\nThe `local` service allows you to omit the category (but you can filter by category if you want):\n\n```console\nfoo@bar:~$ waltz list local\n\nResource         Title                        Path\n---------------  ---------------------------  ---------------------------------------------------------\n[unknown]        NetLogo-2                    NetLogo-2.md\n[unknown]                                     pages\\Abstraction.md\n[Page]           Turtles                      pages\\Turtles.md\n[quiz]           Backup Quiz                  quizzes\\Backup Quiz.md\n[unknown]                                     quizzes\\Mangled Resource.md\n[quiz]           New Quiz                     quizzes\\New Quiz.md\n[assignment]     Video Upload Test            quizzes\\Video Upload Test.md\n[quiz question]  Complex Matching Question    quizzes\\New Quiz Questions\\Complex Matching Question.md\n[quiz question]  Essay Question               quizzes\\New Quiz Questions\\Essay Question.md\n```\n\nThe local service only knows the resource category and title if that information\nis available in the front-matter of the Markdown file. Otherwise, it can\nonly report the path of potential resource files.\n\nSimilar to Git, you can use `waltz` commands from child folders and the\nsystem will search up for the configuration and use it appropriately.\n\n## Managing Resources\n\nThe two basic commands for moving resources are `pull` and `push`, moving resources between\nthe local service and a remote one. If the local version does not already exist,\nit will be created in the current directory with a filename based on the title of\nthe resource from the LMS.\n\n```console\nfoo@bar:~$ waltz pull <service> <category> <title>\nfoo@bar:~$ waltz push <service> <category> <title>\n```\n\nGenerally, you want to fully specify the remote service, resource category,\nand title of the resource you want to use. This avoids ambiguity.\n\n```\nfoo@bar:~$ waltz pull canvas assignment \"Project 3\"\nfoo@bar:~$ ls\nProject 3.md\nfoo@bar:~$ waltz push canvas assignment \"Project 3\"\n```\n\nHowever, the commands attempt to work intelligently to find the appropriate file given minimal amounts\nof information, particularly for pushing. If the given title matches the title of a file\nin the current directory (either based on the filename or the front-matter of the file),\nand that file has a resource specified, then the appropriate service can automatically be inferred.\n\n```console\nfoo@bar:~$ waltz push \"Turtles\"\n```\n\nIf the desired files are not in the present folder, it will search subfolders.\n\nTOOD: We are working on ways to further disambiguate resources that have\nthe same name. This will be accomplished using their IDs or some other attribute.\nIn general, however, you can avoid ambiguity by organizing the file system\nand your LMS resources appropriately.\n\n## Diffing Resources\n\nPushing and pulling resources can get quite complex, so it can be\nhelpful to find out how the local version of a resource is different\nfrom the remote. The `diff` command has similar syntax to the other\nactions, but does not modify the filesystem when it is run.\n\n```console\nfoo@bar:~$ waltz diff canvas page \"Turtles\"\n```\n\n![Screenshot showing web-based output of the Waltz Diff command, with the local version of a file on the left and the remote version on the right. Changes are highlighted using red, green, and yellow highlighting.](docs/example-diff-1.png)\n\nBy default, `diff`ing will create a new HTML file adjacent to the\nresource and automatically open that file in the browser. There is also\nsimplified console output available.\n\n## Download, Decode, Encode, and Upload\n\n`Pull`ing is actually a two-step process: first you `download` a resource, and then you `decode` it.\nSimilarly, `Push`ing requires you to `encode` a resource and then `upload` it.\n\nThe `upload` and `download` commands (which are available in Waltz) move data between Canvas and the\nlocal Waltz SQLite database. Generally, we download raw data from the remote service (e.g., a JSON blob)\nand upload using whatever API is available for the service.\n\nThe `encode` and `decode` commands (also available) move data between the local filesystem and the\nlocal Waltz SQLite database. Generally, this means we are decoding the raw data into a Markdown file\nor potentially even multiple files.\n\n## Templates\n\n**TODO: This feature is still in progress**\n\nThe HTML body of a Resource can be built using a Template. We use the\nJinja2 format to embed data.\n\n## Previewing Resources\n\n**TODO: This feature is still in progress**\n\nIf you want to generate a web-friendly local version of a resource, you\ncan use the `preview` command.\n\n## Building Public Versions\n\n**TODO: This feature is still in progress**\n\nFrequently, you want to create public versions of assignments that are\nviewable by folks who are not authorized to see the answers or other\ncomponents of a resource. We distinguish between *public* (viewable by\nanyone), *private* (viewable by other teachers), and *secret* (viewable\nonly by local course staff).\n\nYou can use the `--hide_answers` flag to remove answers and other\nprivate information from a resource, generating a new file in a parallel\nrepository.\n\nSecret information usually includes data covered by FERPA, such as\nstudents' grades or assignment submissions. Secret information is harder\nto secure in a VCS, but can be done through encryption. In general,\nhowever, data that is meant to be secret should be kept in a different\nrepository. If you want it alongside the data, though, perhaps we could\nencrypt/decrypt the file, suspending git commands until all relevant\nfiles are re-encrypted?\n\n# Terminology\n\n* Resource: A curricular material to be transferred around. Resources can be represented in different styles. Some resources can be composed of other resources.\n* Category: Resources of the same type. The \"Pages\" category is distinct from the \"Assignment\" category. Categories do not belong to a particular Service, meaning their names must be meaningfully distinct if they are different kinds of things.\n* Title: A name that, idealistically, is globally unique within a resource.\n* Markdown: A plain-text format that we use to store files locally. Meant to be easily edited. Can have Front-matter.\n* Front-matter: At the top of Markdown files, you can put \"---\" above and below YAML-encoded data to store extra data. We particularly look for a `waltz` attribute with additional Waltz information.\n\n# Service Specific Information\n\nThis section details information specific to individual services.\n\n## Local\n\nWaltz is not opinionated about file system organization. You are free to\narrange your data into directories as you see fit. By default, new files\nare stored in the current working directory. However, existing files\nwill be matched as best they can.\n\nLargely, Waltz encourages resources to decode themselves as Markdown\nfiles. In particular, Markdown files with front-matter can be parsed and\nhave special Waltz-specific information. The only requirement is to have\nan attribute named `waltz`. We also strongly encourage a `title` and\n`resource` attribute within.\n\n```markdown\n---\nwaltz:\n  title: The Resource's Full Proper Name\n  resource: Page\n---\nThe body of the file\n```\n\nAdditional attributes can be stored in the Markdown, and services\nattempt to leave them unmangled when downloading updated versions.\n\n## Canvas\n\nInstructure's Canvas has a rich API that allows us to control resources\nrelatively well.\n\nCode in Markdown files will be syntax-highlighted.\n\nUploading data into Canvas is somewhat limited. To get around the\nlimitations, we store additional front-matter as HTML content hidden\ninside of an invisible DIV tag at the start of the resource's body. If\nthat information is deleted on the server, however, there isn't much to\nbe done to reclaim it. Be careful when deleting large swathes of data if\nyou are relying on this feature! Also note that we do not currently\nencode any extra `waltz` front-matter, so you should be sure not to rely\non extra data stored in that attribute.\n\nYou can use the `--all` switch to bulk download resources.\n\nYou can use the `--destination <path>` parameter to set the location\nwhere files will be decoded to.\n\n### Canvas Pages\n\nA Canvas Page is decoded into a Markdown file with very minimal\nfront-matter.\n\n### Canvas Assignments\n\nA Canvas Page is decoded into a Markdown file with very minimal\nfront-matter.\n\n### Canvas Quizzes\n\nBy default, quizzes are pulled as multiple files. The core file will\nhold the quiz itself, with the questions stored as a list of their\nnames. Those questions' content will be then be stored as separate files\nin an adjacent folder named `\"<Quiz Title> Questions/\"`. All question\ntypes are supported.\n\nGroups are also supported, and will be represented inside of the core\nMarkdown file alongside other questions. However, due to a limitation of\nthe Canvas API, we are not able to keep the order of grouped questions -\nfurther, we cannot guarantee that question groups will be properly\ndisposed of. When using quizzes, keep an eye on the Canvas side to make\nsure that the content is appropriately transferred.\n\nA Canvas Quiz can be pulled as a combined single file.\n\n```console\nfoo@bar:~$ waltz pull canvas quiz \"My Quiz\" --combine \n```\n\nYou can specify a question bank to reuse questions between quizzes.\n\n### Canvas Files\n\n**TODO: This feature is still in progress**\n\nYou can have either a folder dedicated to being a file system, or\nidentify an individual file with an adjacent Markdown file.\n\n### Linking Resources\n\n**TODO: This feature is still in progress**\n\nWe attempt to decode URLs to images, files, and other resources as\nintelligently as possible so that they make sense locally and remotely.\n\n# Full Command List\n\n```console\nfoo@bar:~$ waltz reset\nfoo@bar:~$ waltz init\nfoo@bar:~$ waltz configure <service> <name>\nfoo@bar:~$ waltz list <service> <category>\nfoo@bar:~$ waltz pull <service> <category> <title>\nfoo@bar:~$ waltz push <service> <category> <title>\nfoo@bar:~$ waltz download <service> <category> <title>\nfoo@bar:~$ waltz upload <service> <category> <title>\nfoo@bar:~$ waltz encode <service> <category> <title>\nfoo@bar:~$ waltz decode <service> <category> <title>\nfoo@bar:~$ waltz diff <service> <category> <title>\n```\n\n\n",
    "bugtrack_url": null,
    "license": "",
    "summary": "Coordinate resources between an LMS like Canvas and a local directory",
    "version": "0.2.7",
    "split_keywords": [
        "lms",
        "learning",
        "management",
        "system",
        "curriculum",
        "curricular",
        "resources"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "md5": "f0634358221b9db2bdab32a5b37d77a0",
                "sha256": "c499792cbf47e7bcc504fd64ae38e10c13468aaa81afedda002c551035229d0c"
            },
            "downloads": -1,
            "filename": "lms_waltz-0.2.7-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "f0634358221b9db2bdab32a5b37d77a0",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.6",
            "size": 101683,
            "upload_time": "2023-01-02T16:55:58",
            "upload_time_iso_8601": "2023-01-02T16:55:58.057405Z",
            "url": "https://files.pythonhosted.org/packages/af/a0/661d6f3f394a42f9bfe89430cfd54ebe6a272895388c52f127cc41a039b2/lms_waltz-0.2.7-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "md5": "9b5171841acf43435069d69d16540a6c",
                "sha256": "6286af81e0417487799b972af9b770f94a2e51dc80944e4673208070cad13c28"
            },
            "downloads": -1,
            "filename": "lms-waltz-0.2.7.tar.gz",
            "has_sig": false,
            "md5_digest": "9b5171841acf43435069d69d16540a6c",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.6",
            "size": 73161,
            "upload_time": "2023-01-02T16:55:59",
            "upload_time_iso_8601": "2023-01-02T16:55:59.703825Z",
            "url": "https://files.pythonhosted.org/packages/eb/ef/c6db818b6a18b104517ace0ae2fb7a51a0790514f94e66841305cc041b33/lms-waltz-0.2.7.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2023-01-02 16:55:59",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "github_user": "acbart",
    "github_project": "waltz",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": false,
    "requirements": [
        {
            "name": "tqdm",
            "specs": []
        },
        {
            "name": "requests",
            "specs": []
        },
        {
            "name": "ruamel.yaml",
            "specs": []
        },
        {
            "name": "jinja2",
            "specs": []
        },
        {
            "name": "markdown",
            "specs": []
        },
        {
            "name": "python-frontmatter",
            "specs": []
        },
        {
            "name": "requests_cache",
            "specs": []
        },
        {
            "name": "python-dateutil",
            "specs": []
        },
        {
            "name": "html2text",
            "specs": []
        },
        {
            "name": "canvasapi",
            "specs": []
        },
        {
            "name": "tabulate",
            "specs": []
        },
        {
            "name": "natsort",
            "specs": []
        },
        {
            "name": "pygments",
            "specs": []
        },
        {
            "name": "beautifulsoup4",
            "specs": []
        }
    ],
    "lcname": "lms-waltz"
}
        
Elapsed time: 0.02398s