mirror of
https://github.com/kiwix/kiwix-android.git
synced 2025-08-03 10:46:53 -04:00
Added upload-apk script (only google play for now) and bm lang flag
This commit is contained in:
parent
4e0e769e90
commit
d904f17fee
BIN
templates/launcher-flag-bm.png
Normal file
BIN
templates/launcher-flag-bm.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 77 KiB |
236
upload-apk.py
Executable file
236
upload-apk.py
Executable file
@ -0,0 +1,236 @@
|
||||
#!/usr/bin/env python
|
||||
# -*- coding: utf-8 -*-
|
||||
# vim: ai ts=4 sts=4 et sw=4 nu
|
||||
|
||||
from __future__ import (unicode_literals, absolute_import,
|
||||
division, print_function)
|
||||
import logging
|
||||
import sys
|
||||
import os
|
||||
import json
|
||||
import requests
|
||||
import tempfile
|
||||
import shutil
|
||||
from subprocess import call
|
||||
try:
|
||||
from StringIO import StringIO
|
||||
except ImportError:
|
||||
from io import StringIO
|
||||
|
||||
# check for python version as google client api is broken on py2
|
||||
if sys.version_info.major < 3:
|
||||
print("You must run this script with python3 as "
|
||||
"Google API Client is broken python2")
|
||||
sys.exit(1)
|
||||
|
||||
PLAY_STORE = 'play_store'
|
||||
ALPHA = 'alpha'
|
||||
BETA = 'beta'
|
||||
PROD = 'prod'
|
||||
|
||||
logging.basicConfig(level=logging.DEBUG)
|
||||
logger = logging.getLogger(__name__)
|
||||
for handler in logging.root.handlers:
|
||||
handler.addFilter(logging.Filter('__main__'))
|
||||
CURRENT_PATH = os.path.dirname(os.path.abspath(__file__))
|
||||
|
||||
|
||||
def usage(arg0, exit=None):
|
||||
print("Usage: {} <json_file>".format(arg0))
|
||||
if exit is not None:
|
||||
sys.exit(exit)
|
||||
|
||||
|
||||
def syscall(args, shell=False, with_print=True):
|
||||
''' execute an external command. Use shell=True if using bash specifics '''
|
||||
args = args.split()
|
||||
if with_print:
|
||||
print(u"-----------\n" + u" ".join(args) + u"\n-----------")
|
||||
|
||||
if shell:
|
||||
args = ' '.join(args)
|
||||
call(args, shell=shell)
|
||||
|
||||
|
||||
def get_remote_content(url):
|
||||
''' file descriptor from remote file using GET '''
|
||||
req = requests.get(url)
|
||||
try:
|
||||
req.raise_for_status()
|
||||
except Exception as e:
|
||||
logger.error("Failed to load data at `{}`".format(url))
|
||||
logger.exception(e)
|
||||
sys.exit(1)
|
||||
return StringIO.StringIO(req.text)
|
||||
|
||||
|
||||
def get_local_content(path):
|
||||
''' file descriptor from local file '''
|
||||
if not os.path.exists(path) or not os.path.isfile(path):
|
||||
logger.error("Unable to find JSON file `{}`".format(path))
|
||||
sys.exit(1)
|
||||
|
||||
try:
|
||||
fd = open(path, 'r')
|
||||
except Exception as e:
|
||||
logger.error("Unable to open file `{}`".format(path))
|
||||
logger.exception(e)
|
||||
sys.exit(1)
|
||||
return fd
|
||||
|
||||
|
||||
def is_remote_path(path):
|
||||
return path.startswith('http:')
|
||||
|
||||
|
||||
def get_local_remote_fd(path):
|
||||
''' file descriptor for a path (either local or remote) '''
|
||||
if is_remote_path(path):
|
||||
return get_remote_content(path)
|
||||
else:
|
||||
return get_local_content(path)
|
||||
|
||||
|
||||
def copy_to(src, dst):
|
||||
''' copy source content (local or remote) to local file '''
|
||||
local = None
|
||||
if is_remote_path(src):
|
||||
local = tempfile.NamedTemporaryFile(delete=False)
|
||||
download_remote_file(src, local.name)
|
||||
src = local.name
|
||||
shutil.copy(src, dst)
|
||||
if local is not None:
|
||||
os.remove(local.name)
|
||||
|
||||
|
||||
def download_remote_file(url, path):
|
||||
''' download url to path '''
|
||||
syscall('wget -c -O {path} {url}'.format(path=path, url=url))
|
||||
|
||||
|
||||
def upload_to_play_store(jsdata, channel=None):
|
||||
if channel is None:
|
||||
channel = ALPHA
|
||||
|
||||
logger.info("Starting Google Play Store using {}".format(channel))
|
||||
|
||||
# ensure dependencies are met
|
||||
try:
|
||||
import httplib2
|
||||
from apiclient.discovery import build
|
||||
from oauth2client import client
|
||||
except ImportError:
|
||||
logger.error("Missing Google API Client dependency.\n"
|
||||
"Please install with: "
|
||||
"pip install google-api-python-client")
|
||||
return
|
||||
|
||||
if 'GOOGLE_API_KEY' not in os.environ:
|
||||
logger.error("You need to set the GOOGLE_API_KEY environment variable "
|
||||
"to use the Google API (using path to google-api.p12)")
|
||||
return
|
||||
|
||||
GOOGLE_CLIENT_ID = '107823297044-nhoqv99cpr86vlfcronskirgib2g7tq' \
|
||||
'9@developer.gserviceaccount.com'
|
||||
|
||||
service = build('androidpublisher', 'v2')
|
||||
|
||||
key = open(os.environ['GOOGLE_API_KEY'], 'rb').read()
|
||||
credentials = client.SignedJwtAssertionCredentials(
|
||||
GOOGLE_CLIENT_ID,
|
||||
key,
|
||||
scope='https://www.googleapis.com/auth/androidpublisher')
|
||||
|
||||
http = httplib2.Http()
|
||||
http = credentials.authorize(http)
|
||||
|
||||
service = build('androidpublisher', 'v2', http=http)
|
||||
|
||||
package_name = jsdata['package']
|
||||
version_name = jsdata['version_name']
|
||||
apk_file = os.path.join(CURRENT_PATH, 'build', 'outputs', 'apk',
|
||||
'{}-{}.apk'.format(package_name, version_name))
|
||||
|
||||
if not jsdata.get('embed_zim', False):
|
||||
comp_file = tempfile.NamedTemporaryFile(suffix='.a').name
|
||||
copy_to(jsdata['zim_file'], comp_file)
|
||||
|
||||
try:
|
||||
# another edit request
|
||||
edit_request = service.edits().insert(body={},
|
||||
packageName=package_name)
|
||||
result = edit_request.execute()
|
||||
edit_id = result['id']
|
||||
|
||||
logger.info("Starting Edit `{}`".format(edit_id))
|
||||
|
||||
# upload APK
|
||||
logger.info("Uploading APK file: {}".format(apk_file))
|
||||
apk_response = service.edits().apks().upload(
|
||||
editId=edit_id,
|
||||
packageName=package_name,
|
||||
media_body=apk_file).execute()
|
||||
|
||||
logger.debug("APK for version code {} has been uploaded"
|
||||
.format(apk_response['versionCode']))
|
||||
|
||||
# release APK into the specified channel
|
||||
track_response = service.edits().tracks().update(
|
||||
editId=edit_id,
|
||||
track=channel,
|
||||
packageName=package_name,
|
||||
body={'versionCodes': [apk_response['versionCode']]}).execute()
|
||||
|
||||
logger.debug("Publicatio set to {} for version code {}"
|
||||
.format(track_response['track'],
|
||||
str(track_response['versionCodes'])))
|
||||
|
||||
# upload companion file
|
||||
if comp_file:
|
||||
logger.info("Uploading Expansion file: {}".format(comp_file))
|
||||
comp_response = service.edits().expansionfiles().upload(
|
||||
editId=edit_id,
|
||||
packageName=package_name,
|
||||
apkVersionCode=jsdata['version_code'],
|
||||
expansionFileType='main',
|
||||
media_body=comp_file).execute()
|
||||
|
||||
logger.debug("Expansion file of size {} has been uploaded"
|
||||
.format(comp_response['expansionFile']['fileSize']))
|
||||
|
||||
commit_request = service.edits().commit(
|
||||
editId=edit_id, packageName=package_name).execute()
|
||||
|
||||
logger.debug("Edit `{}` has been committed. done."
|
||||
.format(commit_request['id']))
|
||||
|
||||
except client.AccessTokenRefreshError:
|
||||
logger.error("The credentials have been revoked or expired, "
|
||||
"please re-run the application to re-authorize")
|
||||
|
||||
STORES = {
|
||||
'play_store': upload_to_play_store,
|
||||
}
|
||||
|
||||
|
||||
def main(json_path, store='{}:{}'.format(PLAY_STORE, ALPHA)):
|
||||
jsdata = json.load(get_local_remote_fd(json_path))
|
||||
|
||||
logger.info("Uploading {} APK to {}".format(jsdata['package'], store))
|
||||
|
||||
try:
|
||||
store, channel = store.split(':', 1)
|
||||
except (IndexError, ValueError):
|
||||
channel = None
|
||||
|
||||
STORES.get(store)(jsdata, channel=channel)
|
||||
|
||||
if __name__ == '__main__':
|
||||
# ensure we were provided a JSON file as first argument
|
||||
if len(sys.argv) < 2:
|
||||
usage(sys.argv[0], 1)
|
||||
else:
|
||||
jspath = sys.argv[1]
|
||||
args = sys.argv[2:]
|
||||
|
||||
main(jspath, *args)
|
Loading…
x
Reference in New Issue
Block a user