#!/usr/bin/env node

/**
* @license Apache-2.0
*
* Copyright (c) 2018 The Stdlib Authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
*    http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/

'use strict';

// MODULES //

var proc = require( 'process' );
var resolve = require( 'path' ).resolve;
var readFileSync = require( '@stdlib/fs/read-file' ).sync;
var writeFileSync = require( '@stdlib/fs/write-file' ).sync;
var CLI = require( '@stdlib/cli/ctor' );
var stdout = require( '@stdlib/streams/node/stdout' );
var cwd = require( '@stdlib/process/cwd' );
var Uint8Array = require( '@stdlib/array/uint8' );
var Int32Array = require( '@stdlib/array/int32' );
var isUint8Array = require( '@stdlib/assert/is-uint8array' );
var isInteger = require( '@stdlib/assert/is-integer' ).isPrimitive;
var gcopy = require( '@stdlib/blas/base/gcopy' );
var array2buffer = require( '@stdlib/buffer/from-array' );
var randomStream = require( './../lib' );


// FUNCTIONS //

/**
* Callback invoked once a source stream ends.
*
* @private
*/
function onEnd() {
	// Append a trailing newline in accordance with standard POSIX behavior:
	console.log( '' ); // eslint-disable-line no-console
}

/**
* Attempts to load a PRNG state.
*
* @private
* @param {string} filepath - absolute path to file containing PRNG state
* @returns {(Int32Array|Error)} PRNG state or an error
*/
function loadState( filepath ) {
	var state;
	var len;

	state = readFileSync( filepath );
	if ( state instanceof Error ) {
		return state;
	}
	len = state.length;

	// For older Node.js environments, convert the `Buffer` to a `Uint8Array`...
	if ( !isUint8Array( state ) ) {
		state = gcopy( len, state, 1, new Uint8Array( len ), 1 );
	}
	// Create a PRNG state array "view":
	len /= Int32Array.BYTES_PER_ELEMENT;
	if ( !isInteger( len ) ) {
		return new RangeError( 'invalid option. `state` has an invalid length.' );
	}
	return new Int32Array( state.buffer, state.byteOffset, len );
}


// MAIN //

/**
* Main execution sequence.
*
* @private
* @returns {void}
*/
function main() {
	var stream;
	var flags;
	var opts;
	var cli;
	var dir;
	var i;

	// Create a command-line interface:
	cli = new CLI({
		'pkg': require( './../package.json' ),
		'options': require( './../etc/cli_opts.json' ),
		'help': readFileSync( resolve( __dirname, '..', 'docs', 'usage.txt' ), {
			'encoding': 'utf8'
		})
	});

	// Get any provided command-line options:
	flags = cli.flags();
	if ( flags.help || flags.version ) {
		return;
	}

	// Get the current working directory:
	dir = cwd();

	// Get any provided command-line flags:
	opts = {};
	if ( flags.iter ) {
		opts.iter = parseInt( flags.iter, 10 );
	}
	if ( flags.sep ) {
		opts.sep = flags.sep;
	}
	if ( flags.normalized ) {
		opts.normalized = flags.normalized;
	}
	if ( flags.state ) {
		opts.state = loadState( resolve( dir, flags.state ) );
		if ( opts.state instanceof Error ) {
			return onError( opts.state );
		}
	} else if ( flags.seed ) {
		opts.seed = flags.seed.split( ',' );
		for ( i = 0; i < opts.seed.length; i++ ) {
			opts.seed[ i ] = parseInt( opts.seed[ i ], 10 );
		}
	}
	if ( flags.snapshot ) {
		proc.on( 'exit', onExit );
	}

	// Create a source stream and pipe to `stdout`:
	stream = randomStream( opts );
	stream.on( 'end', onEnd );

	stream.pipe( stdout );

	/**
	* Callback invoked upon exiting the process.
	*
	* @private
	* @param {integer} code - exit code
	*/
	function onExit( code ) {
		var state;
		var err;

		// Get the current PRNG state:
		state = stream.state;

		// Create a byte array "view":
		state = new Uint8Array( state.buffer, state.byteOffset, stream.byteLength ); // eslint-disable-line max-len

		// Convert the byte array to a `Buffer` (with support for older Node.js environments):
		state = array2buffer( state );

		// Attempt to write the state to file:
		err = writeFileSync( resolve( dir, flags.snapshot ), state );
		if ( err ) {
			onError( err, code || 1 );
		}
	}

	/**
	* Callback invoked upon encountering an error.
	*
	* @private
	* @param {Error} error - error
	* @param {integer} [code] - exit code
	*/
	function onError( error, code ) {
		cli.error( error, code || 1 );
	}
}

main();