var async = require('async');
var logger = require( 'pelias-logger' ).get( 'api' );
var service = require('../service/interpolation');

/**
example response from interpolation web service:
{
  type: 'Feature',
  properties: {
    type: 'interpolated',
    source: 'mixed',
    number: '17',
    lat: -41.2887032,
    lon: 174.767089
  },
  geometry: {
    type: 'Point',
    coordinates: [ 174.767089, -41.2887032 ]
  }
}
**/

function setup() {

  var transport = service.search();
  var middleware = function(req, res, next) {

    // no-op, user did not request an address
    if( !isAddressQuery( req ) ){
      return next();
    }

    // bind parsed_text variables to function call
    var bound = interpolate.bind( transport, req.clean.parsed_text );

    // perform interpolations asynchronously for all relevant hits
    var timer = (new Date()).getTime();
    async.map( res.data, bound, function( err, results ){

      // update res.data with the mapped values
      if( !err ){
        res.data = results;
      }

      // sort the results to ensure that addresses show up higher than street centroids      
      res.data = res.data.sort((a, b) => {
        if (a.layer === 'address' && b.layer !== 'address') { return -1; }
        if (a.layer !== 'address' && b.layer === 'address') { return 1; }
        return 0;
      });

      // log the execution time, continue
      logger.info( '[interpolation] [took]', (new Date()).getTime() - timer, 'ms' );
      next();
    });
  };

  middleware.transport = transport;
  return middleware;
}

function interpolate( parsed_text, hit, cb ){

  // no-op, this hit is not from the 'street' layer
  // note: no network request is performed.
  if( !hit || hit.layer !== 'street' ){
    return cb( null, hit );
  }

  // query variables
  var coord = hit.center_point;
  var number = parsed_text.number;
  var street = hit.address_parts.street || parsed_text.street;

  // query interpolation service
  this.query( coord, number, street, function( err, data ){

    // an error occurred
    // note: leave this hit unmodified
    if( err ){
      logger.error( '[interpolation] [error]', err );
      return cb( null, hit );
    }

    // invalid / not useful response
    // note: leave this hit unmodified
    if( !data || !data.hasOwnProperty('properties') ){
      logger.info( '[interpolation] [miss]', parsed_text );
      return cb( null, hit );
    }

    // the interpolation service returned a valid result
    // note: we now merge thos values with the existing 'street' record.
    logger.info( '[interpolation] [hit]', parsed_text, data );

    // safety first!
    try {

      // -- metatdata --
      hit.layer = 'address';
      hit.match_type = 'interpolated';

      // -- name --
      hit.name.default = data.properties.number + ' ' + hit.name.default;

      // -- source --
      var source = 'mixed';
      if( data.properties.source === 'OSM' ){ source = 'openstreetmap'; }
      else if( data.properties.source === 'OA' ){ source = 'openaddresses'; }
      hit.source = source;

      // -- source_id --
      // note: interpolated values have no source_id
      delete hit.source_id; // remove original street source_id
      if( data.properties.hasOwnProperty( 'source_id' ) ){
        hit.source_id = data.properties.source_id;
      }

      // -- address_parts --
      hit.address_parts.number = data.properties.number;

      // -- geo --
      hit.center_point = {
        lat: data.properties.lat,
        lon: data.properties.lon
      };

      // -- bbox --
      delete hit.bounding_box;

      // return the modified hit
      return cb( null, hit );

    // a syntax error occurred in the code above (this shouldn't happen!)
    // note: the hit object may be partially modified, could possibly be invalid
    } catch( e ){
      logger.error( '[interpolation] [error]', e, e.stack );
      return cb( null, hit );
    }
  });
}

// boolean function to check if an address was requested by the user
function isAddressQuery( req ){
  return req && req.hasOwnProperty('clean') &&
         req.clean.hasOwnProperty('parsed_text') &&
         req.clean.parsed_text.hasOwnProperty('number') &&
         req.clean.parsed_text.hasOwnProperty('street');
}

module.exports = setup;