const fs = require('fs')
const path = require('path')
const semver = require('semver')
const PackageCache = require('../PackageCache')
const PackageInfo = require('../PackageInfo')
const PackageURL = require('../PackageURL')

const FindExtensions = [ '', '.js', '.json' ]

/**
 * Resolves a path like "lib/file" into "lib/file.js" or "lib/file.json"
 * depending on which one is available, similar to require('lib/file').
 */
function findFile(base, useIndex, callback) {
  FindExtensions.reduceRight(function (next, ext) {
    const file = base + ext

    return function () {
      fs.stat(file, function (error, stats) {
        if (error) {
          if (error.code === 'ENOENT' || error.code === 'ENOTDIR') {
            next()
          } else {
            callback(error)
          }
        } else if (useIndex && stats.isDirectory()) {
          findFile(path.join(file, 'index'), false, function (error, indexFile, indexStats) {
            if (error) {
              callback(error)
            } else if (indexFile) {
              callback(null, indexFile, indexStats)
            } else {
              next()
            }
          })
        } else {
          callback(null, file, stats)
        }
      })
    }
  }, callback)()
}

/**
 * Fetch the file from the registry and get its stats. Redirect if the URL
 * does not specify an exact version number or targets a directory with no
 * trailing slash.
 */
function fetchFile(req, res, next) {
  PackageInfo.get(req.packageName, function (error, packageInfo) {
    if (error) {
      console.error(error)
      return res.status(500).type('text').send(`Cannot get info for package "${req.packageName}"`)
    }

    if (packageInfo == null || packageInfo.versions == null)
      return res.status(404).type('text').send(`Cannot find package "${req.packageName}"`)

    req.packageInfo = packageInfo

    const { versions, 'dist-tags': tags } = req.packageInfo

    if (req.packageVersion in versions) {
      // A valid request for a package we haven't downloaded yet.
      req.packageConfig = versions[req.packageVersion]

      PackageCache.get(req.packageConfig, function (error, outputDir) {
        if (error) {
          console.error(error)
          res.status(500).type('text').send(`Cannot fetch package ${req.packageSpec}`)
        } else {
          req.packageDir = outputDir

          if (req.filename) {
            // Based on the URL, figure out which file they want.
            const base = path.join(req.packageDir, req.filename)

            findFile(base, false, function (error, file, stats) {
              if (error)
                console.error(error)

              if (file == null) {
                res.status(404).type('text').send(`Cannot find file "${req.filename}" in package ${req.packageSpec}`)
              } else if (stats.isDirectory() && req.pathname[req.pathname.length - 1] !== '/') {
                // Append / to directory URLs.
                res.status(301).redirect(`${req.pathname}/${req.search}`)
              } else {
                req.file = file.replace(req.packageDir, '')
                req.stats = stats
                next()
              }
            })
          } else {
            // No filename in the URL. Try to figure out which file they want by
            // checking package.json's "unpkg", "browser", and "main" fields.
            let mainFilename

            const packageConfig = req.packageConfig
            const queryMain = req.query.main

            if (queryMain) {
              if (!(queryMain in packageConfig))
                return res.status(404).type('text').send(`Cannot find field "${queryMain}" in ${req.packageSpec} package config`)

              mainFilename = packageConfig[queryMain]
            } else {
              if (typeof packageConfig.unpkg === 'string') {
                // The "unpkg" field allows packages to explicitly declare the
                // file to serve at the bare URL (see #59).
                mainFilename = packageConfig.unpkg
              } else if (typeof packageConfig.browser === 'string') {
                // Fall back to the "browser" field if declared (only support strings).
                mainFilename = packageConfig.browser
              } else {
                // If there is no main, use "index" (same as npm).
                mainFilename = packageConfig.main || 'index'
              }
            }

            findFile(path.join(req.packageDir, mainFilename), true, function (error, file, stats) {
              if (error)
                console.error(error)

              if (file == null) {
                res.status(404).type('text').send(`Cannot find main file "${mainFilename}" in package ${req.packageSpec}`)
              } else {
                req.file = file.replace(req.packageDir, '')
                req.stats = stats
                next()
              }
            })
          }
        }
      })
    } else if (req.packageVersion in tags) {
      // Cache tag redirects for 1 minute.
      res.set({
        'Cache-Control': 'public, max-age=60',
        'Cache-Tag': 'redirect'
      }).redirect(PackageURL.create(req.packageName, tags[req.packageVersion], req.filename, req.search))
    } else {
      const maxVersion = semver.maxSatisfying(Object.keys(versions), req.packageVersion)

      if (maxVersion) {
        // Cache semver redirects for 1 minute.
        res.set({
          'Cache-Control': 'public, max-age=60',
          'Cache-Tag': 'redirect'
        }).redirect(PackageURL.create(req.packageName, maxVersion, req.filename, req.search))
      } else {
        res.status(404).type('text').send(`Cannot find package ${req.packageSpec}`)
      }
    }
  })
}

module.exports = fetchFile