/*
 * decaffeinate suggestions:
 * DS102: Remove unnecessary code created because of implicit returns
 * DS207: Consider shorter variations of null checks
 * Full docs: https://github.com/decaffeinate/decaffeinate/blob/master/docs/suggestions.md
 */
const chai = require('chai')
const { expect } = chai
const path = require('path')
const modulePath = path.join(__dirname, '../../../timeAsyncMethod.js')
const SandboxedModule = require('sandboxed-module')
const sinon = require('sinon')

describe('timeAsyncMethod', function () {
  beforeEach(function () {
    this.Timer = { done: sinon.stub() }
    this.TimerConstructor = sinon.stub().returns(this.Timer)
    this.metrics = {
      Timer: this.TimerConstructor,
      inc: sinon.stub(),
    }
    this.timeAsyncMethod = SandboxedModule.require(modulePath, {
      requires: {
        './index': this.metrics,
      },
    })

    return (this.testObject = {
      nextNumber(n, callback) {
        return setTimeout(() => callback(null, n + 1), 100)
      },
    })
  })

  it('should have the testObject behave correctly before wrapping', function (done) {
    return this.testObject.nextNumber(2, (err, result) => {
      expect(err).to.not.exist
      expect(result).to.equal(3)
      return done()
    })
  })

  it('should wrap method without error', function (done) {
    this.timeAsyncMethod(
      this.testObject,
      'nextNumber',
      'someContext.TestObject'
    )
    return done()
  })

  it('should transparently wrap method invocation in timer', function (done) {
    this.timeAsyncMethod(
      this.testObject,
      'nextNumber',
      'someContext.TestObject'
    )
    return this.testObject.nextNumber(2, (err, result) => {
      expect(err).to.not.exist
      expect(result).to.equal(3)
      expect(this.TimerConstructor.callCount).to.equal(1)
      expect(this.Timer.done.callCount).to.equal(1)
      return done()
    })
  })

  it('should increment success count', function (done) {
    this.metrics.inc = sinon.stub()
    this.timeAsyncMethod(
      this.testObject,
      'nextNumber',
      'someContext.TestObject'
    )
    return this.testObject.nextNumber(2, (err, result) => {
      if (err) {
        return done(err)
      }
      expect(this.metrics.inc.callCount).to.equal(1)
      expect(
        this.metrics.inc.calledWith('someContext_result', 1, {
          method: 'TestObject_nextNumber',
          status: 'success',
        })
      ).to.equal(true)
      return done()
    })
  })

  describe('when base method produces an error', function () {
    beforeEach(function () {
      this.metrics.inc = sinon.stub()
      return (this.testObject.nextNumber = function (n, callback) {
        return setTimeout(() => callback(new Error('woops')), 100)
      })
    })

    it('should propagate the error transparently', function (done) {
      this.timeAsyncMethod(
        this.testObject,
        'nextNumber',
        'someContext.TestObject'
      )
      return this.testObject.nextNumber(2, (err, result) => {
        expect(err).to.exist
        expect(err).to.be.instanceof(Error)
        expect(result).to.not.exist
        return done()
      })
    })

    return it('should increment failure count', function (done) {
      this.timeAsyncMethod(
        this.testObject,
        'nextNumber',
        'someContext.TestObject'
      )
      return this.testObject.nextNumber(2, (err, result) => {
        expect(err).to.exist
        expect(this.metrics.inc.callCount).to.equal(1)
        expect(
          this.metrics.inc.calledWith('someContext_result', 1, {
            method: 'TestObject_nextNumber',
            status: 'failed',
          })
        ).to.equal(true)
        return done()
      })
    })
  })

  describe('when a logger is supplied', function () {
    beforeEach(function () {
      return (this.logger = { debug: sinon.stub(), log: sinon.stub() })
    })

    return it('should also call logger.debug', function (done) {
      this.timeAsyncMethod(
        this.testObject,
        'nextNumber',
        'someContext.TestObject',
        this.logger
      )
      return this.testObject.nextNumber(2, (err, result) => {
        expect(err).to.not.exist
        expect(result).to.equal(3)
        expect(this.TimerConstructor.callCount).to.equal(1)
        expect(this.Timer.done.callCount).to.equal(1)
        expect(this.logger.debug.callCount).to.equal(1)
        return done()
      })
    })
  })

  describe('when the wrapper cannot be applied', function () {
    beforeEach(function () {})

    return it('should raise an error', function () {
      const badWrap = () => {
        return this.timeAsyncMethod(
          this.testObject,
          'DEFINITELY_NOT_A_REAL_METHOD',
          'someContext.TestObject'
        )
      }
      return expect(badWrap).to.throw(
        /^.*expected object property 'DEFINITELY_NOT_A_REAL_METHOD' to be a function.*$/
      )
    })
  })

  return describe('when the wrapped function is not using a callback', function () {
    beforeEach(function () {
      this.realMethod = sinon.stub().returns(42)
      return (this.testObject.nextNumber = this.realMethod)
    })

    it('should not throw an error', function () {
      this.timeAsyncMethod(
        this.testObject,
        'nextNumber',
        'someContext.TestObject'
      )
      const badCall = () => {
        return this.testObject.nextNumber(2)
      }
      return expect(badCall).to.not.throw(Error)
    })

    return it('should call the underlying method', function () {
      this.timeAsyncMethod(
        this.testObject,
        'nextNumber',
        'someContext.TestObject'
      )
      const result = this.testObject.nextNumber(12)
      expect(this.realMethod.callCount).to.equal(1)
      expect(this.realMethod.calledWith(12)).to.equal(true)
      return expect(result).to.equal(42)
    })
  })
})