Python package and Jupyter extension that enables submitting Medium drafts elegantly from a Jupyter Notebook
Project description
nb2medium
Python package and Jupyter extension that enables submitting Medium drafts elegantly from a Jupyter Notebook
nb2medium main assets are:
- Written using
nbdev
and therefore easy to maintain- Simple pythonic implementation of Medium API
- Makes use of
nbconvert
and custom preprocessors to turn notebooks to Markdown documents- Supports uploading blocks as GitHub gists with simple in-cell tags
- Supports hiding of cells sources, cells outputs or cells
- Comes with Jupyter extension and CLI for ease-of-use
nb2medium represents a simple yet sufficient framework to upload Jupyter notebook to Medium. Its main strenghts are that it makes use of great existing tools such as nbconvert
or the requests
package for its main functionality. Moreover the package is developed using nbdev
from Jeremy Howard and the fastai team, which is claimed to accelerate development an debugging time.
Install
pip install nb2medium
Then enable the notebook extension by running:
jupyter notebook install nb2medium --py
jupyter notebook enable nb2medium --py
Add --user
to these commands if you want to activate the extension only for the current user.
Add --sys-prefix
to these commands if you want to activate the extension only in current virtual environment.
Setup
You will need a Medium Integration token to be able to upload your articles to Medium. If you wish to upload some code blocks as gists, you will need a GitHub token too.
Both token should be available as environment variables, hence we recommend you add these 2 lines to your shell configuration file (~/.bashrc
or ~/.zshrc
are the most common ones):
export MEDIUM_TOKEN=<your-medium-token>
export GITHUB_TOKEN=<your-github-token>
Obtain an Medium Integration Token from your Medium settings page
Obtain a GitHub token by following GitHub's docs
How to use
You may choose to use nb2medium as a command line tool, directly from python or using the Jupyter notebook extension button or menu. Note: nb2medium uploads Jupyter notebooks as they are, the notebooks do not get executed before being rendered.
From the Jupyter notebook
You may choose to use the nb2medium
under file or the button on the top toolbar
From the CLI
From the shell (bash, zsh, sh):
nb2medium "My article" path/to/notebook.ipynb
Use nb2medium --help
to see all the different options
From python
from nb2medium.upload import nb2medium
nb2medium(title = 'My First Article', notebook = '../samples/test-notebook.ipynb');
converter:INFO - Found a hide-source tag in cell #35.
converter:INFO - Found a hide-output tag in cell #36.
converter:INFO - Found a hide-cell tag in cell #37.
converter:INFO - Gist notebooktest.py from cell 26 succesfully uploaded!
converter:INFO - Gist print.py from cell 27 succesfully uploaded!
converter:INFO - Gist pandas.py from cell 30 succesfully uploaded!
converter:INFO - Gist pandas.py.csv from cell 30 succesfully uploaded!
converter:INFO - Gist pandas-doubleupload.py from cell 33 succesfully uploaded!
converter:INFO - Gist pandas-doubleupload.py.csv from cell 33 succesfully uploaded!
converter:INFO - Detected 4 plots and 2 local images in notebook.
converter:INFO - Markdown document written to ../samples/test-notebook/test-notebook.md
uploader:INFO - Draft of 'My First Article' from test-notebook.ipynb notebook uploaded to Medium: https://medium.com/@lucha6/946f9176365b
Images, code cells and tables
Images
Images usually come from a local file, online site or are the result of a plot. nb2medium
can handle these 3 situations. If an image correspond to a local file in markdown cell (e.g. ![](path/to/image.png)
)
The image is uploaded to the Medium end point and the path to the image is swapped for the newly generated image URL. If the image is the result of a plot, such as:
import numpy as np
import matplotlib.pyplot as plt
x = np.arange(0, 10, 0.1)
y = np.sin(x)
plt.plot(x,y)
The result of the image is uploaded to the Medium endpoint (without being written to memory) and the corresponding plot is replaced by the plot's URL.
If the image is already online, nothing changes to it, as Medium can access it directly from the internet when loading the article
Code cells as GitHub gists
By default a code cell is rendered without syntax highlighting or anything fancy like:
import pandas as pd
pd.DataFrame({'a': [1,2,3], 'b': [2,4,6]})
Though if a GitHub token is available and the user includes the following header in a cell:
# %gist gistname: pandas.py
import pandas as pd
pd.DataFrame({'a': [1,2,3], 'b': [2,4,6]})
The code block will be uploaded to the user's GitHub as a private gist (by default - can be changed to be made public) and the respective code cell will be replaced by the gist URL. The user can also use the description:
and the public:
flags in the #gist
header, where public:
maybe either True or False and description can just be a string (no single or double quotes required - just avoid using the keywords gistname, public or description)
Hiding cells
It is often convenient to hide either a cell's source (i.e. the code), a cell's output (the result of evaluating the code) or the whole cell altogether. To achieve this the user can place the following header at the start of the relevant cells.
- To hide a cell's source:
# %hide-source
print("This code won't be shown, but it's output will")
- To hide a cell's output:
# %hide-output
print("This code will be shown, but it's output won't")
- To hide a cell completely (source and output):
# %hide-cell
print("This cell won't make it to the final document")
Note: all tags (%hide-*
and %gist
) were not designed with the idea to be combined so such usage has not been tested. In general there should be no need for such behaviour.
Tables
Medium does not have good support for HTML nor Markdown tables. My preferred existing option for tables is gist
. If a cell outputs a pandas dataframe and you choose the #%gist
option with the value of the upload:
flag set to both
or output
, nd2medium
will detect your table and upload it as a CSV to your GitHub gist repo.
#% gist gistname: pandas.py upload: both
import pandas as pd
pd.DataFrame({'a': [1,2,3], 'b': [0,0,0], 'c': ['One', 'Two', 'Three']})
Documentation
The docs are available at https://lucharo.github.io/nb2medium and are rendered automatically from the nbdev
notebooks so they are always up to date with the package source code
Contributing
If you find a bug or think of an enhancement feel free to raise issues or submit pull requests. If you want to contribute to open source projects such as this one have a look at the issues with the label/tag help needed
in particular.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
File details
Details for the file nb2medium-0.0.2.tar.gz
.
File metadata
- Download URL: nb2medium-0.0.2.tar.gz
- Upload date:
- Size: 100.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/3.3.0 pkginfo/1.7.0 requests/2.25.1 setuptools/52.0.0.post20210125 requests-toolbelt/0.9.1 tqdm/4.58.0 CPython/3.9.1
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 43d7c52492f400aa00d4dcc26fb173dbbfd76b835866ead5ce3c417fd31a1bd4 |
|
MD5 | 9a4c02244d1910fdf21ceed4fbfdb9ec |
|
BLAKE2b-256 | 1409b3c2f03a14d9f078570f263456f0514b52010175897abb7225d01d879b9e |
File details
Details for the file nb2medium-0.0.2-py3-none-any.whl
.
File metadata
- Download URL: nb2medium-0.0.2-py3-none-any.whl
- Upload date:
- Size: 305.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/3.3.0 pkginfo/1.7.0 requests/2.25.1 setuptools/52.0.0.post20210125 requests-toolbelt/0.9.1 tqdm/4.58.0 CPython/3.9.1
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | c83ee5336840427e2ccb61b2b00390940e19d4187b57b6935812b5151a5e0a31 |
|
MD5 | af83368ed83458e23122adc7fdccef1f |
|
BLAKE2b-256 | fa5cf275d3f53b67c43e61f54817b08a4177d64b149c4cee85a772c445e1d1ec |