2017-03-21 00:50:13 +03:00
|
|
|
|
# coding: utf8
|
|
|
|
|
from __future__ import unicode_literals
|
|
|
|
|
|
2017-05-22 13:28:58 +03:00
|
|
|
|
import plac
|
2017-03-21 04:06:29 +03:00
|
|
|
|
import shutil
|
2017-03-21 00:50:13 +03:00
|
|
|
|
from pathlib import Path
|
|
|
|
|
|
2017-05-08 00:25:29 +03:00
|
|
|
|
from ..compat import path2str, json_dumps
|
|
|
|
|
from ..util import prints
|
2017-03-21 00:50:13 +03:00
|
|
|
|
from .. import util
|
2017-05-08 00:25:29 +03:00
|
|
|
|
from .. import about
|
2017-03-21 00:50:13 +03:00
|
|
|
|
|
|
|
|
|
|
2017-05-22 13:28:58 +03:00
|
|
|
|
@plac.annotations(
|
|
|
|
|
input_dir=("directory with model data", "positional", None, str),
|
|
|
|
|
output_dir=("output parent directory", "positional", None, str),
|
2017-08-12 22:44:15 +03:00
|
|
|
|
meta_path=("path to meta.json", "option", "m", str),
|
2017-10-30 20:39:38 +03:00
|
|
|
|
create_meta=("create meta.json, even if one exists in directory – if "
|
|
|
|
|
"existing meta is found, entries are shown as defaults in "
|
|
|
|
|
"the command line prompt", "flag", "c", bool),
|
|
|
|
|
force=("force overwriting of existing model directory in output directory",
|
|
|
|
|
"flag", "f", bool))
|
2017-10-27 15:38:39 +03:00
|
|
|
|
def package(cmd, input_dir, output_dir, meta_path=None, create_meta=False,
|
|
|
|
|
force=False):
|
2017-05-27 21:01:46 +03:00
|
|
|
|
"""
|
|
|
|
|
Generate Python package for model data, including meta and required
|
2017-05-22 13:28:58 +03:00
|
|
|
|
installation files. A new directory will be created in the specified
|
|
|
|
|
output directory, and model data will be copied over.
|
|
|
|
|
"""
|
2017-05-08 00:25:29 +03:00
|
|
|
|
input_path = util.ensure_path(input_dir)
|
|
|
|
|
output_path = util.ensure_path(output_dir)
|
2017-08-12 22:44:15 +03:00
|
|
|
|
meta_path = util.ensure_path(meta_path)
|
2017-05-08 00:25:29 +03:00
|
|
|
|
if not input_path or not input_path.exists():
|
2017-05-22 13:28:58 +03:00
|
|
|
|
prints(input_path, title="Model directory not found", exits=1)
|
2017-05-08 00:25:29 +03:00
|
|
|
|
if not output_path or not output_path.exists():
|
2017-05-22 13:28:58 +03:00
|
|
|
|
prints(output_path, title="Output directory not found", exits=1)
|
2017-05-08 00:25:29 +03:00
|
|
|
|
if meta_path and not meta_path.exists():
|
2017-05-22 13:28:58 +03:00
|
|
|
|
prints(meta_path, title="meta.json not found", exits=1)
|
2017-03-21 00:50:13 +03:00
|
|
|
|
|
2017-04-16 14:06:02 +03:00
|
|
|
|
meta_path = meta_path or input_path / 'meta.json'
|
2017-10-30 20:39:38 +03:00
|
|
|
|
if meta_path.is_file():
|
2017-04-16 14:06:02 +03:00
|
|
|
|
meta = util.read_json(meta_path)
|
2017-10-30 20:39:38 +03:00
|
|
|
|
if not create_meta: # only print this if user doesn't want to overwrite
|
|
|
|
|
prints(meta_path, title="Loaded meta.json from file")
|
|
|
|
|
else:
|
|
|
|
|
meta = generate_meta(input_dir, meta)
|
2017-05-27 21:02:01 +03:00
|
|
|
|
meta = validate_meta(meta, ['lang', 'name', 'version'])
|
2017-03-21 00:50:13 +03:00
|
|
|
|
model_name = meta['lang'] + '_' + meta['name']
|
|
|
|
|
model_name_v = model_name + '-' + meta['version']
|
|
|
|
|
main_path = output_path / model_name_v
|
|
|
|
|
package_path = main_path / model_name
|
|
|
|
|
|
2017-03-21 04:06:53 +03:00
|
|
|
|
create_dirs(package_path, force)
|
2017-10-27 15:38:39 +03:00
|
|
|
|
shutil.copytree(path2str(input_path),
|
|
|
|
|
path2str(package_path / model_name_v))
|
2017-04-14 00:30:47 +03:00
|
|
|
|
create_file(main_path / 'meta.json', json_dumps(meta))
|
2017-11-07 14:15:35 +03:00
|
|
|
|
create_file(main_path / 'setup.py', TEMPLATE_SETUP)
|
|
|
|
|
create_file(main_path / 'MANIFEST.in', TEMPLATE_MANIFEST)
|
|
|
|
|
create_file(package_path / '__init__.py', TEMPLATE_INIT)
|
2017-10-27 15:38:39 +03:00
|
|
|
|
prints(main_path, "To build the package, run `python setup.py sdist` in "
|
|
|
|
|
"this directory.",
|
|
|
|
|
title="Successfully created package '%s'" % model_name_v)
|
2017-03-21 00:50:13 +03:00
|
|
|
|
|
|
|
|
|
|
2017-03-21 04:06:53 +03:00
|
|
|
|
def create_dirs(package_path, force):
|
|
|
|
|
if package_path.exists():
|
|
|
|
|
if force:
|
2017-05-08 00:25:29 +03:00
|
|
|
|
shutil.rmtree(path2str(package_path))
|
2017-03-21 04:06:53 +03:00
|
|
|
|
else:
|
2017-10-27 15:38:39 +03:00
|
|
|
|
prints(package_path, "Please delete the directory and try again, "
|
|
|
|
|
"or use the --force flag to overwrite existing "
|
|
|
|
|
"directories.", title="Package directory already exists",
|
|
|
|
|
exits=1)
|
2017-03-21 04:06:53 +03:00
|
|
|
|
Path.mkdir(package_path, parents=True)
|
|
|
|
|
|
|
|
|
|
|
2017-03-21 00:50:13 +03:00
|
|
|
|
def create_file(file_path, contents):
|
|
|
|
|
file_path.touch()
|
2017-03-29 10:11:02 +03:00
|
|
|
|
file_path.open('w', encoding='utf-8').write(contents)
|
2017-03-21 00:50:13 +03:00
|
|
|
|
|
|
|
|
|
|
2017-10-30 20:39:38 +03:00
|
|
|
|
def generate_meta(model_path, existing_meta):
|
|
|
|
|
meta = existing_meta or {}
|
|
|
|
|
settings = [('lang', 'Model language', meta.get('lang', 'en')),
|
|
|
|
|
('name', 'Model name', meta.get('name', 'model')),
|
|
|
|
|
('version', 'Model version', meta.get('version', '0.0.0')),
|
2017-10-27 15:38:39 +03:00
|
|
|
|
('spacy_version', 'Required spaCy version',
|
|
|
|
|
'>=%s,<3.0.0' % about.__version__),
|
2017-10-30 20:39:38 +03:00
|
|
|
|
('description', 'Model description',
|
|
|
|
|
meta.get('description', False)),
|
|
|
|
|
('author', 'Author', meta.get('author', False)),
|
|
|
|
|
('email', 'Author email', meta.get('email', False)),
|
|
|
|
|
('url', 'Author website', meta.get('url', False)),
|
|
|
|
|
('license', 'License', meta.get('license', 'CC BY-SA 3.0'))]
|
2017-10-25 17:03:26 +03:00
|
|
|
|
nlp = util.load_model_from_path(Path(model_path))
|
|
|
|
|
meta['pipeline'] = nlp.pipe_names
|
|
|
|
|
meta['vectors'] = {'width': nlp.vocab.vectors_length,
|
2017-11-01 03:25:09 +03:00
|
|
|
|
'vectors': len(nlp.vocab.vectors),
|
|
|
|
|
'keys': nlp.vocab.vectors.n_keys}
|
2017-10-25 17:03:26 +03:00
|
|
|
|
prints("Enter the package settings for your model. The following "
|
|
|
|
|
"information will be read from your model data: pipeline, vectors.",
|
|
|
|
|
title="Generating meta.json")
|
2017-03-21 00:50:13 +03:00
|
|
|
|
for setting, desc, default in settings:
|
|
|
|
|
response = util.get_raw_input(desc, default)
|
|
|
|
|
meta[setting] = default if response == '' and default else response
|
2017-06-05 21:11:02 +03:00
|
|
|
|
if about.__title__ != 'spacy':
|
|
|
|
|
meta['parent_package'] = about.__title__
|
2017-03-21 00:50:13 +03:00
|
|
|
|
return meta
|
|
|
|
|
|
|
|
|
|
|
2017-04-16 14:13:17 +03:00
|
|
|
|
def validate_meta(meta, keys):
|
|
|
|
|
for key in keys:
|
|
|
|
|
if key not in meta or meta[key] == '':
|
2017-05-08 00:25:29 +03:00
|
|
|
|
prints("This setting is required to build your package.",
|
2017-05-22 13:28:58 +03:00
|
|
|
|
title='No "%s" setting found in meta.json' % key, exits=1)
|
2017-05-27 21:02:01 +03:00
|
|
|
|
return meta
|
2017-04-16 14:13:17 +03:00
|
|
|
|
|
|
|
|
|
|
2017-11-07 14:15:35 +03:00
|
|
|
|
TEMPLATE_SETUP = """
|
|
|
|
|
#!/usr/bin/env python
|
|
|
|
|
# coding: utf8
|
|
|
|
|
from __future__ import unicode_literals
|
|
|
|
|
|
|
|
|
|
import io
|
|
|
|
|
import json
|
|
|
|
|
from os import path, walk
|
|
|
|
|
from shutil import copy
|
|
|
|
|
from setuptools import setup
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def load_meta(fp):
|
|
|
|
|
with io.open(fp, encoding='utf8') as f:
|
|
|
|
|
return json.load(f)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def list_files(data_dir):
|
|
|
|
|
output = []
|
|
|
|
|
for root, _, filenames in walk(data_dir):
|
|
|
|
|
for filename in filenames:
|
|
|
|
|
if not filename.startswith('.'):
|
|
|
|
|
output.append(path.join(root, filename))
|
|
|
|
|
output = [path.relpath(p, path.dirname(data_dir)) for p in output]
|
|
|
|
|
output.append('meta.json')
|
|
|
|
|
return output
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def list_requirements(meta):
|
|
|
|
|
parent_package = meta.get('parent_package', 'spacy')
|
|
|
|
|
requirements = [parent_package + meta['spacy_version']]
|
|
|
|
|
if 'setup_requires' in meta:
|
|
|
|
|
requirements += meta['setup_requires']
|
|
|
|
|
return requirements
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def setup_package():
|
|
|
|
|
root = path.abspath(path.dirname(__file__))
|
|
|
|
|
meta_path = path.join(root, 'meta.json')
|
|
|
|
|
meta = load_meta(meta_path)
|
|
|
|
|
model_name = str(meta['lang'] + '_' + meta['name'])
|
|
|
|
|
model_dir = path.join(model_name, model_name + '-' + meta['version'])
|
|
|
|
|
|
|
|
|
|
copy(meta_path, path.join(model_name))
|
|
|
|
|
copy(meta_path, model_dir)
|
|
|
|
|
|
|
|
|
|
setup(
|
|
|
|
|
name=model_name,
|
|
|
|
|
description=meta['description'],
|
|
|
|
|
author=meta['author'],
|
|
|
|
|
author_email=meta['email'],
|
|
|
|
|
url=meta['url'],
|
|
|
|
|
version=meta['version'],
|
|
|
|
|
license=meta['license'],
|
|
|
|
|
packages=[model_name],
|
|
|
|
|
package_data={model_name: list_files(model_dir)},
|
|
|
|
|
install_requires=list_requirements(meta),
|
|
|
|
|
zip_safe=False,
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
|
setup_package()
|
|
|
|
|
""".strip()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
TEMPLATE_MANIFEST = """
|
|
|
|
|
include meta.json
|
|
|
|
|
""".strip()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
TEMPLATE_INIT = """
|
|
|
|
|
# coding: utf8
|
|
|
|
|
from __future__ import unicode_literals
|
|
|
|
|
|
|
|
|
|
from pathlib import Path
|
|
|
|
|
from spacy.util import load_model_from_init_py, get_model_meta
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
__version__ = get_model_meta(Path(__file__).parent)['version']
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def load(**overrides):
|
|
|
|
|
return load_model_from_init_py(__file__, **overrides)
|
|
|
|
|
""".strip()
|