Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feature/summer2024 #34

Merged
merged 61 commits into from
Sep 2, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
61 commits
Select commit Hold shift + click to select a range
f462b4c
re-factored and rewrote readme
ofloveandhate Jul 11, 2024
8c43d5c
added pygments
ofloveandhate Jul 11, 2024
d86e271
added with docs-folder
ofloveandhate Jul 11, 2024
e4fc688
typo 2
ofloveandhate Jul 11, 2024
93c93d6
messing with the path to try to get import to work in documentation g…
ofloveandhate Jul 11, 2024
fd1a764
changed to unspecific versions.
ofloveandhate Jul 12, 2024
7588070
v4
ofloveandhate Jul 12, 2024
4d0b49a
working on paths
ofloveandhate Jul 12, 2024
63a4cad
using python 3.10
ofloveandhate Jul 12, 2024
f1551c2
removed line
ofloveandhate Jul 12, 2024
29af091
documented the Tool base class
ofloveandhate Jul 12, 2024
93a3152
improving granularity of documentation
ofloveandhate Jul 12, 2024
d184106
a step towards refactoring into submodules
ofloveandhate Jul 12, 2024
9b85ee9
refactor completed?
ofloveandhate Jul 15, 2024
9b777fd
change from even_if_exists to even_if_doesnt_exist
anbeemer Jul 16, 2024
32be458
Unit tests added to test_replacements
anbeemer Jul 16, 2024
268ce4b
Merge branch 'feature/summer2024' of https://github.com/ofloveandhate…
anbeemer Jul 16, 2024
eda9608
Added note about replacement order
anbeemer Jul 16, 2024
3e6e012
Added custom style page to check
anbeemer Jul 16, 2024
1bce33f
Added unit tests to test_style.py
anbeemer Jul 16, 2024
8114200
added tutorial on uploading files
anbeemer Jul 17, 2024
e0ffa03
Added unit tests to test_file.py
anbeemer Jul 17, 2024
d2a1eeb
rst learnin'
anbeemer Jul 17, 2024
c85a7f3
Added example to uploading_files.rst
anbeemer Jul 17, 2024
87ec50c
Fix to test_file.py
anbeemer Jul 17, 2024
ccb72ae
Removed some comments in test_replacements
anbeemer Jul 17, 2024
facc582
Update to uploading_files.rst
anbeemer Jul 17, 2024
357dbd6
Added unit test to test_page.py
anbeemer Jul 17, 2024
50b249f
more clarity in docstrings
ofloveandhate Jul 19, 2024
d4967ae
clearer imports, and __all__
ofloveandhate Jul 19, 2024
1d1e658
Merge branch 'feature/summer2024' of https://github.com/ofloveandhate…
ofloveandhate Jul 19, 2024
d43781e
style order update
anbeemer Jul 19, 2024
532888b
ignore extra_styled_source.md
anbeemer Jul 19, 2024
a8e29c2
even_if_exists became .._doesnt_exist
anbeemer Jul 19, 2024
8d9b4e3
Added missing namespace
anbeemer Jul 19, 2024
d14f5b7
test_style improvements
anbeemer Jul 19, 2024
a43604d
Merge branch 'feature/summer2024' of https://github.com/ofloveandhate…
anbeemer Jul 19, 2024
525cb9c
added `default_` prefix
ofloveandhate Jul 19, 2024
3601bcd
improved module docstring
ofloveandhate Jul 19, 2024
6ba35a2
improved imports to declutter base namespace
ofloveandhate Jul 19, 2024
0dd5ddd
Unit test added for markdown to html conversion
anbeemer Jul 22, 2024
5025f1d
Unit test added for markdown to html conversion
anbeemer Jul 22, 2024
8609580
added missing namespaces
ofloveandhate Jul 22, 2024
52c9b0b
added markdown="1"
ofloveandhate Jul 22, 2024
39bb759
Added info on meta.json file in uploading_files doc
anbeemer Jul 24, 2024
7eb0184
Namespace updates to unit tests due to refactor
anbeemer Jul 24, 2024
fc13081
Added markdown=1 for correct render in unit tests
anbeemer Jul 24, 2024
a1399d6
Removed date updated test
anbeemer Jul 24, 2024
ee6301f
Docs related to markdown="1"
anbeemer Jul 24, 2024
5f32df1
factored out the markdown to html extensions list
ofloveandhate Jul 24, 2024
6ecda24
working towards distributable setup
ofloveandhate Jul 24, 2024
32c6673
Merge branch 'feature/summer2024' of https://github.com/ofloveandhate…
ofloveandhate Jul 24, 2024
7aca334
tutorial on markdown extensions
ofloveandhate Jul 24, 2024
351d622
new documentation pages, and improvements in linkage
ofloveandhate Jul 24, 2024
871ca83
factored out list of steps for publication of Documents
ofloveandhate Jul 24, 2024
654e869
link is now actually a link.
ofloveandhate Jul 24, 2024
eff7da0
resolved duplicate section names, fixed broken references
ofloveandhate Jul 24, 2024
7825cc1
code as code, not strings
ofloveandhate Jul 24, 2024
0d86e32
more thorough notes on `meta.json`.
ofloveandhate Jul 24, 2024
ec5cebf
more clear headings, a slight re-structure
ofloveandhate Jul 24, 2024
57cb242
Reorganized file upload test files
anbeemer Aug 12, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
15 changes: 12 additions & 3 deletions .github/workflows/documentation.yml
Original file line number Diff line number Diff line change
Expand Up @@ -5,23 +5,32 @@ on: push
permissions:
contents: write



