package s3.website import s3.website.model.Site._ import scala.concurrent.{ExecutionContextExecutor, Future, Await} import scala.concurrent.duration._ import com.lexicalscope.jewel.cli.CliFactory import scala.language.postfixOps import s3.website.Diff.{resolveNewFiles, resolveDeletes} import s3.website.S3._ import scala.concurrent.ExecutionContext.fromExecutor import java.util.concurrent.Executors.newFixedThreadPool import s3.website.model.LocalFile.resolveLocalFiles import scala.collection.parallel.ParSeq import java.util.concurrent.ExecutorService import s3.website.model._ import s3.website.model.Update 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 object Push { 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 utils = new Utils val redirects = Redirect.resolveRedirects val redirectResults = redirects.map(new S3() upload(_)) val errorsOrReports = for { localFiles <- resolveLocalFiles.right errorOrS3FilesAndUpdateFutures <- Await.result(resolveS3FilesAndUpdates(localFiles)(), 7 days).right s3Files <- errorOrS3FilesAndUpdateFutures._1.right } yield { val updateReports: PushReports = errorOrS3FilesAndUpdateFutures._2.par val deleteReports: PushReports = utils toParSeq resolveDeletes(localFiles, s3Files, redirects) .map { s3File => new S3() delete s3File.s3Key } .map { Right(_) } // To make delete reports type-compatible with upload reports val uploadReports: PushReports = utils toParSeq resolveNewFiles(localFiles, s3Files) .map { _.right.map(new S3() upload(_)) } uploadReports ++ deleteReports ++ updateReports ++ redirectResults.map(Right(_)) } val errorsOrFinishedPushOps: Either[ErrorReport, FinishedPushOperations] = errorsOrReports.right map { uploadReports => awaitForUploads(uploadReports) } val invalidationSucceeded = invalidateCloudFrontItems(errorsOrFinishedPushOps) afterPushFinished(errorsOrFinishedPushOps, invalidationSucceeded) } def invalidateCloudFrontItems (errorsOrFinishedPushOps: Either[ErrorReport, FinishedPushOperations]) (implicit config: Config, cloudFrontSettings: CloudFrontSetting, ec: ExecutionContextExecutor, logger: Logger, pushMode: PushMode): Option[InvalidationSucceeded] = { config.cloudfront_distribution_id.map { distributionId => val pushSuccessReports = errorsOrFinishedPushOps.fold( errors => Nil, finishedPushOps => { finishedPushOps.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( new 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(errorsOrFinishedUploads: Either[ErrorReport, FinishedPushOperations], invalidationSucceeded: Option[Boolean]) (implicit config: Config, logger: Logger, pushMode: PushMode): ExitCode = { val pushCountsOption = errorsOrFinishedUploads.right.map(resolvePushCounts(_)).right.toOption pushCountsOption.map(pushCountsToString).foreach(pushCounts => logger.info(s"Summary: $pushCounts")) errorsOrFinishedUploads.left foreach (err => logger.fail(s"Encountered an error: ${err.reportMessage}")) val exitCode = errorsOrFinishedUploads.fold( _ => 1, finishedUploads => finishedUploads.foldLeft(0) { (memo, finishedUpload) => memo + finishedUpload.fold( (error: ErrorReport) => 1, (failedOrSucceededUpload: Either[PushFailureReport, PushSuccessReport]) => if (failedOrSucceededUpload.isLeft) 1 else 0 ) } min 1 ) max invalidationSucceeded.fold(0)(allInvalidationsSucceeded => if (allInvalidationsSucceeded) 0 else 1 ) exitCode match { case 0 if !pushMode.dryRun && pushCountsOption.exists(_.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 awaitForUploads(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 SuccessfulUpload(upload, _) => upload.uploadType match { case NewFile => counts.copy(newFiles = counts.newFiles + 1) case Update => counts.copy(updates = counts.updates + 1) case Redirect => counts.copy(redirects = counts.redirects + 1) } 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) => 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"}." reportClauses.mkString(" ") } case class PushCounts( updates: Int = 0, newFiles: Int = 0, failures: Int = 0, redirects: Int = 0, deletes: Int = 0 ) { val thereWasSomethingToPush = updates + newFiles + redirects + deletes > 0 } type FinishedPushOperations = ParSeq[Either[ErrorReport, PushErrorOrSuccess]] type PushReports = ParSeq[Either[ErrorReport, Future[PushErrorOrSuccess]]] case class PushResult(threadPool: ExecutorService, uploadReports: PushReports) type ExitCode = Int 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 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 } 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) } }