/**
* @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';

// VARIABLES //

var noop = 'next();';


// MAIN //

/**
* Generates a source code body for asynchronous execution.
*
* ## Notes
*
* -   Example output:
*
*     ```javascript
*     "use strict";
*
*     var ctx = this;
*     var state1 = {};
*     var t1, d1, i1;
*
*     function before( state, next ) {
*         // {{before}}
*     }
*
*     function cb1( error ) {
*         if ( error ) {
*             return ctx.done( error );
*         }
*         i1 = 0;
*         t1 = ctx.tic();
*         main( state1, cb2 );
*     }
*
*     function main( state, next ) {
*         // {{code}}
*     }
*
*     function cb2( error ) {
*         if ( error ) {
*             return ctx.done( error );
*         }
*         i1 += 1;
*         if ( i1 < 1e6 ) {
*             return main( state1, cb2 );
*         }
*         d1 = ctx.toc( t1 );
*         after( state1, cb3 );
*     }
*
*     function after( state, next ) {
*         // {{after}}
*     }
*
*     function cb3( error ) {
*         if ( error ) {
*             return ctx.done( error );
*         }
*         ctx.done( null, d1 );
*     }
*
*     before( state1, cb1 );
*
*     return 1;
*     ```
*
*
* @private
* @param {number} id - id
* @param {string} code - code to time
* @param {Options} opts - function options
* @param {string} opts.before - setup code
* @param {string} opts.after - cleanup code
* @param {PositiveInteger} opts.iterations - number of iterations
* @returns {string} source code body
*/
function body( id, code, opts ) {
	var before;
	var after;
	var state;
	var main;
	var src;
	var cb1;
	var cb2;
	var cb3;
	var ctx;
	var t;
	var d;
	var i;

	src = '"use strict";';

	// Define identifiers:
	before = '__before$'+id+'__';
	after = '__after$'+id+'__';
	state = '__state$'+id+'__';
	main = '__main$'+id+'__';
	cb1 = '__cb1$'+id+'__';
	cb2 = '__cb2$'+id+'__';
	cb3 = '__cb3$'+id+'__';
	ctx = '__ctx$'+id+'__';
	i = '__i$'+id+'__';
	t = '__t$'+id+'__';
	d = '__d$'+id+'__';

	// Declare variables:
	src += 'var '+ctx+' = this;';
	src += 'var '+state+' = {};';
	src += 'var '+i+','+t+','+d+';';

	// Insert the setup code:
	src += 'function '+before+'( state, next ) {';
	src += '  '+(opts.before||noop)+';';
	src += '}';

	// Insert the setup callback:
	src += 'function '+cb1+'( error ) {';
	src += '  if ( error ) {';
	src += '    return '+ctx+'.done( error );';
	src += '  }';
	src += '  '+i+' = 0;';
	src += '  '+t+' = '+ctx+'.tic();'; // start the timer
	src += '  '+main+'( '+state+','+cb2+' );';
	src += '}';

	// Wrap the code to time in a function:
	src += 'function '+main+'( state, next ) {';
	src += '  '+(code||noop)+';';
	src += '}';

	// Insert the main callback:
	src += 'function '+cb2+'( error ) {';
	src += '  if ( error ) {';
	src += '    return '+ctx+'.done( error );';
	src += '  }';
	src += '  '+i+' += 1;';
	src += '  if ( '+i+' < '+opts.iterations+' ) {';
	src += '    return '+main+'( '+state+','+cb2+' );';
	src += '  }';
	src += '  '+d+' = '+ctx+'.toc( '+t+' );'; // stop the timer
	src += '  '+after+'( '+state+','+cb3+' );';
	src += '}';

	// Insert the cleanup code:
	src += 'function '+after+'( state, next ) {';
	src += '  '+(opts.after||noop)+';';
	src += '}';

	// Insert the cleanup callback:
	src += 'function '+cb3+'( error ) {';
	src += '  if ( error ) {';
	src += '    return '+ctx+'.done( error );';
	src += '  }';
	src += '  '+ctx+'.done( null,'+d+' );'; // return results
	src += '}';

	// Invoke the setup function to begin the execution sequence:
	src += before+'( '+state+','+cb1+' );';

	// Return a value:
	src += 'return '+id+';';

	return src;
}


// EXPORTS //

module.exports = body;