|
|
@ -16,6 +16,7 @@ import json
|
|
|
|
import os
|
|
|
|
import os
|
|
|
|
import time
|
|
|
|
import time
|
|
|
|
from collections import defaultdict
|
|
|
|
from collections import defaultdict
|
|
|
|
|
|
|
|
from collections import OrderedDict
|
|
|
|
from contextlib import nullcontext
|
|
|
|
from contextlib import nullcontext
|
|
|
|
from typing import Optional
|
|
|
|
from typing import Optional
|
|
|
|
|
|
|
|
|
|
|
@ -23,21 +24,18 @@ import jsonlines
|
|
|
|
import numpy as np
|
|
|
|
import numpy as np
|
|
|
|
import paddle
|
|
|
|
import paddle
|
|
|
|
from paddle import distributed as dist
|
|
|
|
from paddle import distributed as dist
|
|
|
|
from paddle.io import DataLoader
|
|
|
|
|
|
|
|
from yacs.config import CfgNode
|
|
|
|
from yacs.config import CfgNode
|
|
|
|
|
|
|
|
|
|
|
|
from paddlespeech.s2t.io.collator import SpeechCollator
|
|
|
|
from paddlespeech.s2t.frontend.featurizer import TextFeaturizer
|
|
|
|
from paddlespeech.s2t.io.collator import TripletSpeechCollator
|
|
|
|
from paddlespeech.s2t.io.dataloader import BatchDataLoader
|
|
|
|
from paddlespeech.s2t.io.dataset import ManifestDataset
|
|
|
|
|
|
|
|
from paddlespeech.s2t.io.sampler import SortagradBatchSampler
|
|
|
|
|
|
|
|
from paddlespeech.s2t.io.sampler import SortagradDistributedBatchSampler
|
|
|
|
|
|
|
|
from paddlespeech.s2t.models.u2_st import U2STModel
|
|
|
|
from paddlespeech.s2t.models.u2_st import U2STModel
|
|
|
|
from paddlespeech.s2t.training.gradclip import ClipGradByGlobalNormWithLog
|
|
|
|
from paddlespeech.s2t.training.optimizer import OptimizerFactory
|
|
|
|
from paddlespeech.s2t.training.scheduler import WarmupLR
|
|
|
|
from paddlespeech.s2t.training.reporter import ObsScope
|
|
|
|
|
|
|
|
from paddlespeech.s2t.training.reporter import report
|
|
|
|
|
|
|
|
from paddlespeech.s2t.training.scheduler import LRSchedulerFactory
|
|
|
|
from paddlespeech.s2t.training.timer import Timer
|
|
|
|
from paddlespeech.s2t.training.timer import Timer
|
|
|
|
from paddlespeech.s2t.training.trainer import Trainer
|
|
|
|
from paddlespeech.s2t.training.trainer import Trainer
|
|
|
|
from paddlespeech.s2t.utils import bleu_score
|
|
|
|
from paddlespeech.s2t.utils import bleu_score
|
|
|
|
from paddlespeech.s2t.utils import ctc_utils
|
|
|
|
|
|
|
|
from paddlespeech.s2t.utils import layer_tools
|
|
|
|
from paddlespeech.s2t.utils import layer_tools
|
|
|
|
from paddlespeech.s2t.utils import mp_tools
|
|
|
|
from paddlespeech.s2t.utils import mp_tools
|
|
|
|
from paddlespeech.s2t.utils.log import Log
|
|
|
|
from paddlespeech.s2t.utils.log import Log
|
|
|
@ -96,6 +94,8 @@ class U2STTrainer(Trainer):
|
|
|
|
# loss div by `batch_size * accum_grad`
|
|
|
|
# loss div by `batch_size * accum_grad`
|
|
|
|
loss /= train_conf.accum_grad
|
|
|
|
loss /= train_conf.accum_grad
|
|
|
|
losses_np = {'loss': float(loss) * train_conf.accum_grad}
|
|
|
|
losses_np = {'loss': float(loss) * train_conf.accum_grad}
|
|
|
|
|
|
|
|
if st_loss:
|
|
|
|
|
|
|
|
losses_np['st_loss'] = float(st_loss)
|
|
|
|
if attention_loss:
|
|
|
|
if attention_loss:
|
|
|
|
losses_np['att_loss'] = float(attention_loss)
|
|
|
|
losses_np['att_loss'] = float(attention_loss)
|
|
|
|
if ctc_loss:
|
|
|
|
if ctc_loss:
|
|
|
@ -125,6 +125,12 @@ class U2STTrainer(Trainer):
|
|
|
|
|
|
|
|
|
|
|
|
iteration_time = time.time() - start
|
|
|
|
iteration_time = time.time() - start
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
for k, v in losses_np.items():
|
|
|
|
|
|
|
|
report(k, v)
|
|
|
|
|
|
|
|
report("batch_size", self.config.collator.batch_size)
|
|
|
|
|
|
|
|
report("accum", train_conf.accum_grad)
|
|
|
|
|
|
|
|
report("step_cost", iteration_time)
|
|
|
|
|
|
|
|
|
|
|
|
if (batch_index + 1) % train_conf.log_interval == 0:
|
|
|
|
if (batch_index + 1) % train_conf.log_interval == 0:
|
|
|
|
msg += "train time: {:>.3f}s, ".format(iteration_time)
|
|
|
|
msg += "train time: {:>.3f}s, ".format(iteration_time)
|
|
|
|
msg += "batch size: {}, ".format(self.config.collator.batch_size)
|
|
|
|
msg += "batch size: {}, ".format(self.config.collator.batch_size)
|
|
|
@ -204,16 +210,34 @@ class U2STTrainer(Trainer):
|
|
|
|
data_start_time = time.time()
|
|
|
|
data_start_time = time.time()
|
|
|
|
for batch_index, batch in enumerate(self.train_loader):
|
|
|
|
for batch_index, batch in enumerate(self.train_loader):
|
|
|
|
dataload_time = time.time() - data_start_time
|
|
|
|
dataload_time = time.time() - data_start_time
|
|
|
|
msg = "Train: Rank: {}, ".format(dist.get_rank())
|
|
|
|
msg = "Train:"
|
|
|
|
msg += "epoch: {}, ".format(self.epoch)
|
|
|
|
observation = OrderedDict()
|
|
|
|
msg += "step: {}, ".format(self.iteration)
|
|
|
|
with ObsScope(observation):
|
|
|
|
msg += "batch : {}/{}, ".format(batch_index + 1,
|
|
|
|
report("Rank", dist.get_rank())
|
|
|
|
len(self.train_loader))
|
|
|
|
report("epoch", self.epoch)
|
|
|
|
msg += "lr: {:>.8f}, ".format(self.lr_scheduler())
|
|
|
|
report('step', self.iteration)
|
|
|
|
msg += "data time: {:>.3f}s, ".format(dataload_time)
|
|
|
|
report("lr", self.lr_scheduler())
|
|
|
|
self.train_batch(batch_index, batch, msg)
|
|
|
|
self.train_batch(batch_index, batch, msg)
|
|
|
|
self.after_train_batch()
|
|
|
|
self.after_train_batch()
|
|
|
|
data_start_time = time.time()
|
|
|
|
report('iter', batch_index + 1)
|
|
|
|
|
|
|
|
report('total', len(self.train_loader))
|
|
|
|
|
|
|
|
report('reader_cost', dataload_time)
|
|
|
|
|
|
|
|
observation['batch_cost'] = observation[
|
|
|
|
|
|
|
|
'reader_cost'] + observation['step_cost']
|
|
|
|
|
|
|
|
observation['samples'] = observation['batch_size']
|
|
|
|
|
|
|
|
observation['ips,sent./sec'] = observation[
|
|
|
|
|
|
|
|
'batch_size'] / observation['batch_cost']
|
|
|
|
|
|
|
|
for k, v in observation.items():
|
|
|
|
|
|
|
|
msg += f" {k.split(',')[0]}: "
|
|
|
|
|
|
|
|
msg += f"{v:>.8f}" if isinstance(v,
|
|
|
|
|
|
|
|
float) else f"{v}"
|
|
|
|
|
|
|
|
msg += f" {k.split(',')[1]}" if len(
|
|
|
|
|
|
|
|
k.split(',')) == 2 else ""
|
|
|
|
|
|
|
|
msg += ","
|
|
|
|
|
|
|
|
msg = msg[:-1] # remove the last ","
|
|
|
|
|
|
|
|
if (batch_index + 1
|
|
|
|
|
|
|
|
) % self.config.training.log_interval == 0:
|
|
|
|
|
|
|
|
logger.info(msg)
|
|
|
|
except Exception as e:
|
|
|
|
except Exception as e:
|
|
|
|
logger.error(e)
|
|
|
|
logger.error(e)
|
|
|
|
raise e
|
|
|
|
raise e
|
|
|
@ -244,95 +268,87 @@ class U2STTrainer(Trainer):
|
|
|
|
|
|
|
|
|
|
|
|
def setup_dataloader(self):
|
|
|
|
def setup_dataloader(self):
|
|
|
|
config = self.config.clone()
|
|
|
|
config = self.config.clone()
|
|
|
|
config.defrost()
|
|
|
|
|
|
|
|
config.collator.keep_transcription_text = False
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# train/valid dataset, return token ids
|
|
|
|
|
|
|
|
config.data.manifest = config.data.train_manifest
|
|
|
|
|
|
|
|
train_dataset = ManifestDataset.from_config(config)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
config.data.manifest = config.data.dev_manifest
|
|
|
|
load_transcript = True if config.model.model_conf.asr_weight > 0 else False
|
|
|
|
dev_dataset = ManifestDataset.from_config(config)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if config.model.model_conf.asr_weight > 0.:
|
|
|
|
if self.train:
|
|
|
|
Collator = TripletSpeechCollator
|
|
|
|
# train/valid dataset, return token ids
|
|
|
|
TestCollator = SpeechCollator
|
|
|
|
self.train_loader = BatchDataLoader(
|
|
|
|
else:
|
|
|
|
json_file=config.data.train_manifest,
|
|
|
|
TestCollator = Collator = SpeechCollator
|
|
|
|
train_mode=True,
|
|
|
|
|
|
|
|
sortagrad=False,
|
|
|
|
collate_fn_train = Collator.from_config(config)
|
|
|
|
|
|
|
|
config.collator.augmentation_config = ""
|
|
|
|
|
|
|
|
collate_fn_dev = Collator.from_config(config)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if self.parallel:
|
|
|
|
|
|
|
|
batch_sampler = SortagradDistributedBatchSampler(
|
|
|
|
|
|
|
|
train_dataset,
|
|
|
|
|
|
|
|
batch_size=config.collator.batch_size,
|
|
|
|
|
|
|
|
num_replicas=None,
|
|
|
|
|
|
|
|
rank=None,
|
|
|
|
|
|
|
|
shuffle=True,
|
|
|
|
|
|
|
|
drop_last=True,
|
|
|
|
|
|
|
|
sortagrad=config.collator.sortagrad,
|
|
|
|
|
|
|
|
shuffle_method=config.collator.shuffle_method)
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
|
|
|
batch_sampler = SortagradBatchSampler(
|
|
|
|
|
|
|
|
train_dataset,
|
|
|
|
|
|
|
|
shuffle=True,
|
|
|
|
|
|
|
|
batch_size=config.collator.batch_size,
|
|
|
|
batch_size=config.collator.batch_size,
|
|
|
|
drop_last=True,
|
|
|
|
maxlen_in=config.collator.maxlen_in,
|
|
|
|
sortagrad=config.collator.sortagrad,
|
|
|
|
maxlen_out=config.collator.maxlen_out,
|
|
|
|
shuffle_method=config.collator.shuffle_method)
|
|
|
|
minibatches=0,
|
|
|
|
self.train_loader = DataLoader(
|
|
|
|
mini_batch_size=1,
|
|
|
|
train_dataset,
|
|
|
|
batch_count='auto',
|
|
|
|
batch_sampler=batch_sampler,
|
|
|
|
batch_bins=0,
|
|
|
|
collate_fn=collate_fn_train,
|
|
|
|
batch_frames_in=0,
|
|
|
|
num_workers=config.collator.num_workers, )
|
|
|
|
batch_frames_out=0,
|
|
|
|
self.valid_loader = DataLoader(
|
|
|
|
batch_frames_inout=0,
|
|
|
|
dev_dataset,
|
|
|
|
preprocess_conf=config.collator.
|
|
|
|
|
|
|
|
augmentation_config, # aug will be off when train_mode=False
|
|
|
|
|
|
|
|
n_iter_processes=config.collator.num_workers,
|
|
|
|
|
|
|
|
subsampling_factor=1,
|
|
|
|
|
|
|
|
load_aux_output=load_transcript,
|
|
|
|
|
|
|
|
num_encs=1)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
self.valid_loader = BatchDataLoader(
|
|
|
|
|
|
|
|
json_file=config.data.dev_manifest,
|
|
|
|
|
|
|
|
train_mode=False,
|
|
|
|
|
|
|
|
sortagrad=False,
|
|
|
|
batch_size=config.collator.batch_size,
|
|
|
|
batch_size=config.collator.batch_size,
|
|
|
|
shuffle=False,
|
|
|
|
maxlen_in=float('inf'),
|
|
|
|
drop_last=False,
|
|
|
|
maxlen_out=float('inf'),
|
|
|
|
collate_fn=collate_fn_dev,
|
|
|
|
minibatches=0,
|
|
|
|
num_workers=config.collator.num_workers, )
|
|
|
|
mini_batch_size=1,
|
|
|
|
|
|
|
|
batch_count='auto',
|
|
|
|
|
|
|
|
batch_bins=0,
|
|
|
|
|
|
|
|
batch_frames_in=0,
|
|
|
|
|
|
|
|
batch_frames_out=0,
|
|
|
|
|
|
|
|
batch_frames_inout=0,
|
|
|
|
|
|
|
|
preprocess_conf=config.collator.
|
|
|
|
|
|
|
|
augmentation_config, # aug will be off when train_mode=False
|
|
|
|
|
|
|
|
n_iter_processes=config.collator.num_workers,
|
|
|
|
|
|
|
|
subsampling_factor=1,
|
|
|
|
|
|
|
|
load_aux_output=load_transcript,
|
|
|
|
|
|
|
|
num_encs=1)
|
|
|
|
|
|
|
|
logger.info("Setup train/valid Dataloader!")
|
|
|
|
|
|
|
|
else:
|
|
|
|
# test dataset, return raw text
|
|
|
|
# test dataset, return raw text
|
|
|
|
config.data.manifest = config.data.test_manifest
|
|
|
|
self.test_loader = BatchDataLoader(
|
|
|
|
# filter test examples, will cause less examples, but no mismatch with training
|
|
|
|
json_file=config.data.test_manifest,
|
|
|
|
# and can use large batch size , save training time, so filter test egs now.
|
|
|
|
train_mode=False,
|
|
|
|
# config.data.min_input_len = 0.0 # second
|
|
|
|
sortagrad=False,
|
|
|
|
# config.data.max_input_len = float('inf') # second
|
|
|
|
|
|
|
|
# config.data.min_output_len = 0.0 # tokens
|
|
|
|
|
|
|
|
# config.data.max_output_len = float('inf') # tokens
|
|
|
|
|
|
|
|
# config.data.min_output_input_ratio = 0.00
|
|
|
|
|
|
|
|
# config.data.max_output_input_ratio = float('inf')
|
|
|
|
|
|
|
|
test_dataset = ManifestDataset.from_config(config)
|
|
|
|
|
|
|
|
# return text ord id
|
|
|
|
|
|
|
|
config.collator.keep_transcription_text = True
|
|
|
|
|
|
|
|
config.collator.augmentation_config = ""
|
|
|
|
|
|
|
|
self.test_loader = DataLoader(
|
|
|
|
|
|
|
|
test_dataset,
|
|
|
|
|
|
|
|
batch_size=config.decoding.batch_size,
|
|
|
|
|
|
|
|
shuffle=False,
|
|
|
|
|
|
|
|
drop_last=False,
|
|
|
|
|
|
|
|
collate_fn=TestCollator.from_config(config),
|
|
|
|
|
|
|
|
num_workers=config.collator.num_workers, )
|
|
|
|
|
|
|
|
# return text token id
|
|
|
|
|
|
|
|
config.collator.keep_transcription_text = False
|
|
|
|
|
|
|
|
self.align_loader = DataLoader(
|
|
|
|
|
|
|
|
test_dataset,
|
|
|
|
|
|
|
|
batch_size=config.decoding.batch_size,
|
|
|
|
batch_size=config.decoding.batch_size,
|
|
|
|
shuffle=False,
|
|
|
|
maxlen_in=float('inf'),
|
|
|
|
drop_last=False,
|
|
|
|
maxlen_out=float('inf'),
|
|
|
|
collate_fn=TestCollator.from_config(config),
|
|
|
|
minibatches=0,
|
|
|
|
num_workers=config.collator.num_workers, )
|
|
|
|
mini_batch_size=1,
|
|
|
|
logger.info("Setup train/valid/test/align Dataloader!")
|
|
|
|
batch_count='auto',
|
|
|
|
|
|
|
|
batch_bins=0,
|
|
|
|
|
|
|
|
batch_frames_in=0,
|
|
|
|
|
|
|
|
batch_frames_out=0,
|
|
|
|
|
|
|
|
batch_frames_inout=0,
|
|
|
|
|
|
|
|
preprocess_conf=config.collator.
|
|
|
|
|
|
|
|
augmentation_config, # aug will be off when train_mode=False
|
|
|
|
|
|
|
|
n_iter_processes=config.collator.num_workers,
|
|
|
|
|
|
|
|
subsampling_factor=1,
|
|
|
|
|
|
|
|
num_encs=1)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
logger.info("Setup test Dataloader!")
|
|
|
|
|
|
|
|
|
|
|
|
def setup_model(self):
|
|
|
|
def setup_model(self):
|
|
|
|
config = self.config
|
|
|
|
config = self.config
|
|
|
|
model_conf = config.model
|
|
|
|
model_conf = config.model
|
|
|
|
with UpdateConfig(model_conf):
|
|
|
|
with UpdateConfig(model_conf):
|
|
|
|
model_conf.input_dim = self.train_loader.collate_fn.feature_size
|
|
|
|
if self.train:
|
|
|
|
model_conf.output_dim = self.train_loader.collate_fn.vocab_size
|
|
|
|
model_conf.input_dim = self.train_loader.feat_dim
|
|
|
|
|
|
|
|
model_conf.output_dim = self.train_loader.vocab_size
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
|
|
|
model_conf.input_dim = self.test_loader.feat_dim
|
|
|
|
|
|
|
|
model_conf.output_dim = self.test_loader.vocab_size
|
|
|
|
|
|
|
|
|
|
|
|
model = U2STModel.from_config(model_conf)
|
|
|
|
model = U2STModel.from_config(model_conf)
|
|
|
|
|
|
|
|
|
|
|
@ -348,35 +364,38 @@ class U2STTrainer(Trainer):
|
|
|
|
scheduler_type = train_config.scheduler
|
|
|
|
scheduler_type = train_config.scheduler
|
|
|
|
scheduler_conf = train_config.scheduler_conf
|
|
|
|
scheduler_conf = train_config.scheduler_conf
|
|
|
|
|
|
|
|
|
|
|
|
if scheduler_type == 'expdecaylr':
|
|
|
|
scheduler_args = {
|
|
|
|
lr_scheduler = paddle.optimizer.lr.ExponentialDecay(
|
|
|
|
"learning_rate": optim_conf.lr,
|
|
|
|
learning_rate=optim_conf.lr,
|
|
|
|
"verbose": False,
|
|
|
|
gamma=scheduler_conf.lr_decay,
|
|
|
|
"warmup_steps": scheduler_conf.warmup_steps,
|
|
|
|
verbose=False)
|
|
|
|
"gamma": scheduler_conf.lr_decay,
|
|
|
|
elif scheduler_type == 'warmuplr':
|
|
|
|
"d_model": model_conf.encoder_conf.output_size,
|
|
|
|
lr_scheduler = WarmupLR(
|
|
|
|
}
|
|
|
|
learning_rate=optim_conf.lr,
|
|
|
|
lr_scheduler = LRSchedulerFactory.from_args(scheduler_type,
|
|
|
|
warmup_steps=scheduler_conf.warmup_steps,
|
|
|
|
scheduler_args)
|
|
|
|
verbose=False)
|
|
|
|
|
|
|
|
elif scheduler_type == 'noam':
|
|
|
|
def optimizer_args(
|
|
|
|
lr_scheduler = paddle.optimizer.lr.NoamDecay(
|
|
|
|
config,
|
|
|
|
learning_rate=optim_conf.lr,
|
|
|
|
parameters,
|
|
|
|
d_model=model_conf.encoder_conf.output_size,
|
|
|
|
lr_scheduler=None, ):
|
|
|
|
warmup_steps=scheduler_conf.warmup_steps,
|
|
|
|
train_config = config.training
|
|
|
|
verbose=False)
|
|
|
|
optim_type = train_config.optim
|
|
|
|
else:
|
|
|
|
optim_conf = train_config.optim_conf
|
|
|
|
raise ValueError(f"Not support scheduler: {scheduler_type}")
|
|
|
|
scheduler_type = train_config.scheduler
|
|
|
|
|
|
|
|
scheduler_conf = train_config.scheduler_conf
|
|
|
|
grad_clip = ClipGradByGlobalNormWithLog(train_config.global_grad_clip)
|
|
|
|
return {
|
|
|
|
weight_decay = paddle.regularizer.L2Decay(optim_conf.weight_decay)
|
|
|
|
"grad_clip": train_config.global_grad_clip,
|
|
|
|
if optim_type == 'adam':
|
|
|
|
"weight_decay": optim_conf.weight_decay,
|
|
|
|
optimizer = paddle.optimizer.Adam(
|
|
|
|
"learning_rate": lr_scheduler
|
|
|
|
learning_rate=lr_scheduler,
|
|
|
|
if lr_scheduler else optim_conf.lr,
|
|
|
|
parameters=model.parameters(),
|
|
|
|
"parameters": parameters,
|
|
|
|
weight_decay=weight_decay,
|
|
|
|
"epsilon": 1e-9 if optim_type == 'noam' else None,
|
|
|
|
grad_clip=grad_clip)
|
|
|
|
"beta1": 0.9 if optim_type == 'noam' else None,
|
|
|
|
else:
|
|
|
|
"beat2": 0.98 if optim_type == 'noam' else None,
|
|
|
|
raise ValueError(f"Not support optim: {optim_type}")
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
optimzer_args = optimizer_args(config, model.parameters(), lr_scheduler)
|
|
|
|
|
|
|
|
optimizer = OptimizerFactory.from_args(optim_type, optimzer_args)
|
|
|
|
|
|
|
|
|
|
|
|
self.model = model
|
|
|
|
self.model = model
|
|
|
|
self.optimizer = optimizer
|
|
|
|
self.optimizer = optimizer
|
|
|
@ -416,26 +435,30 @@ class U2STTester(U2STTrainer):
|
|
|
|
|
|
|
|
|
|
|
|
def __init__(self, config, args):
|
|
|
|
def __init__(self, config, args):
|
|
|
|
super().__init__(config, args)
|
|
|
|
super().__init__(config, args)
|
|
|
|
|
|
|
|
self.text_feature = TextFeaturizer(
|
|
|
|
|
|
|
|
unit_type=self.config.collator.unit_type,
|
|
|
|
|
|
|
|
vocab_filepath=self.config.collator.vocab_filepath,
|
|
|
|
|
|
|
|
spm_model_prefix=self.config.collator.spm_model_prefix)
|
|
|
|
|
|
|
|
self.vocab_list = self.text_feature.vocab_list
|
|
|
|
|
|
|
|
|
|
|
|
def ordid2token(self, texts, texts_len):
|
|
|
|
def id2token(self, texts, texts_len, text_feature):
|
|
|
|
""" ord() id to chr() chr """
|
|
|
|
""" ord() id to chr() chr """
|
|
|
|
trans = []
|
|
|
|
trans = []
|
|
|
|
for text, n in zip(texts, texts_len):
|
|
|
|
for text, n in zip(texts, texts_len):
|
|
|
|
n = n.numpy().item()
|
|
|
|
n = n.numpy().item()
|
|
|
|
ids = text[:n]
|
|
|
|
ids = text[:n]
|
|
|
|
trans.append(''.join([chr(i) for i in ids]))
|
|
|
|
trans.append(text_feature.defeaturize(ids.numpy().tolist()))
|
|
|
|
return trans
|
|
|
|
return trans
|
|
|
|
|
|
|
|
|
|
|
|
def translate(self, audio, audio_len):
|
|
|
|
def translate(self, audio, audio_len):
|
|
|
|
""""E2E translation from extracted audio feature"""
|
|
|
|
""""E2E translation from extracted audio feature"""
|
|
|
|
cfg = self.config.decoding
|
|
|
|
cfg = self.config.decoding
|
|
|
|
text_feature = self.test_loader.collate_fn.text_feature
|
|
|
|
|
|
|
|
self.model.eval()
|
|
|
|
self.model.eval()
|
|
|
|
|
|
|
|
|
|
|
|
hyps = self.model.decode(
|
|
|
|
hyps = self.model.decode(
|
|
|
|
audio,
|
|
|
|
audio,
|
|
|
|
audio_len,
|
|
|
|
audio_len,
|
|
|
|
text_feature=text_feature,
|
|
|
|
text_feature=self.text_feature,
|
|
|
|
decoding_method=cfg.decoding_method,
|
|
|
|
decoding_method=cfg.decoding_method,
|
|
|
|
beam_size=cfg.beam_size,
|
|
|
|
beam_size=cfg.beam_size,
|
|
|
|
word_reward=cfg.word_reward,
|
|
|
|
word_reward=cfg.word_reward,
|
|
|
@ -456,23 +479,20 @@ class U2STTester(U2STTrainer):
|
|
|
|
len_refs, num_ins = 0, 0
|
|
|
|
len_refs, num_ins = 0, 0
|
|
|
|
|
|
|
|
|
|
|
|
start_time = time.time()
|
|
|
|
start_time = time.time()
|
|
|
|
text_feature = self.test_loader.collate_fn.text_feature
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
refs = [
|
|
|
|
refs = self.id2token(texts, texts_len, self.text_feature)
|
|
|
|
"".join(chr(t) for t in text[:text_len])
|
|
|
|
|
|
|
|
for text, text_len in zip(texts, texts_len)
|
|
|
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
hyps = self.model.decode(
|
|
|
|
hyps = self.model.decode(
|
|
|
|
audio,
|
|
|
|
audio,
|
|
|
|
audio_len,
|
|
|
|
audio_len,
|
|
|
|
text_feature=text_feature,
|
|
|
|
text_feature=self.text_feature,
|
|
|
|
decoding_method=cfg.decoding_method,
|
|
|
|
decoding_method=cfg.decoding_method,
|
|
|
|
beam_size=cfg.beam_size,
|
|
|
|
beam_size=cfg.beam_size,
|
|
|
|
word_reward=cfg.word_reward,
|
|
|
|
word_reward=cfg.word_reward,
|
|
|
|
decoding_chunk_size=cfg.decoding_chunk_size,
|
|
|
|
decoding_chunk_size=cfg.decoding_chunk_size,
|
|
|
|
num_decoding_left_chunks=cfg.num_decoding_left_chunks,
|
|
|
|
num_decoding_left_chunks=cfg.num_decoding_left_chunks,
|
|
|
|
simulate_streaming=cfg.simulate_streaming)
|
|
|
|
simulate_streaming=cfg.simulate_streaming)
|
|
|
|
|
|
|
|
|
|
|
|
decode_time = time.time() - start_time
|
|
|
|
decode_time = time.time() - start_time
|
|
|
|
|
|
|
|
|
|
|
|
for utt, target, result in zip(utts, refs, hyps):
|
|
|
|
for utt, target, result in zip(utts, refs, hyps):
|
|
|
@ -505,7 +525,7 @@ class U2STTester(U2STTrainer):
|
|
|
|
cfg = self.config.decoding
|
|
|
|
cfg = self.config.decoding
|
|
|
|
bleu_func = bleu_score.char_bleu if cfg.error_rate_type == 'char-bleu' else bleu_score.bleu
|
|
|
|
bleu_func = bleu_score.char_bleu if cfg.error_rate_type == 'char-bleu' else bleu_score.bleu
|
|
|
|
|
|
|
|
|
|
|
|
stride_ms = self.test_loader.collate_fn.stride_ms
|
|
|
|
stride_ms = self.config.collator.stride_ms
|
|
|
|
hyps, refs = [], []
|
|
|
|
hyps, refs = [], []
|
|
|
|
len_refs, num_ins = 0, 0
|
|
|
|
len_refs, num_ins = 0, 0
|
|
|
|
num_frames = 0.0
|
|
|
|
num_frames = 0.0
|
|
|
@ -522,7 +542,7 @@ class U2STTester(U2STTrainer):
|
|
|
|
len_refs += metrics['len_refs']
|
|
|
|
len_refs += metrics['len_refs']
|
|
|
|
num_ins += metrics['num_ins']
|
|
|
|
num_ins += metrics['num_ins']
|
|
|
|
rtf = num_time / (num_frames * stride_ms)
|
|
|
|
rtf = num_time / (num_frames * stride_ms)
|
|
|
|
logger.info("RTF: %f, BELU (%d) = %f" % (rtf, num_ins, bleu))
|
|
|
|
logger.info("RTF: %f, instance (%d), batch BELU = %f" % (rtf, num_ins, bleu))
|
|
|
|
|
|
|
|
|
|
|
|
rtf = num_time / (num_frames * stride_ms)
|
|
|
|
rtf = num_time / (num_frames * stride_ms)
|
|
|
|
msg = "Test: "
|
|
|
|
msg = "Test: "
|
|
|
@ -553,13 +573,6 @@ class U2STTester(U2STTrainer):
|
|
|
|
})
|
|
|
|
})
|
|
|
|
f.write(data + '\n')
|
|
|
|
f.write(data + '\n')
|
|
|
|
|
|
|
|
|
|
|
|
@paddle.no_grad()
|
|
|
|
|
|
|
|
def align(self):
|
|
|
|
|
|
|
|
ctc_utils.ctc_align(self.config, self.model, self.align_loader,
|
|
|
|
|
|
|
|
self.config.decoding.batch_size,
|
|
|
|
|
|
|
|
self.config.collator.stride_ms, self.vocab_list,
|
|
|
|
|
|
|
|
self.args.result_file)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def load_inferspec(self):
|
|
|
|
def load_inferspec(self):
|
|
|
|
"""infer model and input spec.
|
|
|
|
"""infer model and input spec.
|
|
|
|
|
|
|
|
|
|
|
@ -567,11 +580,11 @@ class U2STTester(U2STTrainer):
|
|
|
|
nn.Layer: inference model
|
|
|
|
nn.Layer: inference model
|
|
|
|
List[paddle.static.InputSpec]: input spec.
|
|
|
|
List[paddle.static.InputSpec]: input spec.
|
|
|
|
"""
|
|
|
|
"""
|
|
|
|
from paddlespeech.s2t.models.u2 import U2InferModel
|
|
|
|
from paddlespeech.s2t.models.u2_st import U2STInferModel
|
|
|
|
infer_model = U2InferModel.from_pretrained(self.test_loader,
|
|
|
|
infer_model = U2STInferModel.from_pretrained(self.test_loader,
|
|
|
|
self.config.model.clone(),
|
|
|
|
self.config.model.clone(),
|
|
|
|
self.args.checkpoint_path)
|
|
|
|
self.args.checkpoint_path)
|
|
|
|
feat_dim = self.test_loader.collate_fn.feature_size
|
|
|
|
feat_dim = self.test_loader.feat_dim
|
|
|
|
input_spec = [
|
|
|
|
input_spec = [
|
|
|
|
paddle.static.InputSpec(shape=[1, None, feat_dim],
|
|
|
|
paddle.static.InputSpec(shape=[1, None, feat_dim],
|
|
|
|
dtype='float32'), # audio, [B,T,D]
|
|
|
|
dtype='float32'), # audio, [B,T,D]
|
|
|
|