import { getPaymentFailureStatus, getPaymentOrNotSent } from '@/api/lnd'
import { walletLogger } from '@/api/resolvers/wallet'
import { formatMsats, formatSats, msatsToSats, toPositiveBigInt } from '@/lib/format'
import { datePivot } from '@/lib/time'
import { notifyWithdrawal } from '@/lib/webPush'
import { Prisma } from '@prisma/client'

async function transitionWithdrawal (jobName,
  { withdrawalId, toStatus, transition, withdrawal, onUnexpectedError },
  { models, lnd, boss }
) {
  console.group(`${jobName}: transitioning withdrawal ${withdrawalId} from null to ${toStatus}`)

  let dbWithdrawal
  try {
    const currentDbWithdrawal = await models.withdrawl.findUnique({ where: { id: withdrawalId } })
    console.log('withdrawal has status', currentDbWithdrawal.status)

    if (currentDbWithdrawal.status) {
      console.log('withdrawal is already has a terminal status, skipping transition')
      return
    }

    const { hash, createdAt } = currentDbWithdrawal
    const lndWithdrawal = withdrawal ?? await getPaymentOrNotSent({ id: hash, lnd, createdAt })

    const transitionedWithdrawal = await models.$transaction(async tx => {
      // grab optimistic concurrency lock and the withdrawal
      dbWithdrawal = await tx.withdrawl.update({
        include: {
          wallet: true
        },
        where: {
          id: withdrawalId,
          status: null
        },
        data: {
          status: toStatus
        }
      })

      // our own optimistic concurrency check
      if (!dbWithdrawal) {
        console.log('record not found in our own concurrency check, assuming concurrent worker transitioned it')
        return
      }

      const data = await transition({ lndWithdrawal, dbWithdrawal, tx })
      if (data) {
        return await tx.withdrawl.update({
          include: {
            wallet: true
          },
          where: { id: dbWithdrawal.id },
          data
        })
      }

      return dbWithdrawal
    }, {
      isolationLevel: Prisma.TransactionIsolationLevel.ReadCommitted,
      // we only need to do this because we settleHodlInvoice inside the transaction
      // ... and it's prone to timing out
      timeout: 60000
    })

    if (transitionedWithdrawal) {
      console.log('transition succeeded')
      return transitionedWithdrawal
    }
  } catch (e) {
    if (e instanceof Prisma.PrismaClientKnownRequestError) {
      if (e.code === 'P2025') {
        console.log('record not found, assuming concurrent worker transitioned it')
        return
      }
      if (e.code === 'P2034') {
        console.log('write conflict, assuming concurrent worker is transitioning it')
        return
      }
    }

    console.error('unexpected error', e)
    onUnexpectedError?.({ error: e, dbWithdrawal, models, boss })
    await boss.send(
      jobName,
      { withdrawalId },
      { startAfter: datePivot(new Date(), { seconds: 30 }), priority: 1000 })
  } finally {
    console.groupEnd()
  }
}

export async function payingActionConfirmed ({ data: args, models, lnd, boss }) {
  const transitionedWithdrawal = await transitionWithdrawal('payingActionConfirmed', {
    toStatus: 'CONFIRMED',
    ...args,
    transition: async ({ dbWithdrawal, lndWithdrawal, tx }) => {
      if (!lndWithdrawal?.is_confirmed) {
        throw new Error('withdrawal is not confirmed')
      }

      const msatsFeePaid = toPositiveBigInt(lndWithdrawal.payment.fee_mtokens)
      const msatsPaid = toPositiveBigInt(lndWithdrawal.payment.mtokens) - msatsFeePaid

      console.log(`withdrawal confirmed paying ${msatsToSats(msatsPaid)} sats with ${msatsToSats(msatsFeePaid)} fee`)

      await tx.user.update({
        where: { id: dbWithdrawal.userId },
        data: { msats: { increment: dbWithdrawal.msatsFeePaying - msatsFeePaid } }
      })

      console.log(`user refunded ${msatsToSats(dbWithdrawal.msatsFeePaying - msatsFeePaid)} sats`)

      return {
        msatsFeePaid,
        msatsPaid,
        preimage: lndWithdrawal.payment.secret
      }
    }
  }, { models, lnd, boss })

  if (transitionedWithdrawal) {
    await notifyWithdrawal(transitionedWithdrawal)

    const logger = walletLogger({ models, wallet: transitionedWithdrawal.wallet })
    logger?.ok(
      `↙ payment received: ${formatSats(msatsToSats(transitionedWithdrawal.msatsPaid))}`,
      {
        bolt11: transitionedWithdrawal.bolt11,
        preimage: transitionedWithdrawal.preimage,
        fee: formatMsats(transitionedWithdrawal.msatsFeePaid)
      })
  }
}

export async function payingActionFailed ({ data: args, models, lnd, boss }) {
  let message
  const transitionedWithdrawal = await transitionWithdrawal('payingActionFailed', {
    toStatus: 'UNKNOWN_FAILURE',
    ...args,
    transition: async ({ dbWithdrawal, lndWithdrawal, tx }) => {
      if (!lndWithdrawal?.is_failed) {
        throw new Error('withdrawal is not failed')
      }

      console.log(`withdrawal failed paying ${msatsToSats(dbWithdrawal.msatsPaying)} sats with ${msatsToSats(dbWithdrawal.msatsFeePaying)} fee`)

      await tx.user.update({
        where: { id: dbWithdrawal.userId },
        data: { msats: { increment: dbWithdrawal.msatsFeePaying + dbWithdrawal.msatsPaying } }
      })

      console.log(`user refunded ${msatsToSats(dbWithdrawal.msatsFeePaying + dbWithdrawal.msatsPaying)} sats`)

      // update to particular status
      const { status, message: failureMessage } = getPaymentFailureStatus(lndWithdrawal)
      message = failureMessage

      console.log('withdrawal failed with status', status)
      return {
        status
      }
    }
  }, { models, lnd, boss })

  if (transitionedWithdrawal) {
    const logger = walletLogger({ models, wallet: transitionedWithdrawal.wallet })
    logger?.error(
      `incoming payment failed: ${message}`,
      {
        bolt11: transitionedWithdrawal.bolt11,
        max_fee: formatMsats(transitionedWithdrawal.msatsFeePaying)
      })
  }
}