jobs:
build:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
- uses: actions/checkout@master

- name: Set up Python 3.10
uses: actions/setup-python@v2
with:
python-version: '3.10'

- name: Build HTML
uses: ammaraskar/sphinx-action@master
with:
docs-folder: "docs"

- name: Upload artifacts
uses: actions/upload-artifact@v3
uses: actions/upload-artifact@master
with:
name: html-docs
path: docs/_build/html/

- name: Deploy
uses: peaceiris/actions-gh-pages@v3
uses: peaceiris/actions-gh-pages@v4
if: github.ref == 'refs/heads/main'
with:
github_token: ${{ secrets.GITHUB_TOKEN }}
Expand Down
4 changes: 3 additions & 1 deletion .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,9 @@ build/
.DS_Store
*/*/result.html
styled_source.md
extra_styled_source.md
*.icloud

_build/
make.bat
make.bat

185 changes: 30 additions & 155 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,13 +1,17 @@
A library for containerizing local markdown content to be published into the Canvas learning management system.

:warning: This library is under active development and the interface is NOT stable.
[Documentation on Github Pages](https://ofloveandhate.github.io/markdown2canvas/)

---

# Why I wrote/maintain this library

The particular problem this library solves is that of putting Canvas content under version control, and also using Markdown for that content. Canvas pages are not well-suited to version control *per se*, because they live on the LMS. I wanted local files, with repos I can share with other designers and instrutors.

Further, I want to be able to find-and-replace across many pieces of Canvas content at once. Local files with my editor of choice is the way to do that; it's impossible on Canvas. Hence, this library.

Additionally, uniform appearance and ability to change much with little effort. I wrote a "style" system that puts headers and footers around my content, eliminating repetitive and error-prone work. Want emester-specific text at the top of all your content? Trivial with `markdown2canvas`: just change the header file, re-publish, and do something better with your time than wait for Canvas pages to load.

A secondary problem this library solves is that of images. Images on Canvas are bare, and it's easy to end up with duplicate versions, as well as not have alt text. By using markdown/html under version control, I can write my alt text directly into page source, instead of using the crappy click-heavy interface on Canvas.

---
Expand All @@ -20,138 +24,26 @@ Containerization is accomplished by making pages/assignments live in folders. T
- `meta.json` -- a json file containing attributes. keys should be compliant with the expectations of `canvasapi`.
- `source.md` -- a markdown file. Can contain latex math, html, references to local images, emoji using double-colon notation and shortcodes, and of course online images.



---

# Testing

This library comes with several test pages/assignments:
- pages:
- `plain_text` -- source is just plain old text. start simple, ya know?
- `uses_latex` -- a page that contains latex math
- `has_remote_images` -- a page that has remote images embedded
- `has_local_images` -- a page that uses local images
- `uses_droplets` -- a page using [the Droplets framework from UWEX](https://media.uwex.edu/app/droplets/index.html).
- `uses_droplets_via_style` -- a page using [the Droplets framework from UWEX](https://media.uwex.edu/app/droplets/index.html). The code enabling Droplets comes from a header/footer contained in a style folder. The main purpose of this test page is the header/footer style thing.
- assignments:
- `programming_assignment` -- an assignment that has a local image

The above list is not exhaustive.
You can also put needed files, images, etc in the folder for a piece of content. `markdown2canvas` aims to automate as much of the process as possible.

---

# Installation

1. Clone the repo / pull from the repo
2. Move to repo location in terminal
3. `pip install .` If you already had it installed, then use `pip install . --upgrade` to make sure you get the newer version.

## Critical setup step, do not skip this

You must also define an environment variable called `CANVAS_CREDENTIAL_FILE`, which is the location of a `.py` file containing two variables:
1. `API_URL` -- a string, the url of how to access your Canvas install.
- At UW Eau Claire, it's `https://uweau.instructure.com/`.
- I cannot possibly tell you your url, but your local Canvas admin can.
2. `API_KEY` -- a string, the key you can get from Canvas. Here's [a link to a guide on how to generate yours](https://community.canvaslms.com/t5/Admin-Guide/How-do-I-obtain-an-API-access-token-in-the-Canvas-Data-Portal/ta-p/157). Do not share it with anyone -- having only this one piece of data, anyone can act as you. Protect it at least as much as you would any other password or sensitive information.


# Alternate-ish Installation for Windows Users

These instructions are tested on Windows 11 on February 26, 2024.

## Get Canvas Credentials and Make Canvas Credential File - this is the same step as above.

Your first step will need to be to get a Canvas API Key.

1. On Canvas, navigate Accounts -> Settings
2. Scroll to the button labeled `+ New Acces Token`
3. Add a description for yourself to know, later, what the access token is for and optionally add an expiration date. (I like to make a new one every semester, for safety.)
4. Copy the text of the token (you won't get to see this again) to a file that we will name `canvas_credential_file.py`.
5. Create a variable in `canvas_credential_file.py` named `API_KEY`, whose value is the string that we just copied from canvas.

Additionally, add a second variable `API_URL` whose value is the string that is the general Canvas URL you use. For UWEC, this is `'https://uweau.instructure.com/'`.

Ultimately, your `canvas_credential_file.py` will contain the lines:
Please see [the documentation](https://ofloveandhate.github.io/markdown2canvas/) -- we have two tutorials, one for Mac/Linux and one for Windows.

```
API_KEY = "stringofrandomcharacters"
API_URL = "https://uweau.instructure.com/"
```

## Initial Setup - Using VS Code

1. Install python via the Microsoft Store
2. Install VS code and GitBash - at UW Eau Claire this is done via the software center, you might use the Microsoft Store for this step as well.
3. Clone the markdown2canvas repo from github
4. Open VS code, open GitBash terminal and run the command

```
pip install /path/to/markdown2canvas
````

Then also run the command

```
pip install lxml beautifulsoup4
```

Note that the default terminal that VSCode opens will be the Windows powershell, don't use that.

## Generate necessary global variables

1. Run the following command to make a file called `.bashrc` and save the location of your canvas credential file in your home directory.

```
echo 'CANVAS_CREDENTIAL_FILE=h:\\path\\to\\canvas_credential_file.py' >> ~/.bashrc
```

Note that you should be using `\\` here as directory separators because you are using Windows. If you use `/` you run the risk of the operating system not understanding the path.

2. Open a new git bash terminal and see if the following works:

```
echo $CANVAS_CREDENTIAL_FILE
```

if not, you might need to run the following command in your bash terminal:

```
source ~/.bashrc
```

---

# Some quick examples

This library is under active development. I suggest checking out the `test_*.py` files in the `test` folder for example code.

Assuming you did my setup step, defining the environment variable and creating that file. Do that first.

### Download all pages, with a filter on the name of the pages

```
import markdown2canvas as mc
course_id = 127210000000003099 # silviana's sandbox for development

canvas = mc.make_canvas_api_obj() # gets link and api key via environment variable
course = canvas.get_course(course_id)

destination = 'downloaded_pages'
# Some things you can do with this library

my_filter = lambda title: '📖' in title # pages about readings have an open book in their names.
mc.download_pages(destination, course, even_if_exists=True, name_filter=my_filter)
```

---

# Things you can do with this library
See the [the documentation](https://ofloveandhate.github.io/markdown2canvas/). This is just highlights in a root readme.

## Replacements during translation

The purpose of this library is to increase modularity and flexibility, while reducing duplication in source code and allowing version control. I implemented a simple text replacement feature as part of this, so that I can create uniform appearances in my content without duplicate code.

That is, you can specify a set of string replacements using a .json file, and during translation from markdown to html, before uploading, each substitution happens.
That is, you can specify a set of string replacements using a `.json` file, and during translation from markdown to html, before uploading, each substitution happens.

For example, you can create a `replacements.json` file in a folder at root level (relative to the folder for the course) called `_course_metadata`, and in this file put the content:

Expand All @@ -164,13 +56,6 @@ For example, you can create a `replacements.json` file in a folder at root level
}
```

I'm using simple python `.replace` to do the replacements. There are consequences:
* It will replace strings exactly, there are no implemented efforts to allow patterns or functions.
* It's case sensitive, and includes exact spacing.
* The dollar signs above are NOT special. They're just a nice way to indicate the text will be replaced.

⚠️ Furthermore, I'm not sure what order the replacements will be done in, so if the target of one replacement includes the source of another, I can't guarantee you at this time that it will actually happen in a deterministic order. If you want this feature, please add it and submit a PR to this repo.

You can specify a default set of replacements to happen for every file (except those with overridden replacements). To do this, make a file `_course_metadata/defaults.json`, and create a record `"replacements": "relative/path/to/replacements_filename.json"`. The name of the replacements file is arbitrary, and it's relative to root of the course folder.

To override the default replacements, put a record in the `meta.json` file for the content (page / assignment) of the form `"replacements": "relative/path/to/replacements_filename.json"`.
Expand All @@ -180,8 +65,19 @@ Examples of content using replacements can be found in the `test/` folder of thi
If a replacements file doesn't exist where you say it should, an exception will be raised at `publish` time for the `CanvasObject` (`Page` or `Assignment`). (You can construct a thing with a bad replacements file and not know it until you try to publish!)


## Referencing existing Canvas assignments, pages, and files
ℹ️ I'm using simple Python `str.replace` to do the replacements. There are consequences:

* It will replace strings exactly, there are no implemented efforts to allow patterns or functions.
* It's case sensitive, and includes exact spacing.
* The dollar signs above are NOT special. They're just a nice way to indicate the text will be replaced.

⚠️ Furthermore, I'm not sure what order the replacements will be done in, so if the target of one replacement includes the source of another, I can't guarantee you at this time that it will actually happen in a deterministic order. If you want this feature, please add it and submit a PR to this repo.



## Reference existing Canvas assignments, pages, and files

Whereas I find it to be a pain to link to other content on Canvas using their editor, it's easy using `markdown2canvas`.

To link to an existing Canvas assignment, use a link of the form

Expand All @@ -197,17 +93,22 @@ To link to an existing Canvas file, use a link of the form
<a href="file:DavidenkoDiffEqn.pdf">Link to file called DavidenkoDiffEqn.pdf</a>
```

If the "existing" content doesn't yet exist when the content is published, a broken link will be made. This is ok. Think of the publishing process using Markdown2Canvas similar to the compilation of a TeX document, which is done in multiple passes. Once the page / assignment / file exists, the link will resolve correctly to it. Publish all content about twice to get links to resolve.
ℹ️ If the "existing" content doesn't yet exist when the content is published, a broken link will be made. This is ok. Think of the publishing process using Markdown2Canvas similar to the compilation of a TeX document, which is done in multiple passes. Once the page / assignment / file exists, the link will resolve correctly to it. Publish all content about twice to get links to resolve.


## Emoji conversion from shortcodes

This library supports the automatic conversion of shortcodes to emoji. For example, `:open_book:` goes to 📖. I use the [`emoji`](https://pypi.org/project/emoji/) library to do this. [Shortcodes can be found here](https://carpedm20.github.io/emoji/).

Right now, emoji shortcodes can only be used in content, not in names of things -- shortcodes in names will not be emojized.



## Automatic uploading and warehousing of images and embedded content

List your images relative the folder containing the `source.md` for the content.
List your images relative the folder containing the `source.md` for the content and they'll automatically be uploaded when you publish. If the image is already uploaded, a link to the existing image will be generated instead of uploading.



## "Styling" -- Automatic inclusion of uniform headers and footers

Expand All @@ -216,37 +117,11 @@ This library attempts to provide a way to uniformly style pages across sections
* I have content in my course in four blocks, and want a different header for each block. But, copypasta for that header content sucks (avoid repitition is a key tenet of programming). So, I'd rather specify a "style" for the four blocks, and make the pages refer to the styles.
* I use Droplets from UWEX, and don't want to have to put that code in *every single page*. I'd rather put it one place (or, at least, only a few places). So the html code that brings in Droplets lives in a header/footer html code file.

### Style basics

Put your "style" folders in a folder in your course. In my DS150 course, I have the following structure:

* `_styles/`
* `/generic.style`
* `/assignments.style`

And in the `meta.json` file for the pages / assignments, I simply have to put the record `"style":"_styles/generic.style"` or whatever.

As of July 2022, there is no default style -- if a page doesn't list a style, it gets no style.

### Additional notes about styles:

The folder for each style should have the following four files:
* `header.html`
* `header.md`
* `footer.md`
* `footer.html`

They'll get concatenated around `source.md` in that order. HTML around markdown, and header/footer around source.

If you want to use images in your header/footer, put them in the markdown part (even if they appear in html tags), and use the text `$PATHTOMD2CANVASSTYLEFILE` before typing the name of the file, so that its filepath gets listed correctly. (This happens via a simple string replacement)



## Assignments

### Possible Upload Types

In the `meta.json` file for an assignment, the submission type is encoded by a line that looks like the following.
Reduce your mental load by specify possible upload types in the `meta.json` file for an assignment. The submission type is encoded by a line that looks like the following.

```
"submission_types":['online_text_entry', 'online_url', 'media_recording', 'online_upload']
Expand Down
50 changes: 50 additions & 0 deletions docs/canvas_objects.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
Concrete Classes for Canvas Objects
--------------------------------------



.. autoclass:: markdown2canvas.Page
:members:
:undoc-members:


.. autoclass:: markdown2canvas.Assignment
:members:
:undoc-members:

.. autoclass:: markdown2canvas.Image
:members:
:undoc-members:


.. autoclass:: markdown2canvas.Link
:members:
:undoc-members:


.. autoclass:: markdown2canvas.File
:members:
:undoc-members:


.. autoclass:: markdown2canvas.BareFile
:members:
:undoc-members:







Base Classes
--------------

.. autoclass:: markdown2canvas.canvas_objects.CanvasObject
:members:
:undoc-members:


.. autoclass:: markdown2canvas.canvas_objects.Document
:members:
:undoc-members:
19 changes: 16 additions & 3 deletions docs/conf.py
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@
# https://www.sphinx-doc.org/en/master/usage/configuration.html#general-configuration

extensions = [
'sphinx.ext.autodoc',
'sphinx.ext.autodoc','sphinx.ext.autosectionlabel'
]

templates_path = ['_templates']
Expand All @@ -28,7 +28,20 @@
html_theme = 'bizstyle'
html_static_path = ['_static']


import os
import sys
sys.path.insert(0, os.path.abspath('../markdown2canvas/'))

_HERE = os.path.dirname(__file__)
_ROOT_DIR = os.path.abspath(os.path.join(_HERE, '..'))
_PACKAGE_DIR = os.path.abspath(os.path.join(_HERE, '../markdown2canvas'))

sys.path.insert(0, _ROOT_DIR)
sys.path.insert(0, _PACKAGE_DIR)

# test the path; not strictly needed
import markdown2canvas


rst_prolog = """
.. |markdowndefaults| replace:: :attr:`markdown2canvas.translation_functions.default_markdown_extensions`
"""
Loading
Loading