Python and Flask
This tutorial will show you how to use the Lokalise Python SDK to manage translation files and how to implement an OAuth 2 flow.
In this tutorial you'll learn how to...
- Work with Lokalise API tokens
- Upload translation files with the Lokalise API
- Download translation files with the Lokalise API
- Implement an OAuth 2 flow
- List customer projects
- Act on the customer's behalf and upload/download translation files
You can find the source code at GitHub.
To learn more about libraries for Python translate, please refer to our dedicated blog post.
Prerequisites
This guide assumes that you have a Lokalise project (if not, learn how to create your Lokalise project here). In this tutorial, we are going to upload English translations. Therefore, make sure that your project has the English language added with the “en” ISO code.
If you want to follow this guide locally on your computer, you need to have the following software installed:
- Python (version 3.7 or above)
- Code editor
What we are going to build
In the first part of this tutorial, we are going to use regular tokens to upload an English translation file to your Lokalise project and download it back.
In the second part, we'll implement an OAuth 2 flow and act on the user's behalf to upload and download translation files to or from their project.
Setting up an app
In this tutorial we will use a minimalist Flask framework, but you can utilize any other solution you see fit.
First, create and activate a new virtual environment in your project's directory:
cd MY_PROJECT_NAME && python -m venv venv
. venv/bin/activate
If you are on Windows, the latter command should look like venv\Scripts\activate
.
Now install Flask, the Lokalise Python SDK, and dotenv to store environment variables:
pip install Flask python-dotenv python-lokalise-api requests
Create a new file called app.py
with the following contents:
import os
import base64
from dotenv import load_dotenv
load_dotenv()
import zipfile
import io
import requests
import lokalise
from flask import Flask, render_template, request, redirect, url_for, session
app = Flask(__name__)
Finally, create a templates
folder and an i18n
folder with an en.json
file. Paste the following contents inside the en.json
file:
{
"app.name": "Demo app",
"app.description": "Demo app description"
}
Working with API tokens
Getting an API token to use the Lokalise API
Create an .env
file in the root of the project and add your Lokalise API token. Make sure to add an API token with read and write access to your Lokalise projects. Learn how to get a Lokalise API token.
LOKALISE_API_KEY=123secret456
Never publicly expose your API key!
Specifically, don’t forget to add the
.env
file to.gitignore
.
Initializing the client
Let's create a method to initialize the Lokalise Python SDK client:
def __client():
return lokalise.Client(os.getenv('LOKALISE_API_KEY'))
Getting your Lokalise project ID
Now add a new environment variable inside your .env
file with your Lokalise project ID. Learn how to get the Lokalise project ID.
LOKALISE_PROJECT_ID=123.abc
Uploading translation files to your Lokalise project
To get started with file uploading, we need to add the corresponding template, so create a new upload.html
file inside the templates
directory:
<!doctype html>
<html lang="en">
<head>
<title>Upload translation file</title>
</head>
<body>
<h1>Upload translation file</h1>
<form method="POST" action="/upload">
<input type="submit" value="Upload!">
</form>
</body>
</html>
Now let's perform the actual upload. Add the upload
method to app.py
:
@app.route('/upload', methods=['GET', 'POST'])
def upload():
if request.method == 'POST':
filename = os.path.join(os.path.dirname(__file__), 'i18n/en.json')
with open(filename) as f:
content = f.read()
__client().upload_file(os.getenv('LOKALISE_PROJECT_ID'), {
"data": base64.b64encode(content.encode()).decode(),
"filename": 'en.json',
"lang_iso": 'en'
})
return redirect(url_for('upload'))
else:
return render_template('upload.html')
We read our JSON file and encode its contents in Base64. Then simply use the upload_file
method to perform the upload.
Now you can boot your server by running:
flask run
Proceed to http://127.0.0.1:5000/upload and click "Upload!". Then return to your Lokalise project and make sure the translations were properly uploaded:
Great job!
Downloading translation files from the Lokalise project
Start by creating a simple template inside the templates/download.html
file:
<!doctype html>
<html lang="en">
<head>
<title>Download translation file</title>
</head>
<body>
<h1>Download translation file</h1>
<form method="POST" action="/download">
<input type="submit" value="Download!">
</form>
</body>
</html>
Next, add a new download
method:
@app.route('/download', methods=['GET', 'POST'])
def download():
if request.method == 'POST':
response = __client().download_files(os.getenv('LOKALISE_PROJECT_ID'), {
"format": "json",
"filter_langs": ["en"],
"original_filenames": True,
"directory_prefix": ""
})
data = io.BytesIO(requests.get(response['bundle_url']).content)
with zipfile.ZipFile(data) as archive:
archive.extract("en.json", path="i18n/")
return redirect(url_for('download'))
else:
return render_template('download.html')
In this case, we are downloading only the English translations in JSON format, getting the download bundle, reading the archive, and extracting the required file to the i18n
folder.
Your existing translations will be overwritten!
Please remember that the current implementation overwrites any translations inside the
en.json
file with the new ones. If you don't want this to happen, save your new translations to a file with another name.
Now you can edit existing translations in your Lokalise project, reload the server, proceed to http://127.0.0.1:5000/download, and click "Download!". Make sure that your translations are replaced with the new ones.
Awesome!
Working with an OAuth 2 flow
Registering an OAuth 2 app
To get started, please reach out to our tech support and ask them to register an OAuth 2 app for you (if you don't have one registered already). You will be presented with the Lokalise client ID and client secret that we will be using in the next steps.
You can store these keys in the .env
file:
OAUTH2_CLIENT_ID=your_id
OAUTH2_CLIENT_SECRET=your_secret
Never publicly expose your OAuth 2 client secret!
Specifically, don’t forget to add the
.env
file to.gitignore
.
Implementing an OAuth 2 flow
To implement an OAuth 2 flow, we'll need two new routes: one to display the actual link, and another to read response from Lokalise and request an OAuth 2 token.
Let's start with the login
method:
@app.route('/login')
def login():
login_url = __auth_client().auth(
["read_projects", "read_files", "write_files"], "http://localhost:5000/callback", "random state"
)
return render_template('login.html', login_url=login_url)
# ...
def __auth_client():
return lokalise.Auth(os.getenv('OAUTH2_CLIENT_ID'), os.getenv('OAUTH2_CLIENT_SECRET'))
login_url
will contain the actual link that the user will need to visit and log in to Lokalise. Note that we are requesting permissions to view the user's projects and read/write translation files. The callback points to the /callback
route that we'll implement in a moment.
Next, add the callback
method:
@app.route('/callback')
def callback():
code = request.args.get('code', '')
response = __auth_client().token(code)
session['token'] = response['access_token']
return redirect(url_for('login'))
code
is a secret alphanumeric string generated by Lokalise. We use it to obtain the user's OAuth 2 token. This token will then be utilized to send API requests; therefore, store it in session
.
Also make sure to add a secret key to protect the session (it should not be exposed):
app = Flask(__name__)
app.secret_key = b'_5#y2L"F4Q8z\n\xec]/' # <=== add this line
Now add the templates/login.html
file:
<!doctype html>
<html lang="en">
<head>
<title>Login to Lokalise</title>
</head>
<body>
<h1>Lokalise OAuth 2 login</h1>
{% if session['token'] %}
<p>
Your OAuth 2 token: {{ session['token'] }}<br>
{% if session['project_id'] %}
Your project ID: {{ session['project_id'] }}<br>
{% endif %}
<a href="/projects">Choose a project to work with</a><br>
<a href="/upload">Upload translation file</a><br>
<a href="/download">Download translation file</a>
</p>
<form action="/logout" method="POST">
<input type="submit" value="Logout">
</form>
{% else %}
<a href="{{ login_url }}">Login!</a>
{% endif %}
</body>
</html>
Choosing a project to work with
The next is step is to allow our user to choose a project that s/he wants to work with. Thus, create yet another route:
@app.route('/projects', methods=['GET', 'POST'])
def projects():
if request.method == 'POST':
session['project_id'] = request.form['project_id']
return redirect(url_for('login'))
else:
projects = __oauth_client().projects().items
return render_template('projects.html', projects=projects)
This route either displays a list of projects obtained via the Lokalise API, or stores the chosen project ID in session
.
Add a new method to prepare the OAuth 2 client:
def __oauth_client():
return lokalise.OAuthClient(session['token'])
Let's add a new template under templates/projects.html
as well:
<!doctype html>
<html lang="en">
<head>
<title>Your Lokalise projects</title>
</head>
<body>
<h1>Choose a Lokalise project to work with</h1>
{% for project in projects %}
<p>
{{ project.name }}
{% if session['project_id'] == project.project_id %}
(currently chosen)
{% endif %}
</p>
<form action="/projects" method="POST">
<input type="hidden" value="{{ project.project_id }}" name="project_id">
<input type="submit" value="Choose">
</form>
<hr>
{% endfor %}
</body>
</html>
Uploading translation files on the user's behalf
So, once the user has chosen a project to work with, we can upload files to that project. Therefore, modify the upload
method in the following way:
@app.route('/upload', methods=['GET', 'POST'])
def upload():
if request.method == 'POST':
filename = os.path.join(os.path.dirname(__file__), 'i18n/en.json')
with open(filename) as f:
content = f.read()
__oauth_client().upload_file(session['project_id'], {
"data": base64.b64encode(content.encode()).decode(),
"filename": 'en.json',
"lang_iso": 'en'
})
return redirect(url_for('upload'))
else:
return render_template('upload.html')
Now we are using the __oauth_client()
method and read the chosen project ID from the session store.
Downloading translation files on the user's behalf
The next stop is the download()
method that we are going to modify in the following way:
@app.route('/download', methods=['GET', 'POST'])
def download():
if request.method == 'POST':
response = __oauth_client().download_files(session['project_id'], {
"format": "json",
"filter_langs": ["en"],
"original_filenames": True,
"directory_prefix": ""
})
data = io.BytesIO(requests.get(response['bundle_url']).content)
with zipfile.ZipFile(data) as archive:
archive.extract("en.json", path="i18n/")
return redirect(url_for('download'))
else:
return render_template('download.html')
Once again, we are using an OAuth 2 client and the project ID stored in session
.
Logging out and a template
Finally let's add the ability to log out, which simply means clearing the session store:
@app.route('/logout', methods=['POST'])
def logout():
session.pop('token', None)
session.pop('project_id', None)
return redirect(url_for('login'))
The last thing to do is code the login.html
template which should provide all the necessary links and buttons:
<!doctype html>
<html lang="en">
<head>
<title>Login to Lokalise</title>
</head>
<body>
<h1>Lokalise OAuth 2 login</h1>
{% if session['token'] %}
<p>
Your OAuth 2 token: {{ session['token'] }}<br>
{% if session['project_id'] %}
Your project ID: {{ session['project_id'] }}<br>
{% endif %}
<a href="/projects">Choose a project to work with</a><br>
<a href="/upload">Upload translation file</a><br>
<a href="/download">Download translation file</a>
</p>
<form action="/logout" method="POST">
<input type="submit" value="Logout">
</form>
{% else %}
<a href="{{ login_url }}">Login!</a>
{% endif %}
</body>
</html>
Testing it out
That's it! Now you can boot your server by running:
flask run
Navigate to http://localhost:5000/login and click "Login!". You'll see the following page:
Click "Allow access". You'll be navigated back to your app.
Next, click "Choose a project to work with", and select one of the projects.
Finally, click either "Upload file" or "Download file", and perform the uploading/downloading process as before. Make sure that your translations are updated accordingly.
That's it, great job!
Updated 4 months ago