package s3.website import s3.website.model.Site._ import scala.concurrent.{ExecutionContextExecutor, Future, Await} import scala.concurrent.duration._ import scala.language.postfixOps import s3.website.Diff.{resolveDeletes, resolveDiff} import s3.website.S3._ import scala.concurrent.ExecutionContext.fromExecutor import java.util.concurrent.Executors.newFixedThreadPool import java.util.concurrent.ExecutorService import s3.website.model._ import s3.website.model.FileUpdate import s3.website.model.NewFile import s3.website.S3.PushSuccessReport import scala.collection.mutable.ArrayBuffer import s3.website.CloudFront._ import s3.website.S3.SuccessfulDelete import s3.website.CloudFront.SuccessfulInvalidation import s3.website.S3.S3Setting import s3.website.CloudFront.CloudFrontSetting import s3.website.S3.SuccessfulUpload import s3.website.CloudFront.FailedInvalidation import scala.Int import java.io.File import com.lexicalscope.jewel.cli.CliFactory.parseArguments import s3.website.ByteHelper.humanReadableByteCount import s3.website.S3.SuccessfulUpload.humanizeUploadSpeed object Push { def main(args: Array[String]) { implicit val cliArgs = parseArguments(classOf[CliArgs], args:_*) implicit val s3Settings = S3Setting() implicit val cloudFrontSettings = CloudFrontSetting() implicit val workingDirectory = new File(System.getProperty("user.dir")).getAbsoluteFile System exit push } trait CliArgs { import com.lexicalscope.jewel.cli.Option @Option(defaultToNull = true) def site: String @Option(longName = Array("config-dir"), defaultToNull = true) def configDir: String @Option def verbose: Boolean @Option(longName = Array("dry-run")) def dryRun: Boolean } def push(implicit cliArgs: CliArgs, s3Settings: S3Setting, cloudFrontSettings: CloudFrontSetting, workingDirectory: File): ExitCode = { implicit val logger: Logger = new Logger(cliArgs.verbose) implicit val pushMode = new PushMode { def dryRun = cliArgs.dryRun } val errorOrSiteDir: ErrorOrFile = Option(cliArgs.site).fold(Ssg.findSiteDirectory(workingDirectory))(siteDirFromCli => Right(new File(siteDirFromCli))) def errorOrSite(siteInDirectory: File): Either[ErrorReport, Site] = loadSite(Option(cliArgs.configDir).getOrElse(workingDirectory.getPath) + "/s3_website.yml", siteInDirectory.getAbsolutePath) val errorOrPushStatus = for { siteInDirectory <- errorOrSiteDir.right loadedSite <- errorOrSite(siteInDirectory).right } yield { implicit val site = loadedSite val threadPool = newFixedThreadPool(site.config.concurrency_level) implicit val executor = fromExecutor(threadPool) val pushStatus = pushSite threadPool.shutdownNow() pushStatus } errorOrPushStatus.left foreach (err => logger.fail(s"Could not load the site: ${err.reportMessage}")) errorOrPushStatus fold((err: ErrorReport) => 1, pushStatus => pushStatus) } def pushSite( implicit site: Site, executor: ExecutionContextExecutor, s3Settings: S3Setting, cloudFrontSettings: CloudFrontSetting, logger: Logger, pushMode: PushMode ): ExitCode = { logger.info(s"${Deploy.renderVerb} ${site.rootDirectory}/* to ${site.config.s3_bucket}") val redirects = Redirect.resolveRedirects val s3FilesFuture = resolveS3Files() val redirectReports: PushReports = redirects.map(S3 uploadRedirect _) map (Right(_)) val pushReports: Future[PushReports] = for { errorOrUploads: Either[ErrorReport, Seq[Upload]] <- resolveDiff(s3FilesFuture) } yield { val uploadReports: PushReports = errorOrUploads.fold( error => Left(error) :: Nil, uploads => { uploads.map(S3 uploadFile _).map(Right(_)) } ) val deleteReports = Await.result(resolveDeletes(s3FilesFuture, redirects), 1 day).right.map { keysToDelete => keysToDelete map (S3 delete _) }.fold( error => Left(error) :: Nil, (pushResults: Seq[Future[PushErrorOrSuccess]]) => pushResults map (Right(_)) ) uploadReports ++ deleteReports ++ redirectReports } val finishedPushOps = awaitForResults(Await.result(pushReports, 1 day)) val invalidationSucceeded = invalidateCloudFrontItems(finishedPushOps) afterPushFinished(finishedPushOps, invalidationSucceeded) } def invalidateCloudFrontItems (finishedPushOperations: FinishedPushOperations) (implicit config: Config, cloudFrontSettings: CloudFrontSetting, ec: ExecutionContextExecutor, logger: Logger, pushMode: PushMode): Option[InvalidationSucceeded] = config.cloudfront_distribution_id.map { distributionId => val pushSuccessReports = finishedPushOperations.map { ops => for { failedOrSucceededPushes <- ops.right successfulPush <- failedOrSucceededPushes.right } yield successfulPush }.foldLeft(Seq(): Seq[PushSuccessReport]) { (reports, failOrSucc) => failOrSucc.fold( _ => reports, (pushSuccessReport: PushSuccessReport) => reports :+ pushSuccessReport ) } val invalidationResults: Seq[Either[FailedInvalidation, SuccessfulInvalidation]] = toInvalidationBatches(pushSuccessReports) map { invalidationBatch => Await.result( CloudFront.invalidate(invalidationBatch, distributionId), atMost = 1 day ) } if (invalidationResults.exists(_.isLeft)) false // If one of the invalidations failed, mark the whole process as failed else true } type InvalidationSucceeded = Boolean def afterPushFinished(finishedPushOps: FinishedPushOperations, invalidationSucceeded: Option[Boolean]) (implicit config: Config, logger: Logger, pushMode: PushMode): ExitCode = { val pushCounts = resolvePushCounts(finishedPushOps) logger.info(s"Summary: ${pushCountsToString(pushCounts)}") val pushOpExitCode = finishedPushOps.foldLeft(0) { (memo, finishedUpload) => memo + finishedUpload.fold( (error: ErrorReport) => 1, (failedOrSucceededUpload: Either[PushFailureReport, PushSuccessReport]) => if (failedOrSucceededUpload.isLeft) 1 else 0 ) } min 1 val cloudFrontInvalidationExitCode = invalidationSucceeded.fold(0)(allInvalidationsSucceeded => if (allInvalidationsSucceeded) 0 else 1 ) val exitCode = (pushOpExitCode + cloudFrontInvalidationExitCode) min 1 exitCode match { case 0 if !pushMode.dryRun && pushCounts.thereWasSomethingToPush => logger.info(s"Successfully pushed the website to http://${config.s3_bucket}.${config.s3_endpoint.s3WebsiteHostname}") case 1 => logger.fail(s"Failed to push the website to http://${config.s3_bucket}.${config.s3_endpoint.s3WebsiteHostname}") case _ => } exitCode } def awaitForResults(uploadReports: PushReports)(implicit executor: ExecutionContextExecutor): FinishedPushOperations = uploadReports map (_.right.map { rep => Await.result(rep, 1 day) }) def resolvePushCounts(implicit finishedOperations: FinishedPushOperations) = finishedOperations.foldLeft(PushCounts()) { (counts: PushCounts, uploadReport) => uploadReport.fold( (error: ErrorReport) => counts.copy(failures = counts.failures + 1), failureOrSuccess => failureOrSuccess.fold( (failureReport: PushFailureReport) => counts.copy(failures = counts.failures + 1), (successReport: PushSuccessReport) => successReport match { case succ: SuccessfulUpload => succ.details.fold(_.uploadType, _.uploadType) match { case NewFile => counts.copy(newFiles = counts.newFiles + 1).addTransferStats(succ) // TODO nasty repetition here case FileUpdate => counts.copy(updates = counts.updates + 1).addTransferStats(succ) case RedirectFile => counts.copy(redirects = counts.redirects + 1).addTransferStats(succ) } case SuccessfulDelete(_) => counts.copy(deletes = counts.deletes + 1) } ) ) } def pushCountsToString(pushCounts: PushCounts)(implicit pushMode: PushMode): String = pushCounts match { case PushCounts(updates, newFiles, failures, redirects, deletes, _, _) if updates == 0 && newFiles == 0 && failures == 0 && redirects == 0 && deletes == 0 => PushNothing.renderVerb case PushCounts(updates, newFiles, failures, redirects, deletes, uploadedBytes, uploadDurations) => val reportClauses: scala.collection.mutable.ArrayBuffer[String] = ArrayBuffer() if (updates > 0) reportClauses += s"${Updated.renderVerb} ${updates ofType "file"}." if (newFiles > 0) reportClauses += s"${Created.renderVerb} ${newFiles ofType "file"}." if (failures > 0) reportClauses += s"${failures ofType "operation"} failed." // This includes both failed uploads and deletes. if (redirects > 0) reportClauses += s"${Applied.renderVerb} ${redirects ofType "redirect"}." if (deletes > 0) reportClauses += s"${Deleted.renderVerb} ${deletes ofType "file"}." if (uploadedBytes > 0) { val transferSuffix = humanizeUploadSpeed(uploadedBytes, uploadDurations: _*).fold(".")(speed => s", $speed.") reportClauses += s"${Transferred.renderVerb} ${humanReadableByteCount(uploadedBytes)}$transferSuffix" } reportClauses.mkString(" ") } case class PushCounts( updates: Int = 0, newFiles: Int = 0, failures: Int = 0, redirects: Int = 0, deletes: Int = 0, uploadedBytes: Long = 0, uploadDurations: Seq[UploadDuration] = Nil ) { val thereWasSomethingToPush = updates + newFiles + redirects + deletes > 0 def addTransferStats(successfulUpload: SuccessfulUpload): PushCounts = copy( uploadedBytes = uploadedBytes + (successfulUpload.uploadSize getOrElse 0L), uploadDurations = uploadDurations ++ successfulUpload.details.fold(_.uploadDuration, _ => None) ) } type FinishedPushOperations = Seq[Either[ErrorReport, PushErrorOrSuccess]] type PushReports = Seq[Either[ErrorReport, Future[PushErrorOrSuccess]]] case class PushResult(threadPool: ExecutorService, uploadReports: PushReports) type ExitCode = Int }