spaCy/spacy/cli/train.py

211 lines
8.7 KiB
Python
Raw Normal View History

2017-03-23 10:08:41 +00:00
# coding: utf8
from __future__ import unicode_literals, division, print_function
import plac
2017-03-23 10:08:41 +00:00
import json
from collections import defaultdict
import cytoolz
from pathlib import Path
import dill
2017-05-21 14:07:06 +00:00
import tqdm
2017-09-21 00:17:10 +00:00
from thinc.neural._classes.model import Model
from thinc.neural.optimizers import linear_decay
from timeit import default_timer as timer
2017-09-23 00:57:31 +00:00
import random
import numpy.random
2017-03-23 10:08:41 +00:00
from ..tokens.doc import Doc
2017-03-23 10:08:41 +00:00
from ..scorer import Scorer
from ..gold import GoldParse, merge_sents
2017-05-25 21:16:30 +00:00
from ..gold import GoldCorpus, minibatch
2017-05-07 21:25:29 +00:00
from ..util import prints
2017-03-23 10:08:41 +00:00
from .. import util
from .. import about
from .. import displacy
2017-06-05 01:18:37 +00:00
from ..compat import json_dumps
2017-03-23 10:08:41 +00:00
2017-09-23 00:57:31 +00:00
random.seed(0)
numpy.random.seed(0)
2017-03-23 10:08:41 +00:00
@plac.annotations(
lang=("model language", "positional", None, str),
output_dir=("output directory to store model in", "positional", None, str),
train_data=("location of JSON-formatted training data", "positional", None, str),
dev_data=("location of JSON-formatted development data (optional)", "positional", None, str),
n_iter=("number of iterations", "option", "n", int),
n_sents=("number of sentences", "option", "ns", int),
2017-06-03 21:10:23 +00:00
use_gpu=("Use GPU", "option", "g", int),
2017-09-23 01:00:40 +00:00
vectors=("Model to load vectors from", "option", "v"),
no_tagger=("Don't train tagger", "flag", "T", bool),
no_parser=("Don't train parser", "flag", "P", bool),
2017-08-20 16:07:00 +00:00
no_entities=("Don't train NER", "flag", "N", bool),
gold_preproc=("Use gold preprocessing", "flag", "G", bool),
2017-09-26 15:59:34 +00:00
version=("Model version", "option", "V", str),
meta_path=("Optional path to meta.json. All relevant properties will be overwritten.", "option", "m", Path)
)
2017-09-27 23:54:37 +00:00
def train(cmd, lang, output_dir, train_data, dev_data, n_iter=10, n_sents=0,
2017-09-23 01:00:40 +00:00
use_gpu=-1, vectors=None, no_tagger=False, no_parser=False, no_entities=False,
2017-09-26 15:34:52 +00:00
gold_preproc=False, version="0.0.0", meta_path=None):
"""
Train a model. Expects data in spaCy's JSON format.
"""
2017-06-03 18:28:20 +00:00
util.set_env_log(True)
n_sents = n_sents or None
2017-05-07 21:25:29 +00:00
output_path = util.ensure_path(output_dir)
train_path = util.ensure_path(train_data)
dev_path = util.ensure_path(dev_data)
meta_path = util.ensure_path(meta_path)
2017-05-07 21:25:29 +00:00
if not output_path.exists():
2017-06-05 01:18:37 +00:00
output_path.mkdir()
2017-05-07 21:25:29 +00:00
if not train_path.exists():
prints(train_path, title="Training data not found", exits=1)
2017-05-07 21:25:29 +00:00
if dev_path and not dev_path.exists():
prints(dev_path, title="Development data not found", exits=1)
if meta_path is not None and not meta_path.exists():
prints(meta_path, title="meta.json not found", exits=1)
meta = util.read_json(meta_path) if meta_path else {}
if not isinstance(meta, dict):
prints("Expected dict but got: {}".format(type(meta)),
title="Not a valid meta.json format", exits=1)
2017-03-23 10:08:41 +00:00
pipeline = ['tagger', 'parser', 'ner']
if no_tagger and 'tagger' in pipeline: pipeline.remove('tagger')
if no_parser and 'parser' in pipeline: pipeline.remove('parser')
if no_entities and 'ner' in pipeline: pipeline.remove('ner')
2017-03-23 10:08:41 +00:00
2017-05-25 21:16:30 +00:00
# Take dropout and batch size as generators of values -- dropout
# starts high and decays sharply, to force the optimizer to explore.
# Batch size starts at 1 and grows, so that we make updates quickly
# at the beginning of training.
2017-10-08 07:08:12 +00:00
dropout_rates = util.decaying(util.env_opt('dropout_from', 0.2),
util.env_opt('dropout_to', 0.2),
util.env_opt('dropout_decay', 0.0))
2017-05-25 21:16:30 +00:00
batch_sizes = util.compounding(util.env_opt('batch_from', 1),
2017-10-08 07:08:12 +00:00
util.env_opt('batch_to', 16),
2017-05-25 21:16:30 +00:00
util.env_opt('batch_compound', 1.001))
2017-05-22 15:41:39 +00:00
corpus = GoldCorpus(train_path, dev_path, limit=n_sents)
2017-06-05 01:18:37 +00:00
n_train_words = corpus.count_train()
2017-03-23 10:08:41 +00:00
2017-09-23 01:00:40 +00:00
lang_class = util.get_lang_class(lang)
nlp = lang_class()
2017-09-23 01:00:40 +00:00
if vectors:
util.load_model(vectors, vocab=nlp.vocab)
for name in pipeline:
nlp.add_pipe(nlp.create_pipe(name), name=name)
2017-06-03 21:10:23 +00:00
optimizer = nlp.begin_training(lambda: corpus.train_tuples, device=use_gpu)
2017-09-02 17:46:01 +00:00
nlp._optimizer = None
2017-05-25 21:16:30 +00:00
2017-09-28 13:05:31 +00:00
print("Itn.\tP.Loss\tN.Loss\tUAS\tNER P.\tNER R.\tNER F.\tTag %\tToken %")
2017-05-26 10:52:09 +00:00
try:
2017-09-23 01:00:40 +00:00
train_docs = corpus.train_docs(nlp, projectivize=True, noise_level=0.0,
gold_preproc=gold_preproc, max_length=0)
train_docs = list(train_docs)
2017-05-26 10:52:09 +00:00
for i in range(n_iter):
2017-06-05 01:18:37 +00:00
with tqdm.tqdm(total=n_train_words, leave=False) as pbar:
2017-05-26 10:52:09 +00:00
losses = {}
for batch in minibatch(train_docs, size=batch_sizes):
docs, golds = zip(*batch)
nlp.update(docs, golds, sgd=optimizer,
2017-09-24 10:00:37 +00:00
drop=next(dropout_rates), losses=losses)
2017-06-05 01:18:37 +00:00
pbar.update(sum(len(doc) for doc in docs))
2017-05-25 21:16:30 +00:00
2017-05-26 10:52:09 +00:00
with nlp.use_params(optimizer.averages):
2017-06-03 18:28:20 +00:00
util.set_env_log(False)
epoch_model_path = output_path / ('model%d' % i)
nlp.to_disk(epoch_model_path)
nlp_loaded = util.load_model_from_path(epoch_model_path)
2017-10-09 13:05:37 +00:00
dev_docs = list(corpus.dev_docs(
2017-06-03 18:28:20 +00:00
nlp_loaded,
2017-10-09 13:05:37 +00:00
gold_preproc=gold_preproc))
nwords = sum(len(doc_gold[0]) for doc_gold in dev_docs)
start_time = timer()
scorer = nlp_loaded.evaluate(dev_docs)
end_time = timer()
if use_gpu < 0:
gpu_wps = None
cpu_wps = nwords/(end_time-start_time)
else:
gpu_wps = nwords/(end_time-start_time)
with Model.use_device('cpu'):
nlp_loaded = util.load_model_from_path(epoch_model_path)
2017-10-09 13:05:37 +00:00
dev_docs = list(corpus.dev_docs(
nlp_loaded, gold_preproc=gold_preproc))
start_time = timer()
scorer = nlp_loaded.evaluate(dev_docs)
end_time = timer()
cpu_wps = nwords/(end_time-start_time)
2017-06-05 01:18:37 +00:00
acc_loc =(output_path / ('model%d' % i) / 'accuracy.json')
with acc_loc.open('w') as file_:
file_.write(json_dumps(scorer.scores))
meta_loc = output_path / ('model%d' % i) / 'meta.json'
meta['accuracy'] = scorer.scores
2017-10-09 13:05:37 +00:00
meta['speed'] = {'nwords': nwords, 'cpu':cpu_wps, 'gpu': gpu_wps}
meta['lang'] = nlp.lang
meta['pipeline'] = pipeline
meta['spacy_version'] = '>=%s' % about.__version__
meta.setdefault('name', 'model%d' % i)
2017-09-26 15:34:52 +00:00
meta.setdefault('version', version)
with meta_loc.open('w') as file_:
file_.write(json_dumps(meta))
2017-06-03 18:28:20 +00:00
util.set_env_log(True)
2017-10-09 13:05:37 +00:00
print_progress(i, losses, scorer.scores, cpu_wps=cpu_wps, gpu_wps=gpu_wps)
2017-05-26 10:52:09 +00:00
finally:
print("Saving model...")
2017-10-05 13:12:50 +00:00
try:
with (output_path / 'model-final.pickle').open('wb') as file_:
with nlp.use_params(optimizer.averages):
dill.dump(nlp, file_, -1)
except:
print("Error saving model")
def _render_parses(i, to_render):
to_render[0].user_data['title'] = "Batch %d" % i
with Path('/tmp/entities.html').open('w') as file_:
html = displacy.render(to_render[:5], style='ent', page=True)
file_.write(html)
with Path('/tmp/parses.html').open('w') as file_:
html = displacy.render(to_render[:5], style='dep', page=True)
file_.write(html)
2017-03-23 10:08:41 +00:00
2017-10-09 13:05:37 +00:00
def print_progress(itn, losses, dev_scores, cpu_wps=0.0, gpu_wps=0.0):
scores = {}
2017-05-22 15:41:39 +00:00
for col in ['dep_loss', 'tag_loss', 'uas', 'tags_acc', 'token_acc',
2017-10-09 13:05:37 +00:00
'ents_p', 'ents_r', 'ents_f', 'cpu_wps', 'gpu_wps']:
scores[col] = 0.0
2017-05-25 01:10:20 +00:00
scores['dep_loss'] = losses.get('parser', 0.0)
2017-09-28 13:05:31 +00:00
scores['ner_loss'] = losses.get('ner', 0.0)
2017-05-25 01:10:20 +00:00
scores['tag_loss'] = losses.get('tagger', 0.0)
scores.update(dev_scores)
2017-10-09 13:05:37 +00:00
scores['cpu_wps'] = cpu_wps
scores['gpu_wps'] = gpu_wps or 0.0
2017-05-22 15:41:39 +00:00
tpl = '\t'.join((
'{:d}',
'{dep_loss:.3f}',
2017-09-28 13:05:31 +00:00
'{ner_loss:.3f}',
2017-05-22 15:41:39 +00:00
'{uas:.3f}',
'{ents_p:.3f}',
'{ents_r:.3f}',
'{ents_f:.3f}',
'{tags_acc:.3f}',
'{token_acc:.3f}',
2017-10-09 13:05:37 +00:00
'{cpu_wps:.1f}',
'{gpu_wps:.1f}',
))
print(tpl.format(itn, **scores))
2017-03-23 10:08:41 +00:00
def print_results(scorer):
2017-03-26 12:16:52 +00:00
results = {
'TOK': '%.2f' % scorer.token_acc,
'POS': '%.2f' % scorer.tags_acc,
'UAS': '%.2f' % scorer.uas,
'LAS': '%.2f' % scorer.las,
'NER P': '%.2f' % scorer.ents_p,
'NER R': '%.2f' % scorer.ents_r,
'NER F': '%.2f' % scorer.ents_f}
2017-03-23 10:08:41 +00:00
util.print_table(results, title="Results")