import argparse
import math
import pathlib

from recognizers.neural_networks.data import load_vocabulary_data

def main():

    parser = argparse.ArgumentParser()
    parser.add_argument('--architecture', choices=['transformer', 'rnn', 'lstm'], required=True)
    parser.add_argument('--parameter-budget', type=int, required=True)
    parser.add_argument('--training-data', type=pathlib.Path, required=True)
    parser.add_argument('--num-layers', type=int,
        help='(transformer, rnn, lstm) Number of layers.')
    parser.add_argument('--num-heads', type=int,
        help='(transformer) Number of attention heads per layer.')
    args = parser.parse_args()

    vocab = load_vocabulary_data(args, parser)
    vocab_size = len(vocab.tokens) + int(vocab.allow_unk)
    num_layers = args.num_layers if args.num_layers is not None else 5

    outputs = [
        '--architecture', args.architecture,
        '--num-layers', str(num_layers)
    ]

    if args.architecture == 'transformer':
        # num_params =
        #   vocab_size * d_model +    # embeddings
        #   num_layers * (
        #       d_model * 4 * (d_model + 1) +    # projection layers
        #       2 * feedforward_size * (d_model + 1) +    # feedforward layer
        #       4 * d_model    # layer norm
        #   ) +
        #   2 * d_model +    # final layer norm
        #   d_model + 1    # recognition head
        # d_model = size * num_heads
        # feedforward_size = feedforward_size_factor * d_model
        num_heads = args.num_heads if args.num_heads is not None else 8
        feedforward_size_factor = 4
        a = num_layers * (4 + 2 * feedforward_size_factor)
        b = vocab_size + num_layers * (8 + 2 * feedforward_size_factor) + 3
        c = 1 - args.parameter_budget
        d_model_float = (-b + math.sqrt(b * b - 4 * a * c)) / (2 * a)
        size_float = d_model_float / num_heads
        size = round(size_float)
        d_model = size * num_heads
        feedforward_size = d_model * feedforward_size_factor
        outputs.extend([
            '--d-model', str(d_model),
            '--num-heads', str(num_heads),
            '--feedforward-size', str(feedforward_size)
        ])
    elif args.architecture in ('rnn', 'lstm'):
        # RNN:
        # num_params =
        #   vocab_size * hidden_units +    # embeddings
        #   num_layers * hidden_units +    # initial hidden state
        #   num_layers * (
        #       hidden_units * (2 * hidden_units + 1)    # input/recurrent layers
        #   ) +
        #   hidden_units + 1    # recognition head
        # LSTM:
        # num_params =
        #   vocab_size * hidden_units +    # embeddings
        #   num_layers * hidden_units +    # initial hidden state
        #   num_layers * (
        #       4 * hidden_units * (2 * hidden_units + 1)    # input/recurrent layers
        #   ) +
        #   hidden_units + 1    # recognition head
        if args.architecture == 'rnn':
            a = 2 * num_layers
            b = vocab_size + 2 * num_layers + 1
        else:
            a = 8 * num_layers
            b = vocab_size + 5 * num_layers + 1
        c = 1 - args.parameter_budget
        hidden_units_float = (-b + math.sqrt(b * b - 4 * a * c)) / (2 * a)
        hidden_units = round(hidden_units_float)
        outputs.extend([
            '--hidden-units', str(hidden_units)
        ])
    else:
        raise NotImplementedError
    print(' '.join(outputs))

if __name__ == '__main__':
    main()
