From 9030635596f016f4522d34bd97b34ee6450f9b47 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Tue, 28 Aug 2018 15:34:54 +0530 Subject: [PATCH 01/40] Move AcmePhp from core to site-command Signed-off-by: Riddhesh Sanghvi --- .../Cli/Exception/AcmeCliActionException.php | 23 + AcmePhp/Cli/Exception/AcmeCliException.php | 23 + .../Exception/AcmeDnsResolutionException.php | 23 + .../Cli/Exception/CommandFlowException.php | 55 +++ AcmePhp/Cli/Repository/Repository.php | 439 ++++++++++++++++++ .../Cli/Repository/RepositoryInterface.php | 198 ++++++++ .../Cli/Repository/RepositoryV2Interface.php | 51 ++ AcmePhp/Cli/Serializer/PemEncoder.php | 55 +++ AcmePhp/Cli/Serializer/PemNormalizer.php | 55 +++ 9 files changed, 922 insertions(+) create mode 100644 AcmePhp/Cli/Exception/AcmeCliActionException.php create mode 100644 AcmePhp/Cli/Exception/AcmeCliException.php create mode 100644 AcmePhp/Cli/Exception/AcmeDnsResolutionException.php create mode 100644 AcmePhp/Cli/Exception/CommandFlowException.php create mode 100644 AcmePhp/Cli/Repository/Repository.php create mode 100644 AcmePhp/Cli/Repository/RepositoryInterface.php create mode 100644 AcmePhp/Cli/Repository/RepositoryV2Interface.php create mode 100644 AcmePhp/Cli/Serializer/PemEncoder.php create mode 100644 AcmePhp/Cli/Serializer/PemNormalizer.php diff --git a/AcmePhp/Cli/Exception/AcmeCliActionException.php b/AcmePhp/Cli/Exception/AcmeCliActionException.php new file mode 100644 index 00000000..a18f3e6f --- /dev/null +++ b/AcmePhp/Cli/Exception/AcmeCliActionException.php @@ -0,0 +1,23 @@ + + * + * For the full copyright and license information, please view the LICENSE + * file that was distributed with this source code. + */ + +namespace AcmePhp\Cli\Exception; + +/** + * @author Titouan Galopin + */ +class AcmeCliActionException extends AcmeCliException +{ + public function __construct($actionName, \Exception $previous = null) + { + parent::__construct(sprintf('An exception was thrown during action "%s"', $actionName), $previous); + } +} diff --git a/AcmePhp/Cli/Exception/AcmeCliException.php b/AcmePhp/Cli/Exception/AcmeCliException.php new file mode 100644 index 00000000..b5f4c0c7 --- /dev/null +++ b/AcmePhp/Cli/Exception/AcmeCliException.php @@ -0,0 +1,23 @@ + + * + * For the full copyright and license information, please view the LICENSE + * file that was distributed with this source code. + */ + +namespace AcmePhp\Cli\Exception; + +/** + * @author Titouan Galopin + */ +class AcmeCliException extends \RuntimeException +{ + public function __construct($message, \Exception $previous = null) + { + parent::__construct($message, 0, $previous); + } +} diff --git a/AcmePhp/Cli/Exception/AcmeDnsResolutionException.php b/AcmePhp/Cli/Exception/AcmeDnsResolutionException.php new file mode 100644 index 00000000..a1137999 --- /dev/null +++ b/AcmePhp/Cli/Exception/AcmeDnsResolutionException.php @@ -0,0 +1,23 @@ + + * + * For the full copyright and license information, please view the LICENSE + * file that was distributed with this source code. + */ + +namespace AcmePhp\Cli\Exception; + +/** + * @author Jérémy Derussé + */ +class AcmeDnsResolutionException extends AcmeCliException +{ + public function __construct($message, \Exception $previous = null) + { + parent::__construct(null === $message ? 'An exception was thrown during resolution of DNS' : $message, $previous); + } +} diff --git a/AcmePhp/Cli/Exception/CommandFlowException.php b/AcmePhp/Cli/Exception/CommandFlowException.php new file mode 100644 index 00000000..f03b5b29 --- /dev/null +++ b/AcmePhp/Cli/Exception/CommandFlowException.php @@ -0,0 +1,55 @@ + + * + * For the full copyright and license information, please view the LICENSE + * file that was distributed with this source code. + */ + +namespace AcmePhp\Cli\Exception; + +/** + * @author Jérémy Derussé + */ +class CommandFlowException extends AcmeCliException +{ + /** + * @var string + */ + private $missing; + /** + * @var string + */ + private $command; + /** + * @var array + */ + private $arguments; + + /** + * @param string $missing Missing requirement to fix the flow + * @param string $command Name of the command to run in order to fix the flow + * @param array $arguments Optional list of missing arguments + * @param \Exception|null $previous + */ + public function __construct($missing, $command, array $arguments = [], \Exception $previous = null) + { + $this->missing = $missing; + $this->command = $command; + $this->arguments = $arguments; + + $message = trim(sprintf( + 'You have to %s first. Run the command%sphp %s %s %s', + $missing, + PHP_EOL.PHP_EOL, + $_SERVER['PHP_SELF'], + $command, + implode(' ', $arguments) + )); + + parent::__construct($message, $previous); + } +} diff --git a/AcmePhp/Cli/Repository/Repository.php b/AcmePhp/Cli/Repository/Repository.php new file mode 100644 index 00000000..29a54c18 --- /dev/null +++ b/AcmePhp/Cli/Repository/Repository.php @@ -0,0 +1,439 @@ + + * + * For the full copyright and license information, please view the LICENSE + * file that was distributed with this source code. + */ + +namespace AcmePhp\Cli\Repository; + +use AcmePhp\Cli\Exception\AcmeCliException; +use AcmePhp\Cli\Serializer\PemEncoder; +use AcmePhp\Core\Protocol\AuthorizationChallenge; +use AcmePhp\Core\Protocol\CertificateOrder; +use AcmePhp\Ssl\Certificate; +use AcmePhp\Ssl\CertificateResponse; +use AcmePhp\Ssl\DistinguishedName; +use AcmePhp\Ssl\KeyPair; +use AcmePhp\Ssl\PrivateKey; +use AcmePhp\Ssl\PublicKey; +use League\Flysystem\FilesystemInterface; +use Symfony\Component\Serializer\Encoder\JsonEncoder; +use Symfony\Component\Serializer\SerializerInterface; + +/** + * @author Titouan Galopin + */ +class Repository implements RepositoryV2Interface +{ + const PATH_ACCOUNT_KEY_PRIVATE = 'account/key.private.pem'; + const PATH_ACCOUNT_KEY_PUBLIC = 'account/key.public.pem'; + + const PATH_DOMAIN_KEY_PUBLIC = 'certs/{domain}/private/key.public.pem'; + const PATH_DOMAIN_KEY_PRIVATE = 'certs/{domain}/private/key.private.pem'; + const PATH_DOMAIN_CERT_CERT = 'certs/{domain}/public/cert.pem'; + const PATH_DOMAIN_CERT_CHAIN = 'certs/{domain}/public/chain.pem'; + const PATH_DOMAIN_CERT_FULLCHAIN = 'certs/{domain}/public/fullchain.pem'; + const PATH_DOMAIN_CERT_COMBINED = 'certs/{domain}/private/combined.pem'; + + const PATH_CACHE_AUTHORIZATION_CHALLENGE = 'var/{domain}/authorization_challenge.json'; + const PATH_CACHE_DISTINGUISHED_NAME = 'var/{domain}/distinguished_name.json'; + const PATH_CACHE_CERTIFICATE_ORDER = 'var/{domains}/certificate_order.json'; + + /** + * @var SerializerInterface + */ + private $serializer; + + /** + * @var FilesystemInterface + */ + private $master; + + /** + * @var FilesystemInterface + */ + private $backup; + + /** + * @var bool + */ + private $enableBackup; + + /** + * @param SerializerInterface $serializer + * @param FilesystemInterface $master + * @param FilesystemInterface $backup + * @param bool $enableBackup + */ + public function __construct(SerializerInterface $serializer, FilesystemInterface $master, FilesystemInterface $backup, $enableBackup) + { + $this->serializer = $serializer; + $this->master = $master; + $this->backup = $backup; + $this->enableBackup = $enableBackup; + } + + /** + * {@inheritdoc} + */ + public function storeCertificateResponse(CertificateResponse $certificateResponse) + { + $distinguishedName = $certificateResponse->getCertificateRequest()->getDistinguishedName(); + $domain = $distinguishedName->getCommonName(); + + $this->storeDomainKeyPair($domain, $certificateResponse->getCertificateRequest()->getKeyPair()); + $this->storeDomainDistinguishedName($domain, $distinguishedName); + $this->storeDomainCertificate($domain, $certificateResponse->getCertificate()); + } + + /** + * {@inheritdoc} + */ + public function storeAccountKeyPair(KeyPair $keyPair) + { + try { + $this->save( + self::PATH_ACCOUNT_KEY_PUBLIC, + $this->serializer->serialize($keyPair->getPublicKey(), PemEncoder::FORMAT) + ); + + $this->save( + self::PATH_ACCOUNT_KEY_PRIVATE, + $this->serializer->serialize($keyPair->getPrivateKey(), PemEncoder::FORMAT) + ); + } catch (\Exception $e) { + throw new AcmeCliException('Storing of account key pair failed', $e); + } + } + + private function getPathForDomain($path, $domain) + { + return strtr($path, ['{domain}' => $this->normalizeDomain($domain)]); + } + + private function getPathForDomainList($path, array $domains) + { + return strtr($path, ['{domains}' => $this->normalizeDomainList($domains)]); + } + + /** + * {@inheritdoc} + */ + public function hasAccountKeyPair() + { + return $this->master->has(self::PATH_ACCOUNT_KEY_PRIVATE); + } + + /** + * {@inheritdoc} + */ + public function loadAccountKeyPair() + { + try { + $publicKeyPem = $this->master->read(self::PATH_ACCOUNT_KEY_PUBLIC); + $privateKeyPem = $this->master->read(self::PATH_ACCOUNT_KEY_PRIVATE); + + return new KeyPair( + $this->serializer->deserialize($publicKeyPem, PublicKey::class, PemEncoder::FORMAT), + $this->serializer->deserialize($privateKeyPem, PrivateKey::class, PemEncoder::FORMAT) + ); + } catch (\Exception $e) { + throw new AcmeCliException('Loading of account key pair failed', $e); + } + } + + /** + * {@inheritdoc} + */ + public function storeDomainKeyPair($domain, KeyPair $keyPair) + { + try { + $this->save( + $this->getPathForDomain(self::PATH_DOMAIN_KEY_PUBLIC, $domain), + $this->serializer->serialize($keyPair->getPublicKey(), PemEncoder::FORMAT) + ); + + $this->save( + $this->getPathForDomain(self::PATH_DOMAIN_KEY_PRIVATE, $domain), + $this->serializer->serialize($keyPair->getPrivateKey(), PemEncoder::FORMAT) + ); + } catch (\Exception $e) { + throw new AcmeCliException(sprintf('Storing of domain %s key pair failed', $domain), $e); + } + } + + /** + * {@inheritdoc} + */ + public function hasDomainKeyPair($domain) + { + return $this->master->has($this->getPathForDomain(self::PATH_DOMAIN_KEY_PRIVATE, $domain)); + } + + /** + * {@inheritdoc} + */ + public function loadDomainKeyPair($domain) + { + try { + $publicKeyPem = $this->master->read($this->getPathForDomain(self::PATH_DOMAIN_KEY_PUBLIC, $domain)); + $privateKeyPem = $this->master->read($this->getPathForDomain(self::PATH_DOMAIN_KEY_PRIVATE, $domain)); + + return new KeyPair( + $this->serializer->deserialize($publicKeyPem, PublicKey::class, PemEncoder::FORMAT), + $this->serializer->deserialize($privateKeyPem, PrivateKey::class, PemEncoder::FORMAT) + ); + } catch (\Exception $e) { + throw new AcmeCliException(sprintf('Loading of domain %s key pair failed', $domain), $e); + } + } + + /** + * {@inheritdoc} + */ + public function storeDomainAuthorizationChallenge($domain, AuthorizationChallenge $authorizationChallenge) + { + try { + $this->save( + $this->getPathForDomain(self::PATH_CACHE_AUTHORIZATION_CHALLENGE, $domain), + $this->serializer->serialize($authorizationChallenge, JsonEncoder::FORMAT) + ); + } catch (\Exception $e) { + throw new AcmeCliException(sprintf('Storing of domain %s authorization challenge failed', $domain), $e); + } + } + + /** + * {@inheritdoc} + */ + public function hasDomainAuthorizationChallenge($domain) + { + return $this->master->has($this->getPathForDomain(self::PATH_CACHE_AUTHORIZATION_CHALLENGE, $domain)); + } + + /** + * {@inheritdoc} + */ + public function loadDomainAuthorizationChallenge($domain) + { + try { + $json = $this->master->read($this->getPathForDomain(self::PATH_CACHE_AUTHORIZATION_CHALLENGE, $domain)); + + return $this->serializer->deserialize($json, AuthorizationChallenge::class, JsonEncoder::FORMAT); + } catch (\Exception $e) { + throw new AcmeCliException(sprintf('Loading of domain %s authorization challenge failed', $domain), $e); + } + } + + /** + * {@inheritdoc} + */ + public function storeDomainDistinguishedName($domain, DistinguishedName $distinguishedName) + { + try { + $this->save( + $this->getPathForDomain(self::PATH_CACHE_DISTINGUISHED_NAME, $domain), + $this->serializer->serialize($distinguishedName, JsonEncoder::FORMAT) + ); + } catch (\Exception $e) { + throw new AcmeCliException(sprintf('Storing of domain %s distinguished name failed', $domain), $e); + } + } + + /** + * {@inheritdoc} + */ + public function hasDomainDistinguishedName($domain) + { + return $this->master->has($this->getPathForDomain(self::PATH_CACHE_DISTINGUISHED_NAME, $domain)); + } + + /** + * {@inheritdoc} + */ + public function loadDomainDistinguishedName($domain) + { + try { + $json = $this->master->read($this->getPathForDomain(self::PATH_CACHE_DISTINGUISHED_NAME, $domain)); + + return $this->serializer->deserialize($json, DistinguishedName::class, JsonEncoder::FORMAT); + } catch (\Exception $e) { + throw new AcmeCliException(sprintf('Loading of domain %s distinguished name failed', $domain), $e); + } + } + + /** + * {@inheritdoc} + */ + public function storeDomainCertificate($domain, Certificate $certificate) + { + // Simple certificate + $certPem = $this->serializer->serialize($certificate, PemEncoder::FORMAT); + + // Issuer chain + $issuerChain = []; + $issuerCertificate = $certificate->getIssuerCertificate(); + + while (null !== $issuerCertificate) { + $issuerChain[] = $this->serializer->serialize($issuerCertificate, PemEncoder::FORMAT); + $issuerCertificate = $issuerCertificate->getIssuerCertificate(); + } + + $chainPem = implode("\n", $issuerChain); + + // Full chain + $fullChainPem = $certPem.$chainPem; + + // Combined + $keyPair = $this->loadDomainKeyPair($domain); + $combinedPem = $fullChainPem.$this->serializer->serialize($keyPair->getPrivateKey(), PemEncoder::FORMAT); + + // Save + $this->save($this->getPathForDomain(self::PATH_DOMAIN_CERT_CERT, $domain), $certPem); + $this->save($this->getPathForDomain(self::PATH_DOMAIN_CERT_CHAIN, $domain), $chainPem); + $this->save($this->getPathForDomain(self::PATH_DOMAIN_CERT_FULLCHAIN, $domain), $fullChainPem); + $this->save($this->getPathForDomain(self::PATH_DOMAIN_CERT_COMBINED, $domain), $combinedPem); + } + + /** + * {@inheritdoc} + */ + public function hasDomainCertificate($domain) + { + return $this->master->has($this->getPathForDomain(self::PATH_DOMAIN_CERT_FULLCHAIN, $domain)); + } + + /** + * {@inheritdoc} + */ + public function loadDomainCertificate($domain) + { + try { + $pems = explode('-----BEGIN CERTIFICATE-----', $this->master->read($this->getPathForDomain(self::PATH_DOMAIN_CERT_FULLCHAIN, $domain))); + } catch (\Exception $e) { + throw new AcmeCliException(sprintf('Loading of domain %s certificate failed', $domain), $e); + } + + $pems = array_map(function ($item) { + return trim(str_replace('-----END CERTIFICATE-----', '', $item)); + }, $pems); + array_shift($pems); + $pems = array_reverse($pems); + + $certificate = null; + + foreach ($pems as $pem) { + $certificate = new Certificate( + "-----BEGIN CERTIFICATE-----\n".$pem."\n-----END CERTIFICATE-----", + $certificate + ); + } + + return $certificate; + } + + /** + * {@inheritdoc} + */ + public function storeCertificateOrder(array $domains, CertificateOrder $order) + { + try { + $this->save( + $this->getPathForDomainList(self::PATH_CACHE_CERTIFICATE_ORDER, $domains), + $this->serializer->serialize($order, JsonEncoder::FORMAT) + ); + } catch (\Exception $e) { + throw new AcmeCliException(sprintf('Storing of domains %s certificate order failed', implode(', ', $domains)), $e); + } + } + + /** + * {@inheritdoc} + */ + public function hasCertificateOrder(array $domains) + { + return $this->master->has($this->getPathForDomainList(self::PATH_CACHE_CERTIFICATE_ORDER, $domains)); + } + + /** + * {@inheritdoc} + */ + public function loadCertificateOrder(array $domains) + { + try { + $json = $this->master->read($this->getPathForDomainList(self::PATH_CACHE_CERTIFICATE_ORDER, $domains)); + + return $this->serializer->deserialize($json, CertificateOrder::class, JsonEncoder::FORMAT); + } catch (\Exception $e) { + throw new AcmeCliException(sprintf('Loading of domains %s certificate order failed', implode(', ', $domains)), $e); + } + } + + /** + * {@inheritdoc} + */ + public function save($path, $content, $visibility = self::VISIBILITY_PRIVATE) + { + if (!$this->master->has($path)) { + // File creation: remove from backup if it existed and warm-up both master and backup + $this->createAndBackup($path, $content); + } else { + // File update: backup before writing + $this->backupAndUpdate($path, $content); + } + + if ($this->enableBackup) { + $this->backup->setVisibility($path, $visibility); + } + + $this->master->setVisibility($path, $visibility); + } + + private function createAndBackup($path, $content) + { + if ($this->enableBackup) { + if ($this->backup->has($path)) { + $this->backup->delete($path); + } + + $this->backup->write($path, $content); + } + + $this->master->write($path, $content); + } + + private function backupAndUpdate($path, $content) + { + if ($this->enableBackup) { + $oldContent = $this->master->read($path); + + if (false !== $oldContent) { + if ($this->backup->has($path)) { + $this->backup->update($path, $oldContent); + } else { + $this->backup->write($path, $oldContent); + } + } + } + + $this->master->update($path, $content); + } + + private function normalizeDomain($domain) + { + return $domain; + } + + private function normalizeDomainList(array $domains) + { + $normalizedDomains = array_unique(array_map([$this, 'normalizeDomain'], $domains)); + sort($normalizedDomains); + + return (isset($domains[0]) ? $this->normalizeDomain($domains[0]) : '-').'/'.sha1(json_encode($normalizedDomains)); + } +} diff --git a/AcmePhp/Cli/Repository/RepositoryInterface.php b/AcmePhp/Cli/Repository/RepositoryInterface.php new file mode 100644 index 00000000..f42c5277 --- /dev/null +++ b/AcmePhp/Cli/Repository/RepositoryInterface.php @@ -0,0 +1,198 @@ + + * + * For the full copyright and license information, please view the LICENSE + * file that was distributed with this source code. + */ + +namespace AcmePhp\Cli\Repository; + +use AcmePhp\Cli\Exception\AcmeCliException; +use AcmePhp\Core\Protocol\AuthorizationChallenge; +use AcmePhp\Ssl\Certificate; +use AcmePhp\Ssl\CertificateResponse; +use AcmePhp\Ssl\DistinguishedName; +use AcmePhp\Ssl\KeyPair; + +/** + * @author Titouan Galopin + */ +interface RepositoryInterface +{ + const VISIBILITY_PUBLIC = 'public'; + const VISIBILITY_PRIVATE = 'private'; + + /** + * Extract important elements from the given certificate response and store them + * in the repository. + * + * This method will use the distinguished name common name as a domain to store: + * - the key pair + * - the certificate request + * - the certificate + * + * @param CertificateResponse $certificateResponse + * + * @throws AcmeCliException + */ + public function storeCertificateResponse(CertificateResponse $certificateResponse); + + /** + * Store a given key pair as the account key pair (the global key pair used to + * interact with the ACME server). + * + * @param KeyPair $keyPair + * + * @throws AcmeCliException + */ + public function storeAccountKeyPair(KeyPair $keyPair); + + /** + * Check if there is an account key pair in the repository. + * + * @return bool + */ + public function hasAccountKeyPair(); + + /** + * Load the account key pair. + * + * @throws AcmeCliException + * + * @return KeyPair + */ + public function loadAccountKeyPair(); + + /** + * Store a given key pair as associated to a given domain. + * + * @param string $domain + * @param KeyPair $keyPair + * + * @throws AcmeCliException + */ + public function storeDomainKeyPair($domain, KeyPair $keyPair); + + /** + * Check if there is a key pair associated to the given domain in the repository. + * + * @param string $domain + * + * @return bool + */ + public function hasDomainKeyPair($domain); + + /** + * Load the key pair associated to a given domain. + * + * @param string $domain + * + * @throws AcmeCliException + * + * @return KeyPair + */ + public function loadDomainKeyPair($domain); + + /** + * Store a given authorization challenge as associated to a given domain. + * + * @param string $domain + * @param AuthorizationChallenge $authorizationChallenge + * + * @throws AcmeCliException + */ + public function storeDomainAuthorizationChallenge($domain, AuthorizationChallenge $authorizationChallenge); + + /** + * Check if there is an authorization challenge associated to the given domain in the repository. + * + * @param string $domain + * + * @return bool + */ + public function hasDomainAuthorizationChallenge($domain); + + /** + * Load the authorization challenge associated to a given domain. + * + * @param string $domain + * + * @throws AcmeCliException + * + * @return AuthorizationChallenge + */ + public function loadDomainAuthorizationChallenge($domain); + + /** + * Store a given distinguished name as associated to a given domain. + * + * @param string $domain + * @param DistinguishedName $distinguishedName + * + * @throws AcmeCliException + */ + public function storeDomainDistinguishedName($domain, DistinguishedName $distinguishedName); + + /** + * Check if there is a distinguished name associated to the given domain in the repository. + * + * @param string $domain + * + * @return bool + */ + public function hasDomainDistinguishedName($domain); + + /** + * Load the distinguished name associated to a given domain. + * + * @param string $domain + * + * @throws AcmeCliException + * + * @return DistinguishedName + */ + public function loadDomainDistinguishedName($domain); + + /** + * Store a given certificate as associated to a given domain. + * + * @param string $domain + * @param Certificate $certificate + * + * @throws AcmeCliException + */ + public function storeDomainCertificate($domain, Certificate $certificate); + + /** + * Check if there is a certificate associated to the given domain in the repository. + * + * @param string $domain + * + * @return bool + */ + public function hasDomainCertificate($domain); + + /** + * Load the certificate associated to a given domain. + * + * @param string $domain + * + * @throws AcmeCliException + * + * @return Certificate + */ + public function loadDomainCertificate($domain); + + /** + * Save a given string into a given path handling backup. + * + * @param string $path + * @param string $content + * @param string $visibility the visibilty to use for this file + */ + public function save($path, $content, $visibility = self::VISIBILITY_PRIVATE); +} diff --git a/AcmePhp/Cli/Repository/RepositoryV2Interface.php b/AcmePhp/Cli/Repository/RepositoryV2Interface.php new file mode 100644 index 00000000..f3b67e03 --- /dev/null +++ b/AcmePhp/Cli/Repository/RepositoryV2Interface.php @@ -0,0 +1,51 @@ + + * + * For the full copyright and license information, please view the LICENSE + * file that was distributed with this source code. + */ + +namespace AcmePhp\Cli\Repository; + +use AcmePhp\Cli\Exception\AcmeCliException; +use AcmePhp\Core\Protocol\CertificateOrder; + +/** + * @author Titouan Galopin + */ +interface RepositoryV2Interface extends RepositoryInterface +{ + /** + * Store a given certificate as associated to a given domain. + * + * @param array $domains + * @param CertificateOrder $order + * + * @throws AcmeCliException + */ + public function storeCertificateOrder(array $domains, CertificateOrder $order); + + /** + * Check if there is a certificate associated to the given domain in the repository. + * + * @param string $domain + * + * @return bool + */ + public function hasCertificateOrder(array $domains); + + /** + * Load the certificate associated to a given domain. + * + * @param string $domain + * + * @throws AcmeCliException + * + * @return CertificateOrder + */ + public function loadCertificateOrder(array $domains); +} diff --git a/AcmePhp/Cli/Serializer/PemEncoder.php b/AcmePhp/Cli/Serializer/PemEncoder.php new file mode 100644 index 00000000..01d3a15d --- /dev/null +++ b/AcmePhp/Cli/Serializer/PemEncoder.php @@ -0,0 +1,55 @@ + + * + * For the full copyright and license information, please view the LICENSE + * file that was distributed with this source code. + */ + +namespace AcmePhp\Cli\Serializer; + +use Symfony\Component\Serializer\Encoder\DecoderInterface; +use Symfony\Component\Serializer\Encoder\EncoderInterface; + +/** + * @author Titouan Galopin + */ +class PemEncoder implements EncoderInterface, DecoderInterface +{ + const FORMAT = 'pem'; + + /** + * {@inheritdoc} + */ + public function encode($data, $format, array $context = []) + { + return trim($data)."\n"; + } + + /** + * {@inheritdoc} + */ + public function decode($data, $format, array $context = []) + { + return trim($data)."\n"; + } + + /** + * {@inheritdoc} + */ + public function supportsEncoding($format) + { + return self::FORMAT === $format; + } + + /** + * {@inheritdoc} + */ + public function supportsDecoding($format) + { + return self::FORMAT === $format; + } +} diff --git a/AcmePhp/Cli/Serializer/PemNormalizer.php b/AcmePhp/Cli/Serializer/PemNormalizer.php new file mode 100644 index 00000000..564f1b93 --- /dev/null +++ b/AcmePhp/Cli/Serializer/PemNormalizer.php @@ -0,0 +1,55 @@ + + * + * For the full copyright and license information, please view the LICENSE + * file that was distributed with this source code. + */ + +namespace AcmePhp\Cli\Serializer; + +use AcmePhp\Ssl\Certificate; +use AcmePhp\Ssl\Key; +use Symfony\Component\Serializer\Normalizer\DenormalizerInterface; +use Symfony\Component\Serializer\Normalizer\NormalizerInterface; + +/** + * @author Titouan Galopin + */ +class PemNormalizer implements NormalizerInterface, DenormalizerInterface +{ + /** + * {@inheritdoc} + */ + public function normalize($object, $format = null, array $context = []) + { + return $object->getPEM(); + } + + /** + * {@inheritdoc} + */ + public function denormalize($data, $class, $format = null, array $context = []) + { + return new $class($data); + } + + /** + * {@inheritdoc} + */ + public function supportsNormalization($data, $format = null) + { + return is_object($data) && ($data instanceof Certificate || $data instanceof Key); + } + + /** + * {@inheritdoc} + */ + public function supportsDenormalization($data, $type, $format = null) + { + return is_string($data); + } +} From 9dcdd842f40f63fe40ac00f2590c38efe38da971 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Tue, 28 Aug 2018 15:35:31 +0530 Subject: [PATCH 02/40] Move site related files from core to site-command Signed-off-by: Riddhesh Sanghvi --- src/Shutdown_Handler.php | 19 ++ src/Site_Letsencrypt.php | 530 ++++++++++++++++++++++++++++++++++++++ src/class-ee-site.php | 539 +++++++++++++++++++++++++++++++++++++++ src/site-utils.php | 370 +++++++++++++++++++++++++++ 4 files changed, 1458 insertions(+) create mode 100644 src/Shutdown_Handler.php create mode 100644 src/Site_Letsencrypt.php create mode 100644 src/class-ee-site.php create mode 100644 src/site-utils.php diff --git a/src/Shutdown_Handler.php b/src/Shutdown_Handler.php new file mode 100644 index 00000000..2df3c30e --- /dev/null +++ b/src/Shutdown_Handler.php @@ -0,0 +1,19 @@ +getMethod( 'shutDownFunction' ); + $method->setAccessible( true ); + $method->invoke( $site_command[0] ); + } +} diff --git a/src/Site_Letsencrypt.php b/src/Site_Letsencrypt.php new file mode 100644 index 00000000..9c396b94 --- /dev/null +++ b/src/Site_Letsencrypt.php @@ -0,0 +1,530 @@ +conf_dir = EE_CONF_ROOT . '/acme-conf'; + $this->setRepository(); + $this->setAcmeClient(); + } + + private function setAcmeClient() { + + if ( ! $this->repository->hasAccountKeyPair() ) { + EE::debug( 'No account key pair was found, generating one.' ); + EE::debug( 'Generating a key pair' ); + + $keygen = new KeyPairGenerator(); + $accountKeyPair = $keygen->generateKeyPair(); + EE::debug( 'Key pair generated, storing' ); + $this->repository->storeAccountKeyPair( $accountKeyPair ); + } else { + EE::debug( 'Loading account keypair' ); + $accountKeyPair = $this->repository->loadAccountKeyPair(); + } + + $this->accountKeyPair ?? $this->accountKeyPair = $accountKeyPair; + + $secureHttpClient = $this->getSecureHttpClient(); + $csrSigner = new CertificateRequestSigner(); + + $this->client = new AcmeClient( $secureHttpClient, 'https://acme-v02.api.letsencrypt.org/directory', $csrSigner ); + + } + + private function setRepository( $enable_backup = false ) { + $this->serializer ?? $this->serializer = new Serializer( + [ new PemNormalizer(), new GetSetMethodNormalizer() ], + [ new PemEncoder(), new JsonEncoder() ] + ); + $this->master ?? $this->master = new Filesystem( new Local( $this->conf_dir ) ); + $this->backup ?? $this->backup = new Filesystem( new NullAdapter() ); + + $this->repository = new Repository( $this->serializer, $this->master, $this->backup, $enable_backup ); + } + + private function getSecureHttpClient() { + $this->httpClient ?? $this->httpClient = new Client(); + $this->base64SafeEncoder ?? $this->base64SafeEncoder = new Base64SafeEncoder(); + $this->keyParser ?? $this->keyParser = new KeyParser(); + $this->dataSigner ?? $this->dataSigner = new DataSigner(); + $this->serverErrorHandler ?? $this->serverErrorHandler = new ServerErrorHandler(); + + return new SecureHttpClient( + $this->accountKeyPair, + $this->httpClient, + $this->base64SafeEncoder, + $this->keyParser, + $this->dataSigner, + $this->serverErrorHandler + ); + } + + + public function register( $email ) { + try { + $this->client->registerAccount( null, $email ); + } + catch ( Exception $e ) { + EE::warning( $e->getMessage() ); + EE::warning( 'It seems you\'re in local environment or there is some issue with network, please check logs. Skipping letsencrypt.' ); + + return false; + } + EE::debug( "Account with email id: $email registered successfully!" ); + return true; + } + + public function authorize( Array $domains, $site_root, $wildcard = false ) { + $solver = $wildcard ? new SimpleDnsSolver( null, new ConsoleOutput() ) : new SimpleHttpSolver(); + $solverName = $wildcard ? 'dns-01' : 'http-01'; + try { + $order = $this->client->requestOrder( $domains ); + } + catch ( Exception $e ) { + EE::warning( $e->getMessage() ); + EE::warning( 'It seems you\'re in local environment or using non-public domain, please check logs. Skipping letsencrypt.' ); + + return false; + } + + $authorizationChallengesToSolve = []; + foreach ( $order->getAuthorizationsChallenges() as $domainKey => $authorizationChallenges ) { + $authorizationChallenge = null; + foreach ( $authorizationChallenges as $candidate ) { + if ( $solver->supports( $candidate ) ) { + $authorizationChallenge = $candidate; + EE::debug( 'Authorization challenge supported by solver. Solver: ' . $solverName . ' Challenge: ' . $candidate->getType() ); + break; + } + // Should not get here as we are handling it. + EE::debug( 'Authorization challenge not supported by solver. Solver: ' . $solverName . ' Challenge: ' . $candidate->getType() ); + } + if ( null === $authorizationChallenge ) { + throw new ChallengeNotSupportedException(); + } + EE::debug( 'Storing authorization challenge. Domain: ' . $domainKey . ' Challenge: ' . print_r( $authorizationChallenge->toArray(), true ) ); + + $this->repository->storeDomainAuthorizationChallenge( $domainKey, $authorizationChallenge ); + $authorizationChallengesToSolve[] = $authorizationChallenge; + } + + /** @var AuthorizationChallenge $authorizationChallenge */ + foreach ( $authorizationChallengesToSolve as $authorizationChallenge ) { + EE::debug( 'Solving authorization challenge: Domain: ' . $authorizationChallenge->getDomain() . ' Challenge: ' . print_r( $authorizationChallenge->toArray(), true ) ); + $solver->solve( $authorizationChallenge ); + + if ( ! $wildcard ) { + $token = $authorizationChallenge->toArray()['token']; + $payload = $authorizationChallenge->toArray()['payload']; + EE::launch( "mkdir -p $site_root/app/src/.well-known/acme-challenge/" ); + EE::debug( "Creating challange file $site_root/app/src/.well-known/acme-challenge/$token" ); + file_put_contents( "$site_root/app/src/.well-known/acme-challenge/$token", $payload ); + EE::launch( "chown www-data: $site_root/app/src/.well-known/acme-challenge/$token" ); + } + } + + $this->repository->storeCertificateOrder( $domains, $order ); + + return true; + } + + public function check( Array $domains, $wildcard = false ) { + EE::debug( ('Starting check with solver ') . ($wildcard ? 'dns' : 'http') ); + $solver = $wildcard ? new SimpleDnsSolver( null, new ConsoleOutput() ) : new SimpleHttpSolver(); + $validator = new ChainValidator( + [ + new WaitingValidator( new HttpValidator() ), + new WaitingValidator( new DnsValidator() ) + ] + ); + + $order = null; + if ( $this->repository->hasCertificateOrder( $domains ) ) { + $order = $this->repository->loadCertificateOrder( $domains ); + EE::debug( sprintf( 'Loading the authorization token for domains %s ...', implode( ', ', $domains ) ) ); + } + + $authorizationChallengeToCleanup = []; + foreach ( $domains as $domain ) { + if ( $order ) { + $authorizationChallenge = null; + $authorizationChallenges = $order->getAuthorizationChallenges( $domain ); + foreach ( $authorizationChallenges as $challenge ) { + if ( $solver->supports( $challenge ) ) { + $authorizationChallenge = $challenge; + break; + } + } + if ( null === $authorizationChallenge ) { + throw new ChallengeNotSupportedException(); + } + } else { + if ( ! $this->repository->hasDomainAuthorizationChallenge( $domain ) ) { + EE::error( "Domain: $domain not yet authorized/has not been started of with EasyEngine letsencrypt site creation." ); + } + $authorizationChallenge = $this->repository->loadDomainAuthorizationChallenge( $domain ); + if ( ! $solver->supports( $authorizationChallenge ) ) { + throw new ChallengeNotSupportedException(); + } + } + EE::debug( 'Challenge loaded.' ); + + $authorizationChallenge = $this->client->reloadAuthorization( $authorizationChallenge ); + if ( ! $authorizationChallenge->isValid() ) { + EE::debug( sprintf( 'Testing the challenge for domain %s', $domain ) ); + if ( ! $validator->isValid( $authorizationChallenge ) ) { + EE::warning( sprintf( 'Can not valid challenge for domain %s', $domain ) ); + } + + EE::debug( sprintf( 'Requesting authorization check for domain %s', $domain ) ); + try { + $this->client->challengeAuthorization( $authorizationChallenge ); + } + catch ( Exception $e ) { + EE::debug( $e->getMessage() ); + EE::warning( 'Challange Authorization failed. Check logs and check if your domain is pointed correctly to this server.' ); + $site_name = isset( $domains[1] ) ? $domains[1] : $domains[0]; + EE::log( "Re-run `ee site le $site_name` after fixing the issue." ); + + return false; + } + $authorizationChallengeToCleanup[] = $authorizationChallenge; + } + } + + EE::log( 'The authorization check was successful!' ); + + if ( $solver instanceof MultipleChallengesSolverInterface ) { + $solver->cleanupAll( $authorizationChallengeToCleanup ); + } else { + /** @var AuthorizationChallenge $authorizationChallenge */ + foreach ( $authorizationChallengeToCleanup as $authorizationChallenge ) { + $solver->cleanup( $authorizationChallenge ); + } + } + return true; + } + + public function request( $domain, $altNames = [], $email, $force=false ) { + $alternativeNames = array_unique( $altNames ); + sort( $alternativeNames ); + + // Certificate renewal + if ( $this->hasValidCertificate( $domain, $alternativeNames ) ) { + EE::debug( "Certificate found for $domain, executing renewal" ); + + return $this->executeRenewal( $domain, $alternativeNames, $force ); + } + + EE::debug( "No certificate found, executing first request for $domain" ); + + // Certificate first request + return $this->executeFirstRequest( $domain, $alternativeNames, $email ); + } + + /** + * Request a first certificate for the given domain. + * + * @param string $domain + * @param array $alternativeNames + */ + private function executeFirstRequest( $domain, array $alternativeNames, $email ) { + EE::log( 'Executing first request.' ); + + // Generate domain key pair + $keygen = new KeyPairGenerator(); + $domainKeyPair = $keygen->generateKeyPair(); + $this->repository->storeDomainKeyPair( $domain, $domainKeyPair ); + + EE::debug( "$domain Domain key pair generated and stored" ); + + $distinguishedName = $this->getOrCreateDistinguishedName( $domain, $alternativeNames, $email ); + // TODO: ask them ;) + EE::debug( 'Distinguished name informations have been stored locally for this domain (they won\'t be asked on renewal).' ); + + // Order + $domains = array_merge( [ $domain ], $alternativeNames ); + EE::debug( sprintf( 'Loading the order related to the domains %s .', implode( ', ', $domains ) ) ); + if ( ! $this->repository->hasCertificateOrder( $domains ) ) { + EE::error( "$domain has not yet been authorized." ); + } + $order = $this->repository->loadCertificateOrder( $domains ); + + // Request + EE::log( sprintf( 'Requesting first certificate for domain %s.', $domain ) ); + $csr = new CertificateRequest( $distinguishedName, $domainKeyPair ); + $response = $this->client->finalizeOrder( $order, $csr ); + EE::log( 'Certificate received' ); + + // Store + $this->repository->storeDomainCertificate( $domain, $response->getCertificate() ); + EE::log( 'Certificate stored' ); + + // Post-generate actions + $this->moveCertsToNginxProxy( $domain ); + } + + private function moveCertsToNginxProxy( string $domain ) { + + $key_source_file = strtr( $this->conf_dir . '/' . Repository::PATH_DOMAIN_KEY_PRIVATE, [ '{domain}' => $domain ] ); + $crt_source_file = strtr( $this->conf_dir . '/' . Repository::PATH_DOMAIN_CERT_FULLCHAIN, [ '{domain}' => $domain ] ); + $chain_source_file = strtr( $this->conf_dir . '/' . Repository::PATH_DOMAIN_CERT_CHAIN, [ '{domain}' => $domain ] ); + + $key_dest_file = EE_CONF_ROOT . '/nginx/certs/' . $domain . '.key'; + $crt_dest_file = EE_CONF_ROOT . '/nginx/certs/' . $domain . '.crt'; + $chain_dest_file = EE_CONF_ROOT . '/nginx/certs/' . $domain . '.chain.pem'; + + copy( $key_source_file, $key_dest_file ); + copy( $crt_source_file, $crt_dest_file ); + copy( $chain_source_file, $chain_dest_file ); + } + + /** + * Renew a given domain certificate. + * + * @param string $domain + * @param array $alternativeNames + * @param bool $force + */ + private function executeRenewal( $domain, array $alternativeNames, $force = false ) { + try { + // Check expiration date to avoid too much renewal + EE::log( "Loading current certificate for $domain" ); + + $certificate = $this->repository->loadDomainCertificate( $domain ); + + if ( ! $force ) { + $certificateParser = new CertificateParser(); + $parsedCertificate = $certificateParser->parse( $certificate ); + + if ( $parsedCertificate->getValidTo()->format( 'U' ) - time() >= 604800 ) { + + EE::log( + sprintf( + 'Current certificate is valid until %s, renewal is not necessary.', + $parsedCertificate->getValidTo()->format( 'Y-m-d H:i:s' ) + ) + ); + + return; + } + + EE::log( + sprintf( + 'Current certificate will expire in less than a week (%s), renewal is required.', + $parsedCertificate->getValidTo()->format( 'Y-m-d H:i:s' ) + ) + ); + } else { + EE::log( 'Forced renewal.' ); + } + + // Key pair + EE::debug( 'Loading domain key pair...' ); + $domainKeyPair = $this->repository->loadDomainKeyPair( $domain ); + + // Distinguished name + EE::debug( 'Loading domain distinguished name...' ); + $distinguishedName = $this->getOrCreateDistinguishedName( $domain, $alternativeNames ); + + // Order + $domains = array_merge( [ $domain ], $alternativeNames ); + EE::debug( sprintf( 'Loading the order related to the domains %s.', implode( ', ', $domains ) ) ); + if ( ! $this->repository->hasCertificateOrder( $domains ) ) { + EE::error( "$domain has not yet been authorized." ); + } + $order = $this->repository->loadCertificateOrder( $domains ); + + // Renewal + EE::log( sprintf( 'Renewing certificate for domain %s.', $domain ) ); + $csr = new CertificateRequest( $distinguishedName, $domainKeyPair ); + $response = $this->client->finalizeOrder( $order, $csr ); + EE::log( 'Certificate received' ); + + $this->repository->storeDomainCertificate( $domain, $response->getCertificate() ); + $this->log( 'Certificate stored' ); + + // Post-generate actions + $this->moveCertsToNginxProxy( $domain ); + EE::log( 'Certificate renewed successfully!' ); + + } + catch ( \Exception $e ) { + EE::warning( 'A critical error occured during certificate renewal' ); + EE::debug( print_r( $e, true ) ); + + throw $e; + } + catch ( \Throwable $e ) { + EE::warning( 'A critical error occured during certificate renewal' ); + EE::debug( print_r( $e, true ) ); + + throw $e; + } + } + + private function hasValidCertificate( $domain, array $alternativeNames ) { + if ( ! $this->repository->hasDomainCertificate( $domain ) ) { + return false; + } + + if ( ! $this->repository->hasDomainKeyPair( $domain ) ) { + return false; + } + + if ( ! $this->repository->hasDomainDistinguishedName( $domain ) ) { + return false; + } + + if ( $this->repository->loadDomainDistinguishedName( $domain )->getSubjectAlternativeNames() !== $alternativeNames ) { + return false; + } + + return true; + } + + /** + * Retrieve the stored distinguishedName or create a new one if needed. + * + * @param string $domain + * @param array $alternativeNames + * + * @return DistinguishedName + */ + private function getOrCreateDistinguishedName( $domain, array $alternativeNames, $email ) { + if ( $this->repository->hasDomainDistinguishedName( $domain ) ) { + $original = $this->repository->loadDomainDistinguishedName( $domain ); + + $distinguishedName = new DistinguishedName( + $domain, + $original->getCountryName(), + $original->getStateOrProvinceName(), + $original->getLocalityName(), + $original->getOrganizationName(), + $original->getOrganizationalUnitName(), + $original->getEmailAddress(), + $alternativeNames + ); + } else { + // Ask DistinguishedName + $distinguishedName = new DistinguishedName( + $domain, + // TODO: Ask and fill these values properly + 'US', + 'CA', + 'Mountain View', + 'Let\'s Encrypt', + 'Let\'s Encrypt Authority X3', + $email, + $alternativeNames + ); + + } + + $this->repository->storeDomainDistinguishedName( $domain, $distinguishedName ); + + return $distinguishedName; + } + + + public function status() { + $this->master ?? $this->master = new Filesystem( new Local( $this->conf_dir ) ); + + $certificateParser = new CertificateParser(); + + $table = new Table( $output ); + $table->setHeaders( [ 'Domain', 'Issuer', 'Valid from', 'Valid to', 'Needs renewal?' ] ); + + $directories = $this->master->listContents( 'certs' ); + + foreach ( $directories as $directory ) { + if ( 'dir' !== $directory['type'] ) { + continue; + } + + $parsedCertificate = $certificateParser->parse( $this->repository->loadDomainCertificate( $directory['basename'] ) ); + if ( ! $input->getOption( 'all' ) && $parsedCertificate->isExpired() ) { + continue; + } + $domainString = $parsedCertificate->getSubject(); + + $alternativeNames = array_diff( $parsedCertificate->getSubjectAlternativeNames(), [ $parsedCertificate->getSubject() ] ); + if ( count( $alternativeNames ) ) { + sort( $alternativeNames ); + $last = array_pop( $alternativeNames ); + foreach ( $alternativeNames as $alternativeName ) { + $domainString .= "\n ├── " . $alternativeName; + } + $domainString .= "\n └── " . $last; + } + + $table->addRow( + [ + $domainString, + $parsedCertificate->getIssuer(), + $parsedCertificate->getValidFrom()->format( 'Y-m-d H:i:s' ), + $parsedCertificate->getValidTo()->format( 'Y-m-d H:i:s' ), + ( $parsedCertificate->getValidTo()->format( 'U' ) - time() < 604800 ) ? 'Yes' : 'No', + ] + ); + } + + $table->render(); + } + + public function cleanup( $site_root ) { + $challange_dir = "$site_root/app/src/.well-known"; + if ( file_exists( "$site_root/app/src/.well-known" ) ) { + EE::debug( 'Cleaning up webroot files.' ); + EE\Utils\delete_dir( $challange_dir ); + } + } +} diff --git a/src/class-ee-site.php b/src/class-ee-site.php new file mode 100644 index 00000000..1dcbae7a --- /dev/null +++ b/src/class-ee-site.php @@ -0,0 +1,539 @@ +] + * : Render output in a particular format. + * --- + * default: table + * options: + * - table + * - csv + * - yaml + * - json + * - count + * - text + * --- + * + * @subcommand list + */ + public function _list( $args, $assoc_args ) { + + EE\Utils\delem_log( 'site list start' ); + $format = EE\Utils\get_flag_value( $assoc_args, 'format' ); + $enabled = EE\Utils\get_flag_value( $assoc_args, 'enabled' ); + $disabled = EE\Utils\get_flag_value( $assoc_args, 'disabled' ); + + $sites = Site::all(); + + if ( $enabled && ! $disabled ) { + $sites = Site::where( 'is_enabled', true ); + } elseif ( $disabled && ! $enabled ) { + $sites = Site::where( 'is_enabled', false ); + } + + if ( empty( $sites ) ) { + EE::error( 'No sites found!' ); + } + + if ( 'text' === $format ) { + foreach ( $sites as $site ) { + EE::log( $site->site_url ); + } + } else { + $result = array_map( + function ( $site ) { + $site->site = $site->site_url; + $site->status = $site->site_enabled ? 'enabled' : 'disabled'; + + return $site; + }, $sites + ); + + $formatter = new EE\Formatter( $assoc_args, [ 'site', 'status' ] ); + + $formatter->display_items( $result ); + } + + EE\Utils\delem_log( 'site list end' ); + } + + + /** + * Deletes a website. + * + * ## OPTIONS + * + * + * : Name of website to be deleted. + * + * [--yes] + * : Do not prompt for confirmation. + */ + public function delete( $args, $assoc_args ) { + + EE\Utils\delem_log( 'site delete start' ); + $this->populate_site_info( $args ); + EE::confirm( sprintf( 'Are you sure you want to delete %s?', $this->site['url'] ), $assoc_args ); + $this->delete_site( 5, $this->site['url'], $this->site['root'] ); + EE\Utils\delem_log( 'site delete end' ); + } + + /** + * Function to delete the given site. + * + * @param int $level Level of deletion. + * Level - 0: No need of clean-up. + * Level - 1: Clean-up only the site-root. + * Level - 2: Try to remove network. The network may or may not have been created. + * Level - 3: Disconnect & remove network and try to remove containers. The containers may + * not have been created. Level - 4: Remove containers. Level - 5: Remove db entry. + * @param string $site_name Name of the site to be deleted. + * @param string $site_root Webroot of the site. + */ + protected function delete_site( $level, $site_name, $site_root ) { + + $this->fs = new Filesystem(); + $proxy_type = EE_PROXY_TYPE; + if ( $level >= 3 ) { + if ( EE::docker()::docker_compose_down( $site_root ) ) { + EE::log( "[$site_name] Docker Containers removed." ); + } else { + EE::exec( "docker rm -f $(docker ps -q -f=label=created_by=EasyEngine -f=label=site_name=$site_name)" ); + if ( $level > 3 ) { + EE::warning( 'Error in removing docker containers.' ); + } + } + } + + if ( $this->fs->exists( $site_root ) ) { + try { + $this->fs->remove( $site_root ); + } catch ( Exception $e ) { + EE::debug( $e ); + EE::error( 'Could not remove site root. Please check if you have sufficient rights.' ); + } + EE::log( "[$site_name] site root removed." ); + } + + $config_file_path = EE_CONF_ROOT . '/nginx/conf.d/' . $site_name . '-redirect.conf'; + + if ( $this->fs->exists( $config_file_path ) ) { + try { + $this->fs->remove( $config_file_path ); + } catch ( Exception $e ) { + EE::debug( $e ); + EE::error( 'Could not remove site redirection file. Please check if you have sufficient rights.' ); + } + } + + + if ( $level > 4 ) { + if ( $this->ssl ) { + EE::log( 'Removing ssl certs.' ); + $crt_file = EE_CONF_ROOT . "/nginx/certs/$site_name.crt"; + $key_file = EE_CONF_ROOT . "/nginx/certs/$site_name.key"; + $conf_certs = EE_CONF_ROOT . "/acme-conf/certs/$site_name"; + $conf_var = EE_CONF_ROOT . "/acme-conf/var/$site_name"; + + $cert_files = [$conf_certs, $conf_var, $crt_file, $key_file]; + try { + $this->fs->remove( $cert_files ); + } catch ( Exception $e ) { + EE::warning( $e ); + } + } + + if ( Site::find( $site_name )->delete() ) { + EE::log( 'Removed database entry.' ); + } else { + EE::error( 'Could not remove the database entry' ); + } + } + EE::log( "Site $site_name deleted." ); + } + + /** + * Enables a website. It will start the docker containers of the website if they are stopped. + * + * ## OPTIONS + * + * [] + * : Name of website to be enabled. + * + * [--force] + * : Force execution of site up. + */ + public function up( $args, $assoc_args ) { + + EE\Utils\delem_log( 'site enable start' ); + $force = EE\Utils\get_flag_value( $assoc_args, 'force' ); + $args = EE\SiteUtils\auto_site_name( $args, 'site', __FUNCTION__ ); + $this->populate_site_info( $args ); + $site = Site::find( $this->site['url'] ); + + if ( $site->site_enabled && ! $force ) { + EE::error( sprintf( '%s is already enabled!', $site->site_url ) ); + } + + EE::log( sprintf( 'Enabling site %s.', $site->site_url ) ); + + if ( EE::docker()::docker_compose_up( $this->site['root'] ) ) { + $site->site_enabled = 1; + $site->save(); + EE::success( "Site $site->site_url enabled." ); + } else { + EE::error( sprintf( 'There was error in enabling %s. Please check logs.', $site->site_url ) ); + } + EE\Utils\delem_log( 'site enable end' ); + } + + /** + * Disables a website. It will stop and remove the docker containers of the website if they are running. + * + * ## OPTIONS + * + * [] + * : Name of website to be disabled. + */ + public function down( $args, $assoc_args ) { + + EE\Utils\delem_log( 'site disable start' ); + $args = EE\SiteUtils\auto_site_name( $args, 'site', __FUNCTION__ ); + $this->populate_site_info( $args ); + + $site = Site::find($this->site['url']); + + EE::log( sprintf( 'Disabling site %s.', $site->site_url ) ); + + if ( EE::docker()::docker_compose_down( $this->site['root'] ) ) { + $site->site_enabled = 0; + $site->save(); + + EE::success( sprintf( 'Site %s disabled.', $this->site['url'] ) ); + } else { + EE::error( sprintf( 'There was error in disabling %s. Please check logs.', $this->site['url'] ) ); + } + EE\Utils\delem_log( 'site disable end' ); + } + + /** + * Restarts containers associated with site. + * When no service(--nginx etc.) is specified, all site containers will be restarted. + * + * [] + * : Name of the site. + * + * [--all] + * : Restart all containers of site. + * + * [--nginx] + * : Restart nginx container of site. + */ + public function restart( $args, $assoc_args, $whitelisted_containers = [] ) { + + EE\Utils\delem_log( 'site restart start' ); + $args = EE\SiteUtils\auto_site_name( $args, 'site', __FUNCTION__ ); + $all = EE\Utils\get_flag_value( $assoc_args, 'all' ); + $no_service_specified = count( $assoc_args ) === 0; + + $this->populate_site_info( $args ); + + chdir( $this->site['root'] ); + + if ( $all || $no_service_specified ) { + $containers = $whitelisted_containers; + } else { + $containers = array_keys( $assoc_args ); + } + + foreach ( $containers as $container ) { + EE\Siteutils\run_compose_command( 'restart', $container ); + } + EE\Utils\delem_log( 'site restart stop' ); + } + + /** + * Reload services in containers without restarting container(s) associated with site. + * When no service(--nginx etc.) is specified, all services will be reloaded. + * + * [] + * : Name of the site. + * + * [--all] + * : Reload all services of site(which are supported). + * + * [--nginx] + * : Reload nginx service in container. + * + */ + public function reload( $args, $assoc_args, $whitelisted_containers = [], $reload_commands = [] ) { + + EE\Utils\delem_log( 'site reload start' ); + $args = EE\SiteUtils\auto_site_name( $args, 'site', __FUNCTION__ ); + $all = EE\Utils\get_flag_value( $assoc_args, 'all' ); + if ( !array_key_exists( 'nginx', $reload_commands ) ) { + $reload_commands['nginx'] = 'nginx sh -c \'nginx -t && service openresty reload\''; + } + $no_service_specified = count( $assoc_args ) === 0; + + $this->populate_site_info( $args ); + + chdir( $this->site['root'] ); + + if ( $all || $no_service_specified ) { + $this->reload_services( $whitelisted_containers, $reload_commands ); + } else { + $this->reload_services( array_keys( $assoc_args ), $reload_commands ); + } + EE\Utils\delem_log( 'site reload stop' ); + } + + /** + * Executes reload commands. It needs separate handling as commands to reload each service is different. + * + * @param array $services Services to reload. + * @param array $reload_commands Commands to reload the services. + */ + private function reload_services( $services, $reload_commands ) { + + foreach ( $services as $service ) { + EE\SiteUtils\run_compose_command( 'exec', $reload_commands[$service], 'reload', $service ); + } + } + + /** + * Runs the acme le registration and authorization. + * + * @param string $site_name Name of the site for ssl. + * + * @throws Exception + */ + protected function inherit_certs( $site_name ) { + $parent_site_name = implode( '.', array_slice( explode( '.', $site_name ), 1 ) ); + $parent_site = Site::find( $parent_site_name, [ 'site_ssl', 'site_ssl_wildcard' ] ); + + if ( ! $parent_site ) { + throw new Exception( 'Unable to find existing site: ' . $parent_site_name ); + } + + if ( ! $parent_site->site_ssl ) { + throw new Exception( "Cannot inherit from $parent_site_name as site does not have SSL cert" . var_dump( $parent_site ) ); + } + + if ( ! $parent_site->site_ssl_wildcard ) { + throw new Exception( "Cannot inherit from $parent_site_name as site does not have wildcard SSL cert" ); + } + + // We don't have to do anything now as nginx-proxy handles everything for us. + EE::success( 'Inherited certs from parent' ); + } + + /** + * Runs SSL procedure. + * + * @param string $site_name Name of the site for ssl. + * @param string $site_root Webroot of the site. + * @param string $ssl_type Type of ssl cert to issue. + * @param bool $wildcard SSL with wildcard or not. + * + * @throws \EE\ExitException If --ssl flag has unrecognized value + */ + protected function init_ssl( $site_name, $site_root, $ssl_type, $wildcard = false ) { + EE::debug( 'Starting SSL procedure' ); + if ( 'le' === $ssl_type ) { + EE::debug( 'Initializing LE' ); + $this->init_le( $site_name, $site_root, $wildcard ); + } elseif ( 'inherit' === $ssl_type ) { + if ( $wildcard ) { + EE::error( 'Cannot use --wildcard with --ssl=inherit', false ); + } + EE::debug( 'Inheriting certs' ); + $this->inherit_certs( $site_name ); + } else { + EE::error( "Unrecognized value in --ssl flag: $ssl_type" ); + } + } + + /** + * Runs the acme le registration and authorization. + * + * @param string $site_name Name of the site for ssl. + * @param string $site_root Webroot of the site. + * @param bool $wildcard SSL with wildcard or not. + */ + protected function init_le( $site_name, $site_root, $wildcard = false ) { + EE::debug( "Wildcard in init_le: $wildcard" ); + + $this->site['url'] = $site_name; + $this->site['root'] = $site_root; + $this->wildcard = $wildcard; + $client = new Site_Letsencrypt(); + $this->le_mail = EE::get_runner()->config['le-mail'] ?? EE::input( 'Enter your mail id: ' ); + EE::get_runner()->ensure_present_in_config( 'le-mail', $this->le_mail ); + if ( ! $client->register( $this->le_mail ) ) { + $this->ssl = null; + + return; + } + + $domains = $this->get_cert_domains( $site_name, $wildcard ); + + if ( ! $client->authorize( $domains, $this->site['root'], $wildcard ) ) { + $this->le = false; + + return; + } + if ( $wildcard ) { + echo \cli\Colors::colorize( '%YIMPORTANT:%n Run `ee site le ' . $this->site['url'] . '` once the dns changes have propogated to complete the certification generation and installation.', null ); + } else { + $this->le( [], [] ); + } + } + + /** + * Returns all domains required by cert + * + * @param string $site_name Name of site + * @param $wildcard Wildcard cert required? + * + * @return array + */ + private function get_cert_domains( string $site_name, $wildcard ) : array { + $domains = [ $site_name ]; + $has_www = ( strpos( $site_name, 'www.' ) === 0 ); + + if ( $wildcard ) { + $domains[] = "*.{$site_name}"; + } else { + $domains[] = $this->get_www_domain( $site_name ); + } + return $domains; + } + + /** + * If the domain has www in it, returns a domain without www in it. + * Else returns a domain with www in it. + * + * @param string $site_name Name of site + * + * @return string Domain name with or without www + */ + private function get_www_domain( string $site_name ) : string { + $has_www = ( strpos( $site_name, 'www.' ) === 0 ); + + if ( $has_www ) { + return ltrim( $site_name, 'www.' ); + } else { + return 'www.' . $site_name; + } + } + + + /** + * Runs the acme le. + * + * ## OPTIONS + * + * + * : Name of website. + * + * [--force] + * : Force renewal. + */ + public function le( $args = [], $assoc_args = [] ) { + + if ( !isset( $this->site['url'] ) ) { + $this->populate_site_info( $args ); + } + + if ( !isset( $this->le_mail ) ) { + $this->le_mail = EE::get_config( 'le-mail' ) ?? EE::input( 'Enter your mail id: ' ); + } + + $force = EE\Utils\get_flag_value( $assoc_args, 'force' ); + $domains = $this->get_cert_domains( $this->site['url'], $this->wildcard ); + $client = new Site_Letsencrypt(); + + if ( ! $client->check( $domains, $this->wildcard ) ) { + $this->ssl = null; + return; + } + + $san = array_values( array_diff( $domains, [ $this->site['url'] ] ) ); + $client->request( $this->site['url'], $san, $this->le_mail, $force ); + + if ( ! $this->wildcard ) { + $client->cleanup( $this->site['root'] ); + } + EE::launch( 'docker exec ee-nginx-proxy sh -c "/app/docker-entrypoint.sh /usr/local/bin/docker-gen /app/nginx.tmpl /etc/nginx/conf.d/default.conf; /usr/sbin/nginx -s reload"' ); + } + + /** + * Populate basic site info from db. + */ + private function populate_site_info( $args ) { + + $this->site['url'] = EE\Utils\remove_trailing_slash( $args[0] ); + $site = Site::find( $this->site['url'] ); + if ( $site ) { + + $db_select = $site->site_url; + + $this->site['type'] = $site->site_type; + $this->site['root'] = $site->site_fs_path; + $this->ssl = $site->site_ssl; + $this->wildcard = $site->site_ssl_wildcard; + } else { + EE::error( sprintf( 'Site %s does not exist.', $this->site['url'] ) ); + } + } + + abstract public function create( $args, $assoc_args ); + +} diff --git a/src/site-utils.php b/src/site-utils.php new file mode 100644 index 00000000..0e44cb34 --- /dev/null +++ b/src/site-utils.php @@ -0,0 +1,370 @@ +site_fs_path; + if ( substr( $cwd, 0, strlen( $site_path ) ) === $site_path ) { + return $name; + } + } + } + } + + return false; +} + +/** + * Function to set the site-name in the args when ee is running in a site folder and the site-name has not been passed + * in the args. If the site-name could not be found it will throw an error. + * + * @param array $args The passed arguments. + * @param String $command The command passing the arguments to auto-detect site-name. + * @param String $function The function passing the arguments to auto-detect site-name. + * @param integer $arg_pos Argument position where Site-name will be present. + * + * @return array Arguments with site-name set. + */ +function auto_site_name( $args, $command, $function, $arg_pos = 0 ) { + + if ( isset( $args[ $arg_pos ] ) ) { + $possible_site_name = $args[ $arg_pos ]; + if( substr( $possible_site_name, 0, 4 ) === 'http' ) { + $possible_site_name = str_replace(['https','http'],'',$possible_site_name); + } + $url_path = parse_url(EE\Utils\remove_trailing_slash($possible_site_name), PHP_URL_PATH); + if ( Site::find( $url_path ) ) { + return $args; + } + } + $site_name = get_site_name(); + if ( $site_name ) { + if ( isset( $args[ $arg_pos ] ) ) { + EE::error( $args[ $arg_pos ] . " is not a valid site-name. Did you mean `ee $command $function $site_name`?" ); + } + array_splice( $args, $arg_pos, 0, $site_name ); + } else { + EE::error( "Could not find the site you wish to run $command $function command on.\nEither pass it as an argument: `ee $command $function ` \nor run `ee $command $function` from inside the site folder." ); + } + + return $args; +} + + +/** + * Function to check all the required configurations needed to create the site. + * + * Boots up the container if it is stopped or not running. + */ +function init_checks() { + + $proxy_type = EE_PROXY_TYPE; + if ( 'running' !== EE::docker()::container_status( $proxy_type ) ) { + /** + * Checking ports. + */ + $port_80_status = get_curl_info( 'localhost', 80, true ); + $port_443_status = get_curl_info( 'localhost', 443, true ); + + // if any/both the port/s is/are occupied. + if ( ! ( $port_80_status && $port_443_status ) ) { + EE::error( 'Cannot create/start proxy container. Please make sure port 80 and 443 are free.' ); + } else { + + $fs = new Filesystem(); + + if ( ! $fs->exists( EE_CONF_ROOT . '/docker-compose.yml' ) ) { + generate_global_docker_compose_yml( $fs ); + } + + $EE_CONF_ROOT = EE_CONF_ROOT; + if ( ! EE::docker()::docker_network_exists( 'ee-global-network' ) ) { + if ( ! EE::docker()::create_network( 'ee-global-network' ) ) { + EE::error( 'Unable to create network ee-global-network' ); + } + } + if ( EE::docker()::docker_compose_up( EE_CONF_ROOT, [ 'nginx-proxy' ] ) ) { + $fs->dumpFile( "$EE_CONF_ROOT/nginx/conf.d/custom.conf", file_get_contents( EE_ROOT . '/templates/custom.conf.mustache' ) ); + EE::success( "$proxy_type container is up." ); + } else { + EE::error( "There was some error in starting $proxy_type container. Please check logs." ); + } + } + } +} + +/** + * Generates global docker-compose.yml at EE_CONF_ROOT + * + * @param Filesystem $fs Filesystem object to write file + */ +function generate_global_docker_compose_yml( Filesystem $fs ) { + $img_versions = EE\Utils\get_image_versions(); + + $data = [ + 'services' => [ + 'name' => 'nginx-proxy', + 'container_name' => 'ee-nginx-proxy', + 'image' => 'easyengine/nginx-proxy:' . $img_versions['easyengine/nginx-proxy'], + 'restart' => 'always', + 'ports' => [ + '80:80', + '443:443', + ], + 'environment' => [ + 'LOCAL_USER_ID=' . posix_geteuid(), + 'LOCAL_GROUP_ID=' . posix_getegid(), + ], + 'volumes' => [ + EE_CONF_ROOT . '/nginx/certs:/etc/nginx/certs', + EE_CONF_ROOT . '/nginx/dhparam:/etc/nginx/dhparam', + EE_CONF_ROOT . '/nginx/conf.d:/etc/nginx/conf.d', + EE_CONF_ROOT . '/nginx/htpasswd:/etc/nginx/htpasswd', + EE_CONF_ROOT . '/nginx/vhost.d:/etc/nginx/vhost.d', + '/usr/share/nginx/html', + '/var/run/docker.sock:/tmp/docker.sock:ro', + ], + 'networks' => [ + 'global-network', + ], + ], + ]; + + $contents = EE\Utils\mustache_render( EE_ROOT . '/templates/global_docker_compose.yml.mustache', $data ); + $fs->dumpFile( EE_CONF_ROOT . '/docker-compose.yml', $contents ); +} + +/** + * Creates site root directory if does not exist. + * Throws error if it does exist. + * + * @param string $site_root Root directory of the site. + * @param string $site_name Name of the site. + */ +function create_site_root( $site_root, $site_name ) { + + $fs = new Filesystem(); + if ( $fs->exists( $site_root ) ) { + EE::error( "Webroot directory for site $site_name already exists." ); + } + + $whoami = EE::launch( 'whoami', false, true ); + $terminal_username = rtrim( $whoami->stdout ); + + $fs->mkdir( $site_root ); + $fs->chown( $site_root, $terminal_username ); +} + +/** + * Reloads configuration of ee-nginx-proxy container + * + * @return bool + */ +function reload_proxy_configuration() { + return EE::exec( 'docker exec ee-nginx-proxy sh -c "/app/docker-entrypoint.sh /usr/local/bin/docker-gen /app/nginx.tmpl /etc/nginx/conf.d/default.conf; /usr/sbin/nginx -s reload"' ); +} + +/** + * Adds www to non-www redirection to site + * + * @param string $site_name name of the site. + * @param bool $ssl enable ssl or not. + * @param bool $inherit inherit cert or not. + */ +function add_site_redirects( string $site_name, bool $ssl, bool $inherit ) { + + $fs = new Filesystem(); + $confd_path = EE_CONF_ROOT . '/nginx/conf.d/'; + $config_file_path = $confd_path . $site_name . '-redirect.conf'; + $has_www = strpos( $site_name, 'www.' ) === 0; + $cert_site_name = $site_name; + + if ( $inherit ) { + $cert_site_name = implode( '.', array_slice( explode( '.', $site_name ), 1 ) ); + } + + if ( $has_www ) { + $server_name = ltrim( $site_name, '.www' ); + } else { + $server_name = 'www.' . $site_name; + } + + $conf_data = [ + 'site_name' => $site_name, + 'cert_site_name' => $cert_site_name, + 'server_name' => $server_name, + 'ssl' => $ssl, + ]; + + $content = EE\Utils\mustache_render( EE_ROOT . '/templates/redirect.conf.mustache', $conf_data ); + $fs->dumpFile( $config_file_path, ltrim( $content, PHP_EOL ) ); +} + +/** + * Function to create entry in /etc/hosts. + * + * @param string $site_name Name of the site. + */ +function create_etc_hosts_entry( $site_name ) { + + $host_line = LOCALHOST_IP . "\t$site_name"; + $etc_hosts = file_get_contents( '/etc/hosts' ); + if ( ! preg_match( "/\s+$site_name\$/m", $etc_hosts ) ) { + if ( EE::exec( "/bin/bash -c 'echo \"$host_line\" >> /etc/hosts'" ) ) { + EE::success( 'Host entry successfully added.' ); + } else { + EE::warning( "Failed to add $site_name in host entry, Please do it manually!" ); + } + } else { + EE::log( 'Host entry already exists.' ); + } +} + + +/** + * Checking site is running or not. + * + * @param string $site_name Name of the site. + * + * @throws \Exception when fails to connect to site. + */ +function site_status_check( $site_name ) { + + EE::log( 'Checking and verifying site-up status. This may take some time.' ); + $httpcode = get_curl_info( $site_name ); + $i = 0; + while ( 200 !== $httpcode && 302 !== $httpcode && 301 !== $httpcode ) { + EE::debug( "$site_name status httpcode: $httpcode" ); + $httpcode = get_curl_info( $site_name ); + echo '.'; + sleep( 2 ); + if ( $i ++ > 60 ) { + break; + } + } + if ( 200 !== $httpcode && 302 !== $httpcode && 301 !== $httpcode ) { + throw new \Exception( 'Problem connecting to site!' ); + } + +} + +/** + * Function to get httpcode or port occupancy info. + * + * @param string $url url to get info about. + * @param int $port The port to check. + * @param bool $port_info Return port info or httpcode. + * + * @return bool|int port occupied or httpcode. + */ +function get_curl_info( $url, $port = 80, $port_info = false ) { + + $ch = curl_init( $url ); + curl_setopt( $ch, CURLOPT_HEADER, true ); + curl_setopt( $ch, CURLOPT_RETURNTRANSFER, 1 ); + curl_setopt( $ch, CURLOPT_NOBODY, true ); + curl_setopt( $ch, CURLOPT_TIMEOUT, 10 ); + curl_setopt( $ch, CURLOPT_PORT, $port ); + curl_exec( $ch ); + if ( $port_info ) { + return empty( curl_getinfo( $ch, CURLINFO_PRIMARY_IP ) ); + } + + return curl_getinfo( $ch, CURLINFO_HTTP_CODE ); +} + +/** + * Function to pull the latest images and bring up the site containers. + * + * @param string $site_root Root directory of the site. + * @param array $containers The minimum required conatainers to start the site. Default null, leads to starting of all containers. + * + * @throws \Exception when docker-compose up fails. + */ +function start_site_containers( $site_root, $containers = [] ) { + + EE::log( 'Pulling latest images. This may take some time.' ); + chdir( $site_root ); + EE::exec( 'docker-compose pull' ); + EE::log( 'Starting site\'s services.' ); + if ( ! EE::docker()::docker_compose_up( $site_root, $containers ) ) { + throw new \Exception( 'There was some error in docker-compose up.' ); + } +} + + +/** + * Generic function to run a docker compose command. Must be ran inside correct directory. + * + * @param string $action docker-compose action to run. + * @param string $container The container on which action has to be run. + * @param string $action_to_display The action message to be displayed. + * @param string $service_to_display The service message to be displayed. + */ +function run_compose_command( $action, $container, $action_to_display = null, $service_to_display = null ) { + + $display_action = $action_to_display ? $action_to_display : $action; + $display_service = $service_to_display ? $service_to_display : $container; + + EE::log( ucfirst( $display_action ) . 'ing ' . $display_service ); + EE::exec( "docker-compose $action $container", true, true ); +} + +/** + * Function to copy and configure files needed for postfix. + * + * @param string $site_name Name of the site to configure postfix files for. + * @param string $site_conf_dir Configuration directory of the site `site_root/config`. + */ +function set_postfix_files( $site_name, $site_conf_dir ) { + + $fs = new Filesystem(); + $fs->mkdir( $site_conf_dir . '/postfix' ); + $fs->mkdir( $site_conf_dir . '/postfix/ssl' ); + $ssl_dir = $site_conf_dir . '/postfix/ssl'; + + if ( ! EE::exec( sprintf( "openssl req -new -x509 -nodes -days 365 -subj \"/CN=smtp.%s\" -out $ssl_dir/server.crt -keyout $ssl_dir/server.key", $site_name ) ) + && EE::exec( "chmod 0600 $ssl_dir/server.key" ) ) { + throw new Exception( 'Unable to generate ssl key for postfix' ); + } +} + +/** + * Function to execute docker-compose exec calls to postfix to get it configured and running for the site. + * + * @param string $site_name Name of the for which postfix has to be configured. + * @param strin $site_root Site root. + */ +function configure_postfix( $site_name, $site_root ) { + + chdir( $site_root ); + EE::exec( 'docker-compose exec postfix postconf -e \'relayhost =\'' ); + EE::exec( 'docker-compose exec postfix postconf -e \'smtpd_recipient_restrictions = permit_mynetworks\'' ); + $launch = EE::launch( sprintf( 'docker inspect -f \'{{ with (index .IPAM.Config 0) }}{{ .Subnet }}{{ end }}\' %s', $site_name ) ); + $subnet_cidr = trim( $launch->stdout ); + EE::exec( sprintf( 'docker-compose exec postfix postconf -e \'mynetworks = %s 127.0.0.0/8\'', $subnet_cidr ) ); + EE::exec( sprintf( 'docker-compose exec postfix postconf -e \'myhostname = %s\'', $site_name ) ); + EE::exec( 'docker-compose exec postfix postconf -e \'syslog_name = $myhostname\'' ); + EE::exec( 'docker-compose restart postfix' ); +} From ab8c4d0e2825a9bb218e9cce7f22c56fb6f02b52 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Tue, 28 Aug 2018 15:38:13 +0530 Subject: [PATCH 03/40] Update composer dependencies Signed-off-by: Riddhesh Sanghvi --- composer.json | 16 +++++++++------- 1 file changed, 9 insertions(+), 7 deletions(-) diff --git a/composer.json b/composer.json index 8faa7653..2303c5bc 100644 --- a/composer.json +++ b/composer.json @@ -18,12 +18,14 @@ "branch-alias": { "dev-master": "1.x-dev" }, - "bundled": true, - "commands": [ - "site", - "site create", - "site list", - "site delete" - ] + "bundled": true + }, + "require": { + "acmephp/core": "dev-master", + "ext-openssl": "*", + "guzzlehttp/guzzle": "^6.0", + "league/flysystem": "^1.0.19", + "symfony/serializer": "^3.0", + "webmozart/assert": "^1.0" } } From 74dbc3a5baf482b1d9917d88a7a4d36c45f7c1f3 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Tue, 28 Aug 2018 16:45:03 +0530 Subject: [PATCH 04/40] Update namespaces and change Site_Command to HTML site-type Signed-off-by: Riddhesh Sanghvi --- src/HTML.php | 355 +++++++++++++++++++++++++++++++++++++++++++++ src/site-utils.php | 2 +- 2 files changed, 356 insertions(+), 1 deletion(-) create mode 100644 src/HTML.php diff --git a/src/HTML.php b/src/HTML.php new file mode 100644 index 00000000..6f5b5eb8 --- /dev/null +++ b/src/HTML.php @@ -0,0 +1,355 @@ +level = 0; + pcntl_signal( SIGTERM, [ $this, "rollback" ] ); + pcntl_signal( SIGHUP, [ $this, "rollback" ] ); + pcntl_signal( SIGUSR1, [ $this, "rollback" ] ); + pcntl_signal( SIGINT, [ $this, "rollback" ] ); + $shutdown_handler = new Shutdown_Handler(); + register_shutdown_function( [ $shutdown_handler, "cleanup" ], [ &$this ] ); + $this->docker = EE::docker(); + $this->logger = EE::get_file_logger()->withName( 'site_command' ); + $this->fs = new Filesystem(); + } + + /** + * Runs the standard WordPress site installation. + * + * ## OPTIONS + * + * + * : Name of website. + * + * [--ssl=] + * : Enables ssl via letsencrypt certificate. + * + * [--wildcard] + * : Gets wildcard SSL . + * [--type=] + * : Type of the site to be created. Values: html,php,wp. + * + * [--skip-status-check] + * : Skips site status check. + */ + public function create( $args, $assoc_args ) { + + EE\Utils\delem_log( 'site create start' ); + EE::warning( 'This is a beta version. Please don\'t use it in production.' ); + $this->logger->debug( 'args:', $args ); + $this->logger->debug( 'assoc_args:', empty( $assoc_args ) ? [ 'NULL' ] : $assoc_args ); + $this->site['url'] = strtolower( EE\Utils\remove_trailing_slash( $args[0] ) ); + $this->site['type'] = EE\Utils\get_flag_value( $assoc_args, 'type', 'html' ); + if ( 'html' !== $this->site['type'] ) { + EE::error( sprintf( 'Invalid site-type: %s', $this->site['type'] ) ); + } + + if ( Site::find( $this->site['url'] ) ) { + EE::error( sprintf( "Site %1\$s already exists. If you want to re-create it please delete the older one using:\n`ee site delete %1\$s`", $this->site['url'] ) ); + } + + $this->ssl = EE\Utils\get_flag_value( $assoc_args, 'ssl' ); + $this->ssl_wildcard = EE\Utils\get_flag_value( $assoc_args, 'wildcard' ); + $this->skip_chk = EE\Utils\get_flag_value( $assoc_args, 'skip-status-check' ); + + EE\Site\Utils\init_checks(); + + EE::log( 'Configuring project.' ); + + $this->create_site(); + EE\Utils\delem_log( 'site create end' ); + } + + /** + * Display all the relevant site information, credentials and useful links. + * + * [] + * : Name of the website whose info is required. + */ + public function info( $args, $assoc_args ) { + + EE\Utils\delem_log( 'site info start' ); + if ( ! isset( $this->site['url'] ) ) { + $args = EE\Site\Utils\auto_site_name( $args, 'site', __FUNCTION__ ); + $this->populate_site_info( $args ); + } + $ssl = $this->ssl ? 'Enabled' : 'Not Enabled'; + $prefix = ( $this->ssl ) ? 'https://' : 'http://'; + $info = [ + [ 'Site', $prefix . $this->site['url'] ], + [ 'Site Root', $this->site['root'] ], + [ 'SSL', $ssl ], + ]; + + if ( $this->ssl ) { + $info[] = [ 'SSL Wildcard', $this->ssl_wildcard ? 'Yes': 'No' ]; + } + + EE\Utils\format_table( $info ); + + EE\Utils\delem_log( 'site info end' ); + } + + + /** + * Function to configure site and copy all the required files. + */ + private function configure_site_files() { + + $site_conf_dir = $this->site['root'] . '/config'; + $site_docker_yml = $this->site['root'] . '/docker-compose.yml'; + $site_conf_env = $this->site['root'] . '/.env'; + $site_nginx_default_conf = $site_conf_dir . '/nginx/default.conf'; + $site_src_dir = $this->site['root'] . '/app/src'; + $process_user = posix_getpwuid( posix_geteuid() ); + + EE::log( sprintf( 'Creating site %s.', $this->site['url'] ) ); + EE::log( 'Copying configuration files.' ); + + $filter = []; + $filter[] = $this->site['type']; + $site_docker = new Site_Docker(); + $docker_compose_content = $site_docker->generate_docker_compose_yml( $filter ); + $default_conf_content = $default_conf_content = EE\Utils\mustache_render( SITE_TEMPLATE_ROOT . '/config/nginx/default.conf.mustache', [ 'server_name' => $this->site['url'] ] ); + + $env_data = [ + 'virtual_host' => $this->site['url'], + 'user_id' => $process_user['uid'], + 'group_id' => $process_user['gid'], + ]; + $env_content = EE\Utils\mustache_render( SITE_TEMPLATE_ROOT . '/config/.env.mustache', $env_data ); + + try { + $this->fs->dumpFile( $site_docker_yml, $docker_compose_content ); + $this->fs->dumpFile( $site_conf_env, $env_content ); + $this->fs->mkdir( $site_conf_dir ); + $this->fs->mkdir( $site_conf_dir . '/nginx' ); + $this->fs->dumpFile( $site_nginx_default_conf, $default_conf_content ); + + $index_data = [ + 'version' => 'v' . EE_VERSION, + 'site_src_root' => $this->site['root'] . '/app/src', + ]; + $index_html = EE\Utils\mustache_render( SITE_TEMPLATE_ROOT . '/index.html.mustache', $index_data ); + $this->fs->mkdir( $site_src_dir ); + $this->fs->dumpFile( $site_src_dir . '/index.html', $index_html ); + + EE::success( 'Configuration files copied.' ); + } catch ( Exception $e ) { + $this->catch_clean( $e ); + } + } + + /** + * Function to create the site. + */ + private function create_site() { + + $this->site['root'] = WEBROOT . $this->site['url']; + $this->level = 1; + try { + EE\Site\Utils\create_site_root( $this->site['root'], $this->site['url'] ); + $this->level = 3; + $this->configure_site_files(); + + EE\Site\Utils\start_site_containers( $this->site['root'] ); + + EE\Site\Utils\create_etc_hosts_entry( $this->site['url'] ); + if ( ! $this->skip_chk ) { + $this->level = 4; + EE\Site\Utils\site_status_check( $this->site['url'] ); + } + + /* + * This adds http www redirection which is needed for issuing cert for a site. + * i.e. when you create example.com site, certs are issued for example.com and www.example.com + * + * We're issuing certs for both domains as it is needed in order to perform redirection of + * https://www.example.com -> https://example.com + * + * We add redirection config two times in case of ssl as we need http redirection + * when certs are being requested and http+https redirection after we have certs. + */ + EE\Site\Utils\add_site_redirects( $this->site['url'], false, 'inherit' === $this->ssl ); + EE\Site\Utils\reload_proxy_configuration(); + + if ( $this->ssl ) { + $this->init_ssl( $this->site['url'], $this->site['root'], $this->ssl, $this->ssl_wildcard ); + EE\Site\Utils\add_site_redirects( $this->site['url'], true, 'inherit' === $this->ssl ); + EE\Site\Utils\reload_proxy_configuration(); + } + } catch ( Exception $e ) { + $this->catch_clean( $e ); + } + + $this->info( [ $this->site['url'] ], [] ); + $this->create_site_db_entry(); + } + + /** + * Function to save the site configuration entry into database. + */ + private function create_site_db_entry() { + + $ssl = $this->ssl ? 1 : 0; + $ssl_wildcard = $this->ssl_wildcard ? 1 : 0; + + $site = Site::create([ + 'site_url' => $this->site['url'], + 'site_type' => $this->site['type'], + 'site_fs_path' => $this->site['root'], + 'site_ssl' => $ssl, + 'site_ssl_wildcard' => $ssl_wildcard, + 'created_on' => date( 'Y-m-d H:i:s', time() ), + ]); + + try { + if ( $site ) { + EE::log( 'Site entry created.' ); + } else { + throw new Exception( 'Error creating site entry in database.' ); + } + } catch ( Exception $e ) { + $this->catch_clean( $e ); + } + } + + /** + * Populate basic site info from db. + */ + private function populate_site_info( $args ) { + + $this->site['url'] = EE\Utils\remove_trailing_slash( $args[0] ); + + $site = Site::find( $this->site['url'] ); + + if ( $site ) { + $this->site['type'] = $site->site_type; + $this->site['root'] = $site->site_fs_path; + $this->ssl = $site->site_ssl; + $this->ssl_wildcard = $site->site_ssl_wildcard; + } else { + EE::error( sprintf( 'Site %s does not exist.', $this->site['url'] ) ); + } + } + + /** + * @inheritdoc + */ + public function restart( $args, $assoc_args, $whitelisted_containers = [] ) { + $whitelisted_containers = [ 'nginx' ]; + parent::restart( $args, $assoc_args, $whitelisted_containers ); + } + + /** + * @inheritdoc + */ + public function reload( $args, $assoc_args, $whitelisted_containers = [], $reload_commands = [] ) { + $whitelisted_containers = [ 'nginx' ]; + parent::reload( $args, $assoc_args, $whitelisted_containers, $reload_commands = [] ); + } + + /** + * Catch and clean exceptions. + * + * @param Exception $e + */ + private function catch_clean( $e ) { + + EE\Utils\delem_log( 'site cleanup start' ); + EE::warning( $e->getMessage() ); + EE::warning( 'Initiating clean-up.' ); + $this->delete_site( $this->level, $this->site['url'], $this->site['root'] ); + EE\Utils\delem_log( 'site cleanup end' ); + exit; + } + + /** + * Roll back on interrupt. + */ + private function rollback() { + + EE::warning( 'Exiting gracefully after rolling back. This may take some time.' ); + if ( $this->level > 0 ) { + $this->delete_site( $this->level, $this->site['url'], $this->site['root'] ); + } + EE::success( 'Rollback complete. Exiting now.' ); + exit; + } + + /** + * Shutdown function to catch and rollback from fatal errors. + */ + private function shutDownFunction() { + + $error = error_get_last(); + if ( isset( $error ) && $error['type'] === E_ERROR ) { + EE::warning( 'An Error occurred. Initiating clean-up.' ); + $this->logger->error( 'Type: ' . $error['type'] ); + $this->logger->error( 'Message: ' . $error['message'] ); + $this->logger->error( 'File: ' . $error['file'] ); + $this->logger->error( 'Line: ' . $error['line'] ); + $this->rollback(); + } + } +} diff --git a/src/site-utils.php b/src/site-utils.php index 0e44cb34..0783dbc9 100644 --- a/src/site-utils.php +++ b/src/site-utils.php @@ -1,6 +1,6 @@ Date: Tue, 28 Aug 2018 16:46:39 +0530 Subject: [PATCH 05/40] Use constants where applicable Signed-off-by: Riddhesh Sanghvi --- src/site-utils.php | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/site-utils.php b/src/site-utils.php index 0783dbc9..df9825f5 100644 --- a/src/site-utils.php +++ b/src/site-utils.php @@ -125,7 +125,7 @@ function generate_global_docker_compose_yml( Filesystem $fs ) { $data = [ 'services' => [ 'name' => 'nginx-proxy', - 'container_name' => 'ee-nginx-proxy', + 'container_name' => EE_PROXY_TYPE, 'image' => 'easyengine/nginx-proxy:' . $img_versions['easyengine/nginx-proxy'], 'restart' => 'always', 'ports' => [ @@ -177,12 +177,12 @@ function create_site_root( $site_root, $site_name ) { } /** - * Reloads configuration of ee-nginx-proxy container + * Reloads configuration of global-proxy container * * @return bool */ function reload_proxy_configuration() { - return EE::exec( 'docker exec ee-nginx-proxy sh -c "/app/docker-entrypoint.sh /usr/local/bin/docker-gen /app/nginx.tmpl /etc/nginx/conf.d/default.conf; /usr/sbin/nginx -s reload"' ); + return EE::exec( sprintf( 'docker exec %s sh -c "/app/docker-entrypoint.sh /usr/local/bin/docker-gen /app/nginx.tmpl /etc/nginx/conf.d/default.conf; /usr/sbin/nginx -s reload"', EE_PROXY_TYPE ) ); } /** From 7dbb02fe9e9bf3f9e382be01edcb9d4f7e2e6012 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Tue, 28 Aug 2018 17:44:05 +0530 Subject: [PATCH 06/40] Add site-type routing in site-command Signed-off-by: Riddhesh Sanghvi --- src/Site_Command.php | 389 ++++++++++--------------------------------- 1 file changed, 89 insertions(+), 300 deletions(-) diff --git a/src/Site_Command.php b/src/Site_Command.php index e5b8c355..07a600e5 100644 --- a/src/Site_Command.php +++ b/src/Site_Command.php @@ -1,354 +1,143 @@ level = 0; - pcntl_signal( SIGTERM, [ $this, "rollback" ] ); - pcntl_signal( SIGHUP, [ $this, "rollback" ] ); - pcntl_signal( SIGUSR1, [ $this, "rollback" ] ); - pcntl_signal( SIGINT, [ $this, "rollback" ] ); - $shutdown_handler = new Shutdown_Handler(); - register_shutdown_function( [ $shutdown_handler, "cleanup" ], [ &$this ] ); - $this->docker = EE::docker(); - $this->logger = EE::get_file_logger()->withName( 'site_command' ); - $this->fs = new Filesystem(); - } + private static $instance; /** - * Runs the standard WordPress site installation. + * The singleton method to hold the instance of site-command. * - * ## OPTIONS - * - * - * : Name of website. - * - * [--ssl=] - * : Enables ssl via letsencrypt certificate. - * - * [--wildcard] - * : Gets wildcard SSL . - * [--type=] - * : Type of the site to be created. Values: html,php,wp. - * - * [--skip-status-check] - * : Skips site status check. + * @return Object|Site_Command */ - public function create( $args, $assoc_args ) { - - EE\Utils\delem_log( 'site create start' ); - EE::warning( 'This is a beta version. Please don\'t use it in production.' ); - $this->logger->debug( 'args:', $args ); - $this->logger->debug( 'assoc_args:', empty( $assoc_args ) ? [ 'NULL' ] : $assoc_args ); - $this->site['url'] = strtolower( EE\Utils\remove_trailing_slash( $args[0] ) ); - $this->site['type'] = EE\Utils\get_flag_value( $assoc_args, 'type', 'html' ); - if ( 'html' !== $this->site['type'] ) { - EE::error( sprintf( 'Invalid site-type: %s', $this->site['type'] ) ); - } + public static function instance() { - if ( Site::find( $this->site['url'] ) ) { - EE::error( sprintf( "Site %1\$s already exists. If you want to re-create it please delete the older one using:\n`ee site delete %1\$s`", $this->site['url'] ) ); + if ( ! isset( self::$instance ) ) { + self::$instance = new Site_Command(); } - $this->ssl = EE\Utils\get_flag_value( $assoc_args, 'ssl' ); - $this->ssl_wildcard = EE\Utils\get_flag_value( $assoc_args, 'wildcard' ); - $this->skip_chk = EE\Utils\get_flag_value( $assoc_args, 'skip-status-check' ); - - EE\SiteUtils\init_checks(); - - EE::log( 'Configuring project.' ); - - $this->create_site(); - EE\Utils\delem_log( 'site create end' ); + return self::$instance; } /** - * Display all the relevant site information, credentials and useful links. + * Function to register different site-types. * - * [] - * : Name of the website whose info is required. + * @param string $name Name of the site-type. + * @param string $callback The callback function/class for that type. */ - public function info( $args, $assoc_args ) { - - EE\Utils\delem_log( 'site info start' ); - if ( ! isset( $this->site['url'] ) ) { - $args = EE\SiteUtils\auto_site_name( $args, 'site', __FUNCTION__ ); - $this->populate_site_info( $args ); - } - $ssl = $this->ssl ? 'Enabled' : 'Not Enabled'; - $prefix = ( $this->ssl ) ? 'https://' : 'http://'; - $info = [ - [ 'Site', $prefix . $this->site['url'] ], - [ 'Site Root', $this->site['root'] ], - [ 'SSL', $ssl ], - ]; + public static function add_site_type( $name, $callback ) { - if ( $this->ssl ) { - $info[] = [ 'SSL Wildcard', $this->ssl_wildcard ? 'Yes': 'No' ]; + if ( isset( self::$instance->site_types[ $name ] ) ) { + EE::warning( sprintf( '%s site-type has already been previously registered by %s. It will be over-written by the new package class %s. Please update your packages to resolve this.', $name, self::$instance->site_types[ $name ], $callback ) ); } - - EE\Utils\format_table( $info ); - - EE\Utils\delem_log( 'site info end' ); + self::$instance->site_types[ $name ] = $callback; } - /** - * Function to configure site and copy all the required files. + * Method to get the list of registered site-types. + * + * @return array associative array of site-types and their callbacks. */ - private function configure_site_files() { - - $site_conf_dir = $this->site['root'] . '/config'; - $site_docker_yml = $this->site['root'] . '/docker-compose.yml'; - $site_conf_env = $this->site['root'] . '/.env'; - $site_nginx_default_conf = $site_conf_dir . '/nginx/default.conf'; - $site_src_dir = $this->site['root'] . '/app/src'; - $process_user = posix_getpwuid( posix_geteuid() ); - - EE::log( sprintf( 'Creating site %s.', $this->site['url'] ) ); - EE::log( 'Copying configuration files.' ); - - $filter = []; - $filter[] = $this->site['type']; - $site_docker = new Site_Docker(); - $docker_compose_content = $site_docker->generate_docker_compose_yml( $filter ); - $default_conf_content = $default_conf_content = EE\Utils\mustache_render( SITE_TEMPLATE_ROOT . '/config/nginx/default.conf.mustache', [ 'server_name' => $this->site['url'] ] ); - - $env_data = [ - 'virtual_host' => $this->site['url'], - 'user_id' => $process_user['uid'], - 'group_id' => $process_user['gid'], - ]; - $env_content = EE\Utils\mustache_render( SITE_TEMPLATE_ROOT . '/config/.env.mustache', $env_data ); - - try { - $this->fs->dumpFile( $site_docker_yml, $docker_compose_content ); - $this->fs->dumpFile( $site_conf_env, $env_content ); - $this->fs->mkdir( $site_conf_dir ); - $this->fs->mkdir( $site_conf_dir . '/nginx' ); - $this->fs->dumpFile( $site_nginx_default_conf, $default_conf_content ); - - $index_data = [ - 'version' => 'v' . EE_VERSION, - 'site_src_root' => $this->site['root'] . '/app/src', - ]; - $index_html = EE\Utils\mustache_render( SITE_TEMPLATE_ROOT . '/index.html.mustache', $index_data ); - $this->fs->mkdir( $site_src_dir ); - $this->fs->dumpFile( $site_src_dir . '/index.html', $index_html ); - - EE::success( 'Configuration files copied.' ); - } catch ( Exception $e ) { - $this->catch_clean( $e ); - } + public static function get_site_types() { + return self::$instance->site_types; } /** - * Function to create the site. + * Invoked function of site-type routing. Called when `ee site` is invoked. + * Performs the routing to respective site-type passed using either `--type=`, + * Or discovers the type from the site-name and fetches the type from it, + * Or falls down to the default site-type defined by the user, + * Or finally the most basic site-type and the default included in this package, type=html. */ - private function create_site() { - - $this->site['root'] = WEBROOT . $this->site['url']; - $this->level = 1; - try { - EE\SiteUtils\create_site_root( $this->site['root'], $this->site['url'] ); - $this->level = 3; - $this->configure_site_files(); - - EE\SiteUtils\start_site_containers( $this->site['root'] ); - - EE\SiteUtils\create_etc_hosts_entry( $this->site['url'] ); - if ( ! $this->skip_chk ) { - $this->level = 4; - EE\SiteUtils\site_status_check( $this->site['url'] ); - } + public function __invoke( $args, $assoc_args ) { - /* - * This adds http www redirection which is needed for issuing cert for a site. - * i.e. when you create example.com site, certs are issued for example.com and www.example.com - * - * We're issuing certs for both domains as it is needed in order to perform redirection of - * https://www.example.com -> https://example.com - * - * We add redirection config two times in case of ssl as we need http redirection - * when certs are being requested and http+https redirection after we have certs. - */ - EE\SiteUtils\add_site_redirects( $this->site['url'], false, 'inherit' === $this->ssl ); - EE\SiteUtils\reload_proxy_configuration(); + $site_types = self::get_site_types(); - if ( $this->ssl ) { - $this->init_ssl( $this->site['url'], $this->site['root'], $this->ssl, $this->ssl_wildcard ); - EE\SiteUtils\add_site_redirects( $this->site['url'], true, 'inherit' === $this->ssl ); - EE\SiteUtils\reload_proxy_configuration(); - } - } catch ( Exception $e ) { - $this->catch_clean( $e ); + if ( isset( $assoc_args['type'] ) ) { + $type = $assoc_args['type']; + unset( $assoc_args['type'] ); + } else { + $type = $this->determine_type( $args ); } - - $this->info( [ $this->site['url'] ], [] ); - $this->create_site_db_entry(); - } - - /** - * Function to save the site configuration entry into database. - */ - private function create_site_db_entry() { - - $ssl = $this->ssl ? 1 : 0; - $ssl_wildcard = $this->ssl_wildcard ? 1 : 0; - - $site = Site::create([ - 'site_url' => $this->site['url'], - 'site_type' => $this->site['type'], - 'site_fs_path' => $this->site['root'], - 'site_ssl' => $ssl, - 'site_ssl_wildcard' => $ssl_wildcard, - 'created_on' => date( 'Y-m-d H:i:s', time() ), - ]); - - try { - if ( $site ) { - EE::log( 'Site entry created.' ); - } else { - throw new Exception( 'Error creating site entry in database.' ); - } - } catch ( Exception $e ) { - $this->catch_clean( $e ); + array_unshift( $args, 'site' ); + + if ( ! isset( $site_types[ $type ] ) ) { + $error = sprintf( + "'%s' is not a registered site type of 'ee site --type=%s'. See 'ee help site --type=%s' for available subcommands.", + $type, + $type, + $type + ); + EE::error( $error ); } - } - /** - * Populate basic site info from db. - */ - private function populate_site_info( $args ) { + $callback = $site_types[ $type ]; - $this->site['url'] = EE\Utils\remove_trailing_slash( $args[0] ); + $command = EE::get_root_command(); + $leaf_command = CommandFactory::create( 'site', $callback, $command ); + $command->add_subcommand( 'site', $leaf_command ); - $site = Site::find( $this->site['url'] ); - - if ( $site ) { - $this->site['type'] = $site->site_type; - $this->site['root'] = $site->site_fs_path; - $this->ssl = $site->site_ssl; - $this->ssl_wildcard = $site->site_ssl_wildcard; - } else { - EE::error( sprintf( 'Site %s does not exist.', $this->site['url'] ) ); - } - } - - /** - * @inheritdoc - */ - public function restart( $args, $assoc_args, $whitelisted_containers = [] ) { - $whitelisted_containers = [ 'nginx' ]; - parent::restart( $args, $assoc_args, $whitelisted_containers ); + EE::run_command( $args, $assoc_args ); } /** - * @inheritdoc - */ - public function reload( $args, $assoc_args, $whitelisted_containers = [], $reload_commands = [] ) { - $whitelisted_containers = [ 'nginx' ]; - parent::reload( $args, $assoc_args, $whitelisted_containers, $reload_commands = [] ); - } - - /** - * Catch and clean exceptions. + * Function to determine type. + * + * Discovers the type from the site-name and fetches the type from it, + * Or falls down to the default site-type defined by the user, + * Or finally the most basic site-type and the default included in this package, type=html. + * + * @param array $args Command line arguments passed to site-command. * - * @param Exception $e + * @return string site-type. */ - private function catch_clean( $e ) { + private function determine_type( $args ) { - EE\Utils\delem_log( 'site cleanup start' ); - EE::warning( $e->getMessage() ); - EE::warning( 'Initiating clean-up.' ); - $this->delete_site( $this->level, $this->site['url'], $this->site['root'] ); - EE\Utils\delem_log( 'site cleanup end' ); - exit; - } + // default site-type + $type = 'html'; - /** - * Roll back on interrupt. - */ - private function rollback() { + // TODO: get type from config file as below + // $config_type = EE::get_config('type'); + // $type = empty( $config_type ) ? 'html' : $config_type; - EE::warning( 'Exiting gracefully after rolling back. This may take some time.' ); - if ( $this->level > 0 ) { - $this->delete_site( $this->level, $this->site['url'], $this->site['root'] ); + $last_arg = array_pop( $args ); + if ( substr( $last_arg, 0, 4 ) === 'http' ) { + $last_arg = str_replace( [ 'https://', 'http://' ], '', $last_arg ); } - EE::success( 'Rollback complete. Exiting now.' ); - exit; - } + $url_path = EE\Utils\remove_trailing_slash( $last_arg ); - /** - * Shutdown function to catch and rollback from fatal errors. - */ - private function shutDownFunction() { + $arg_search = Site::find( $url_path, [ 'site_type' ] ); - $error = error_get_last(); - if ( isset( $error ) && $error['type'] === E_ERROR ) { - EE::warning( 'An Error occurred. Initiating clean-up.' ); - $this->logger->error( 'Type: ' . $error['type'] ); - $this->logger->error( 'Message: ' . $error['message'] ); - $this->logger->error( 'File: ' . $error['file'] ); - $this->logger->error( 'Line: ' . $error['line'] ); - $this->rollback(); + if ( $arg_search ) { + return $arg_search->site_type; } + + $site_name = EE\Site\Utils\get_site_name(); + if ( $site_name ) { + if ( strpos( $url_path, '.' ) !== false ) { + $args[] = $site_name; + EE::error( + sprintf( + '%s is not a valid site-name. Did you mean `ee site %s`?', + $last_arg, + implode( ' ', $args ) + ) + ); + } + $type = Site::find( $site_name, [ 'site_type' ] )->site_type; + } + + return $type; } } From 253511755b7345fd18b48b70065f660ed5aae4fa Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Tue, 28 Aug 2018 17:45:37 +0530 Subject: [PATCH 07/40] add help hook for site-type and register html site-type Signed-off-by: Riddhesh Sanghvi --- site-command.php | 36 +++++++++++++++++++++++++++++++++++- 1 file changed, 35 insertions(+), 1 deletion(-) diff --git a/site-command.php b/site-command.php index 8a10588b..a36c10c5 100644 --- a/site-command.php +++ b/site-command.php @@ -1,6 +1,8 @@ get_args(); + $args = $all_args[0]; + $assoc_args = $all_args[1]; + + if ( isset( $args[1] ) && 'site' === $args[1] ) { + $site_types = Site_Command::get_site_types(); + if ( ! isset( $assoc_args['type'] ) ) { + EE::error( 'No `--type` passed.' ); + } + $type = $assoc_args['type']; + if ( isset( $site_types[ $type ] ) ) { + $callback = $site_types[ $type ]; + + $command = EE::get_root_command(); + $leaf_command = CommandFactory::create( 'site', $callback, $command ); + $command->add_subcommand( 'site', $leaf_command ); + } else { + $error = sprintf( + "'%s' is not a registered site type of 'ee site --type=%s'. See 'ee help site --type=%s' for available subcommands.", + $type, + $type, + $type + ); + EE::error( $error ); + } + } +} + EE::add_command( 'site', 'Site_Command' ); +EE::add_hook( 'before_invoke:help', 'Before_Help_Command' ); +Site_Command::add_site_type( 'html', 'EE\Site\Type\HTML' ); From a2cbf6a0ee4fae7dd1bb82f787c0dd015f6baf06 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Tue, 28 Aug 2018 18:04:02 +0530 Subject: [PATCH 08/40] Move template from core to site-command Signed-off-by: Riddhesh Sanghvi --- templates/global_docker_compose.yml.mustache | 54 ++++++++++++++++++++ 1 file changed, 54 insertions(+) create mode 100644 templates/global_docker_compose.yml.mustache diff --git a/templates/global_docker_compose.yml.mustache b/templates/global_docker_compose.yml.mustache new file mode 100644 index 00000000..bb231c72 --- /dev/null +++ b/templates/global_docker_compose.yml.mustache @@ -0,0 +1,54 @@ +version: '3.5' + +services: + +{{#services}} + {{name}}: + container_name: {{container_name}} + image: {{image}} + {{#ports.0}} + ports: + {{#ports}} + - "{{.}}" + {{/ports}} + {{/ports.0}} + {{#depends_on}} + depends_on: + - {{.}} + {{/depends_on}} + {{#restart}} + restart: {{.}} + {{/restart}} + {{#command}} + command: {{.}} + {{/command}} + {{#labels.0}} + labels: + {{#labels}} + - "{{.}}" + {{/labels}} + {{/labels.0}} + {{#volumes.0}} + volumes: + {{#volumes}} + - "{{.}}" + {{/volumes}} + {{/volumes.0}} + {{#environment.0}} + environment: + {{#environment}} + - {{.}} + {{/environment}} + {{/environment.0}} + {{#networks.0}} + networks: + {{#networks}} + - {{.}} + {{/networks}} + {{/networks.0}} +{{/services}} + +networks: + global-network: + external: + name: ee-global-network From b947d2470c599ee61cc76deb22291852f4a17e77 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Tue, 28 Aug 2018 18:05:01 +0530 Subject: [PATCH 09/40] Update template path Signed-off-by: Riddhesh Sanghvi --- src/site-utils.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/site-utils.php b/src/site-utils.php index df9825f5..e912cb02 100644 --- a/src/site-utils.php +++ b/src/site-utils.php @@ -151,7 +151,7 @@ function generate_global_docker_compose_yml( Filesystem $fs ) { ], ]; - $contents = EE\Utils\mustache_render( EE_ROOT . '/templates/global_docker_compose.yml.mustache', $data ); + $contents = EE\Utils\mustache_render( SITE_TEMPLATE_ROOT . '/global_docker_compose.yml.mustache', $data ); $fs->dumpFile( EE_CONF_ROOT . '/docker-compose.yml', $contents ); } From 81e5159ee130e449d6a7d174d8cdb86bf46a2a60 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Tue, 28 Aug 2018 18:07:20 +0530 Subject: [PATCH 10/40] Update namespace and filenames Signed-off-by: Riddhesh Sanghvi --- src/{Site_Docker.php => Docker_Compose_Generator.php} | 3 ++- src/HTML.php | 2 +- src/Site_Command.php | 2 +- 3 files changed, 4 insertions(+), 3 deletions(-) rename src/{Site_Docker.php => Docker_Compose_Generator.php} (96%) diff --git a/src/Site_Docker.php b/src/Docker_Compose_Generator.php similarity index 96% rename from src/Site_Docker.php rename to src/Docker_Compose_Generator.php index fe5cfb2b..a60ec648 100644 --- a/src/Site_Docker.php +++ b/src/Docker_Compose_Generator.php @@ -1,8 +1,9 @@ site['type']; - $site_docker = new Site_Docker(); + $site_docker = new Docker_Compose_Generator(); $docker_compose_content = $site_docker->generate_docker_compose_yml( $filter ); $default_conf_content = $default_conf_content = EE\Utils\mustache_render( SITE_TEMPLATE_ROOT . '/config/nginx/default.conf.mustache', [ 'server_name' => $this->site['url'] ] ); diff --git a/src/Site_Command.php b/src/Site_Command.php index 07a600e5..758a469e 100644 --- a/src/Site_Command.php +++ b/src/Site_Command.php @@ -1,7 +1,7 @@ Date: Tue, 28 Aug 2018 18:14:57 +0530 Subject: [PATCH 11/40] Update namespace and filenames Signed-off-by: Riddhesh Sanghvi --- src/HTML.php | 2 +- src/{Docker_Compose_Generator.php => Site_HTML_Docker.php} | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) rename src/{Docker_Compose_Generator.php => Site_HTML_Docker.php} (98%) diff --git a/src/HTML.php b/src/HTML.php index 38c52f41..e169e1da 100644 --- a/src/HTML.php +++ b/src/HTML.php @@ -168,7 +168,7 @@ private function configure_site_files() { $filter = []; $filter[] = $this->site['type']; - $site_docker = new Docker_Compose_Generator(); + $site_docker = new Site_HTML_Docker(); $docker_compose_content = $site_docker->generate_docker_compose_yml( $filter ); $default_conf_content = $default_conf_content = EE\Utils\mustache_render( SITE_TEMPLATE_ROOT . '/config/nginx/default.conf.mustache', [ 'server_name' => $this->site['url'] ] ); diff --git a/src/Docker_Compose_Generator.php b/src/Site_HTML_Docker.php similarity index 98% rename from src/Docker_Compose_Generator.php rename to src/Site_HTML_Docker.php index a60ec648..42c5e357 100644 --- a/src/Docker_Compose_Generator.php +++ b/src/Site_HTML_Docker.php @@ -3,7 +3,7 @@ namespace EE\Site\Type; use function \EE\Utils\mustache_render; -class Docker_Compose_Generator { +class Site_HTML_Docker { /** * Generate docker-compose.yml according to requirement. From b8ff0f0bb0ffc5f6873f1a7c1e600f466624fefd Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Tue, 28 Aug 2018 20:15:52 +0530 Subject: [PATCH 12/40] Load site utility function file Signed-off-by: Riddhesh Sanghvi --- site-command.php | 3 +++ 1 file changed, 3 insertions(+) diff --git a/site-command.php b/site-command.php index a36c10c5..88482c5d 100644 --- a/site-command.php +++ b/site-command.php @@ -15,6 +15,9 @@ require_once $autoload; } +// Load utility functions +require_once 'src/site-utils.php'; + function Before_Help_Command() { $all_args = EE::get_runner()->get_args(); From 2976ebe1adec13aaa718633b3a58c10b2292a337 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Tue, 28 Aug 2018 20:17:47 +0530 Subject: [PATCH 13/40] Add type namespace in shutdownhandler and letsencrypt class Signed-off-by: Riddhesh Sanghvi --- src/Shutdown_Handler.php | 3 ++- src/Site_HTML_Docker.php | 2 +- 2 files changed, 3 insertions(+), 2 deletions(-) diff --git a/src/Shutdown_Handler.php b/src/Shutdown_Handler.php index 2df3c30e..7a7bdebf 100644 --- a/src/Shutdown_Handler.php +++ b/src/Shutdown_Handler.php @@ -1,5 +1,6 @@ getMethod( 'shutDownFunction' ); $method->setAccessible( true ); $method->invoke( $site_command[0] ); diff --git a/src/Site_HTML_Docker.php b/src/Site_HTML_Docker.php index 42c5e357..8ce8dfc2 100644 --- a/src/Site_HTML_Docker.php +++ b/src/Site_HTML_Docker.php @@ -13,7 +13,7 @@ class Site_HTML_Docker { * @return String docker-compose.yml content string. */ public function generate_docker_compose_yml( array $filters = [] ) { - $img_versions = EE\Utils\get_image_versions(); + $img_versions = \EE\Utils\get_image_versions(); $base = []; $restart_default = [ 'name' => 'always' ]; From a7382dfad9cf2eac188582e0346d289b668c70f5 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Tue, 28 Aug 2018 20:18:22 +0530 Subject: [PATCH 14/40] Remove unnecessary namespace Signed-off-by: Riddhesh Sanghvi --- src/Site_Command.php | 1 - 1 file changed, 1 deletion(-) diff --git a/src/Site_Command.php b/src/Site_Command.php index 758a469e..636454d6 100644 --- a/src/Site_Command.php +++ b/src/Site_Command.php @@ -1,6 +1,5 @@ Date: Tue, 28 Aug 2018 20:18:48 +0530 Subject: [PATCH 15/40] Add type namespace to Abstract class Signed-off-by: Riddhesh Sanghvi --- src/class-ee-site.php | 119 +++++++++++++++++++++--------------------- 1 file changed, 60 insertions(+), 59 deletions(-) diff --git a/src/class-ee-site.php b/src/class-ee-site.php index 1dcbae7a..db3ce914 100644 --- a/src/class-ee-site.php +++ b/src/class-ee-site.php @@ -1,5 +1,6 @@ site_url ); + \EE::log( $site->site_url ); } } else { $result = array_map( @@ -95,12 +96,12 @@ function ( $site ) { }, $sites ); - $formatter = new EE\Formatter( $assoc_args, [ 'site', 'status' ] ); + $formatter = new \EE\Formatter( $assoc_args, [ 'site', 'status' ] ); $formatter->display_items( $result ); } - EE\Utils\delem_log( 'site list end' ); + \EE\Utils\delem_log( 'site list end' ); } @@ -117,11 +118,11 @@ function ( $site ) { */ public function delete( $args, $assoc_args ) { - EE\Utils\delem_log( 'site delete start' ); + \EE\Utils\delem_log( 'site delete start' ); $this->populate_site_info( $args ); - EE::confirm( sprintf( 'Are you sure you want to delete %s?', $this->site['url'] ), $assoc_args ); + \EE::confirm( sprintf( 'Are you sure you want to delete %s?', $this->site['url'] ), $assoc_args ); $this->delete_site( 5, $this->site['url'], $this->site['root'] ); - EE\Utils\delem_log( 'site delete end' ); + \EE\Utils\delem_log( 'site delete end' ); } /** @@ -141,12 +142,12 @@ protected function delete_site( $level, $site_name, $site_root ) { $this->fs = new Filesystem(); $proxy_type = EE_PROXY_TYPE; if ( $level >= 3 ) { - if ( EE::docker()::docker_compose_down( $site_root ) ) { - EE::log( "[$site_name] Docker Containers removed." ); + if ( \EE::docker()::docker_compose_down( $site_root ) ) { + \EE::log( "[$site_name] Docker Containers removed." ); } else { - EE::exec( "docker rm -f $(docker ps -q -f=label=created_by=EasyEngine -f=label=site_name=$site_name)" ); + \EE::exec( "docker rm -f $(docker ps -q -f=label=created_by=EasyEngine -f=label=site_name=$site_name)" ); if ( $level > 3 ) { - EE::warning( 'Error in removing docker containers.' ); + \EE::warning( 'Error in removing docker containers.' ); } } } @@ -155,10 +156,10 @@ protected function delete_site( $level, $site_name, $site_root ) { try { $this->fs->remove( $site_root ); } catch ( Exception $e ) { - EE::debug( $e ); - EE::error( 'Could not remove site root. Please check if you have sufficient rights.' ); + \EE::debug( $e ); + \EE::error( 'Could not remove site root. Please check if you have sufficient rights.' ); } - EE::log( "[$site_name] site root removed." ); + \EE::log( "[$site_name] site root removed." ); } $config_file_path = EE_CONF_ROOT . '/nginx/conf.d/' . $site_name . '-redirect.conf'; @@ -167,15 +168,15 @@ protected function delete_site( $level, $site_name, $site_root ) { try { $this->fs->remove( $config_file_path ); } catch ( Exception $e ) { - EE::debug( $e ); - EE::error( 'Could not remove site redirection file. Please check if you have sufficient rights.' ); + \EE::debug( $e ); + \EE::error( 'Could not remove site redirection file. Please check if you have sufficient rights.' ); } } if ( $level > 4 ) { if ( $this->ssl ) { - EE::log( 'Removing ssl certs.' ); + \EE::log( 'Removing ssl certs.' ); $crt_file = EE_CONF_ROOT . "/nginx/certs/$site_name.crt"; $key_file = EE_CONF_ROOT . "/nginx/certs/$site_name.key"; $conf_certs = EE_CONF_ROOT . "/acme-conf/certs/$site_name"; @@ -185,17 +186,17 @@ protected function delete_site( $level, $site_name, $site_root ) { try { $this->fs->remove( $cert_files ); } catch ( Exception $e ) { - EE::warning( $e ); + \EE::warning( $e ); } } if ( Site::find( $site_name )->delete() ) { - EE::log( 'Removed database entry.' ); + \EE::log( 'Removed database entry.' ); } else { - EE::error( 'Could not remove the database entry' ); + \EE::error( 'Could not remove the database entry' ); } } - EE::log( "Site $site_name deleted." ); + \EE::log( "Site $site_name deleted." ); } /** @@ -211,26 +212,26 @@ protected function delete_site( $level, $site_name, $site_root ) { */ public function up( $args, $assoc_args ) { - EE\Utils\delem_log( 'site enable start' ); - $force = EE\Utils\get_flag_value( $assoc_args, 'force' ); + \EE\Utils\delem_log( 'site enable start' ); + $force = \EE\Utils\get_flag_value( $assoc_args, 'force' ); $args = EE\SiteUtils\auto_site_name( $args, 'site', __FUNCTION__ ); $this->populate_site_info( $args ); $site = Site::find( $this->site['url'] ); if ( $site->site_enabled && ! $force ) { - EE::error( sprintf( '%s is already enabled!', $site->site_url ) ); + \EE::error( sprintf( '%s is already enabled!', $site->site_url ) ); } - EE::log( sprintf( 'Enabling site %s.', $site->site_url ) ); + \EE::log( sprintf( 'Enabling site %s.', $site->site_url ) ); - if ( EE::docker()::docker_compose_up( $this->site['root'] ) ) { + if ( \EE::docker()::docker_compose_up( $this->site['root'] ) ) { $site->site_enabled = 1; $site->save(); - EE::success( "Site $site->site_url enabled." ); + \EE::success( "Site $site->site_url enabled." ); } else { - EE::error( sprintf( 'There was error in enabling %s. Please check logs.', $site->site_url ) ); + \EE::error( sprintf( 'There was error in enabling %s. Please check logs.', $site->site_url ) ); } - EE\Utils\delem_log( 'site enable end' ); + \EE\Utils\delem_log( 'site enable end' ); } /** @@ -243,23 +244,23 @@ public function up( $args, $assoc_args ) { */ public function down( $args, $assoc_args ) { - EE\Utils\delem_log( 'site disable start' ); + \EE\Utils\delem_log( 'site disable start' ); $args = EE\SiteUtils\auto_site_name( $args, 'site', __FUNCTION__ ); $this->populate_site_info( $args ); $site = Site::find($this->site['url']); - EE::log( sprintf( 'Disabling site %s.', $site->site_url ) ); + \EE::log( sprintf( 'Disabling site %s.', $site->site_url ) ); - if ( EE::docker()::docker_compose_down( $this->site['root'] ) ) { + if ( \EE::docker()::docker_compose_down( $this->site['root'] ) ) { $site->site_enabled = 0; $site->save(); - EE::success( sprintf( 'Site %s disabled.', $this->site['url'] ) ); + \EE::success( sprintf( 'Site %s disabled.', $this->site['url'] ) ); } else { - EE::error( sprintf( 'There was error in disabling %s. Please check logs.', $this->site['url'] ) ); + \EE::error( sprintf( 'There was error in disabling %s. Please check logs.', $this->site['url'] ) ); } - EE\Utils\delem_log( 'site disable end' ); + \EE\Utils\delem_log( 'site disable end' ); } /** @@ -277,9 +278,9 @@ public function down( $args, $assoc_args ) { */ public function restart( $args, $assoc_args, $whitelisted_containers = [] ) { - EE\Utils\delem_log( 'site restart start' ); + \EE\Utils\delem_log( 'site restart start' ); $args = EE\SiteUtils\auto_site_name( $args, 'site', __FUNCTION__ ); - $all = EE\Utils\get_flag_value( $assoc_args, 'all' ); + $all = \EE\Utils\get_flag_value( $assoc_args, 'all' ); $no_service_specified = count( $assoc_args ) === 0; $this->populate_site_info( $args ); @@ -295,7 +296,7 @@ public function restart( $args, $assoc_args, $whitelisted_containers = [] ) { foreach ( $containers as $container ) { EE\Siteutils\run_compose_command( 'restart', $container ); } - EE\Utils\delem_log( 'site restart stop' ); + \EE\Utils\delem_log( 'site restart stop' ); } /** @@ -314,9 +315,9 @@ public function restart( $args, $assoc_args, $whitelisted_containers = [] ) { */ public function reload( $args, $assoc_args, $whitelisted_containers = [], $reload_commands = [] ) { - EE\Utils\delem_log( 'site reload start' ); + \EE\Utils\delem_log( 'site reload start' ); $args = EE\SiteUtils\auto_site_name( $args, 'site', __FUNCTION__ ); - $all = EE\Utils\get_flag_value( $assoc_args, 'all' ); + $all = \EE\Utils\get_flag_value( $assoc_args, 'all' ); if ( !array_key_exists( 'nginx', $reload_commands ) ) { $reload_commands['nginx'] = 'nginx sh -c \'nginx -t && service openresty reload\''; } @@ -331,7 +332,7 @@ public function reload( $args, $assoc_args, $whitelisted_containers = [], $reloa } else { $this->reload_services( array_keys( $assoc_args ), $reload_commands ); } - EE\Utils\delem_log( 'site reload stop' ); + \EE\Utils\delem_log( 'site reload stop' ); } /** @@ -371,7 +372,7 @@ protected function inherit_certs( $site_name ) { } // We don't have to do anything now as nginx-proxy handles everything for us. - EE::success( 'Inherited certs from parent' ); + \EE::success( 'Inherited certs from parent' ); } /** @@ -385,18 +386,18 @@ protected function inherit_certs( $site_name ) { * @throws \EE\ExitException If --ssl flag has unrecognized value */ protected function init_ssl( $site_name, $site_root, $ssl_type, $wildcard = false ) { - EE::debug( 'Starting SSL procedure' ); + \EE::debug( 'Starting SSL procedure' ); if ( 'le' === $ssl_type ) { - EE::debug( 'Initializing LE' ); + \EE::debug( 'Initializing LE' ); $this->init_le( $site_name, $site_root, $wildcard ); } elseif ( 'inherit' === $ssl_type ) { if ( $wildcard ) { - EE::error( 'Cannot use --wildcard with --ssl=inherit', false ); + \EE::error( 'Cannot use --wildcard with --ssl=inherit', false ); } - EE::debug( 'Inheriting certs' ); + \EE::debug( 'Inheriting certs' ); $this->inherit_certs( $site_name ); } else { - EE::error( "Unrecognized value in --ssl flag: $ssl_type" ); + \EE::error( "Unrecognized value in --ssl flag: $ssl_type" ); } } @@ -408,14 +409,14 @@ protected function init_ssl( $site_name, $site_root, $ssl_type, $wildcard = fals * @param bool $wildcard SSL with wildcard or not. */ protected function init_le( $site_name, $site_root, $wildcard = false ) { - EE::debug( "Wildcard in init_le: $wildcard" ); + \EE::debug( "Wildcard in init_le: $wildcard" ); $this->site['url'] = $site_name; $this->site['root'] = $site_root; $this->wildcard = $wildcard; $client = new Site_Letsencrypt(); - $this->le_mail = EE::get_runner()->config['le-mail'] ?? EE::input( 'Enter your mail id: ' ); - EE::get_runner()->ensure_present_in_config( 'le-mail', $this->le_mail ); + $this->le_mail = \EE::get_runner()->config['le-mail'] ?? \EE::input( 'Enter your mail id: ' ); + \EE::get_runner()->ensure_present_in_config( 'le-mail', $this->le_mail ); if ( ! $client->register( $this->le_mail ) ) { $this->ssl = null; @@ -493,10 +494,10 @@ public function le( $args = [], $assoc_args = [] ) { } if ( !isset( $this->le_mail ) ) { - $this->le_mail = EE::get_config( 'le-mail' ) ?? EE::input( 'Enter your mail id: ' ); + $this->le_mail = \EE::get_config( 'le-mail' ) ?? \EE::input( 'Enter your mail id: ' ); } - $force = EE\Utils\get_flag_value( $assoc_args, 'force' ); + $force = \EE\Utils\get_flag_value( $assoc_args, 'force' ); $domains = $this->get_cert_domains( $this->site['url'], $this->wildcard ); $client = new Site_Letsencrypt(); @@ -511,7 +512,7 @@ public function le( $args = [], $assoc_args = [] ) { if ( ! $this->wildcard ) { $client->cleanup( $this->site['root'] ); } - EE::launch( 'docker exec ee-nginx-proxy sh -c "/app/docker-entrypoint.sh /usr/local/bin/docker-gen /app/nginx.tmpl /etc/nginx/conf.d/default.conf; /usr/sbin/nginx -s reload"' ); + \EE::launch( 'docker exec ee-nginx-proxy sh -c "/app/docker-entrypoint.sh /usr/local/bin/docker-gen /app/nginx.tmpl /etc/nginx/conf.d/default.conf; /usr/sbin/nginx -s reload"' ); } /** @@ -519,7 +520,7 @@ public function le( $args = [], $assoc_args = [] ) { */ private function populate_site_info( $args ) { - $this->site['url'] = EE\Utils\remove_trailing_slash( $args[0] ); + $this->site['url'] = \EE\Utils\remove_trailing_slash( $args[0] ); $site = Site::find( $this->site['url'] ); if ( $site ) { @@ -530,7 +531,7 @@ private function populate_site_info( $args ) { $this->ssl = $site->site_ssl; $this->wildcard = $site->site_ssl_wildcard; } else { - EE::error( sprintf( 'Site %s does not exist.', $this->site['url'] ) ); + \EE::error( sprintf( 'Site %s does not exist.', $this->site['url'] ) ); } } From ebd3eff308be30a9c8cc0ac0b748b1c1a76beb42 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Tue, 28 Aug 2018 20:19:16 +0530 Subject: [PATCH 16/40] Add type namespace to letsencrypt class Signed-off-by: Riddhesh Sanghvi --- src/Site_Letsencrypt.php | 1 + 1 file changed, 1 insertion(+) diff --git a/src/Site_Letsencrypt.php b/src/Site_Letsencrypt.php index 9c396b94..fb2afb39 100644 --- a/src/Site_Letsencrypt.php +++ b/src/Site_Letsencrypt.php @@ -1,5 +1,6 @@ Date: Tue, 28 Aug 2018 20:25:59 +0530 Subject: [PATCH 17/40] Update EE usage to \EE acc to new namespace Signed-off-by: Riddhesh Sanghvi --- src/HTML.php | 87 ++++++++++++++++++++++++++-------------------------- 1 file changed, 44 insertions(+), 43 deletions(-) diff --git a/src/HTML.php b/src/HTML.php index e169e1da..f55dc371 100644 --- a/src/HTML.php +++ b/src/HTML.php @@ -14,6 +14,7 @@ */ namespace EE\Site\Type; +use \EE\Model\Site; use \Symfony\Component\Filesystem\Filesystem; @@ -25,7 +26,7 @@ class HTML extends EE_Site_Command { private $site; /** - * @var object $docker Object to access `EE::docker()` functions. + * @var object $docker Object to access `\EE::docker()` functions. */ private $docker; @@ -68,8 +69,8 @@ public function __construct() { pcntl_signal( SIGINT, [ $this, "rollback" ] ); $shutdown_handler = new Shutdown_Handler(); register_shutdown_function( [ $shutdown_handler, "cleanup" ], [ &$this ] ); - $this->docker = EE::docker(); - $this->logger = EE::get_file_logger()->withName( 'site_command' ); + $this->docker = \EE::docker(); + $this->logger = \EE::get_file_logger()->withName( 'site_command' ); $this->fs = new Filesystem(); } @@ -94,30 +95,30 @@ public function __construct() { */ public function create( $args, $assoc_args ) { - EE\Utils\delem_log( 'site create start' ); - EE::warning( 'This is a beta version. Please don\'t use it in production.' ); + \EE\Utils\delem_log( 'site create start' ); + \EE::warning( 'This is a beta version. Please don\'t use it in production.' ); $this->logger->debug( 'args:', $args ); $this->logger->debug( 'assoc_args:', empty( $assoc_args ) ? [ 'NULL' ] : $assoc_args ); - $this->site['url'] = strtolower( EE\Utils\remove_trailing_slash( $args[0] ) ); - $this->site['type'] = EE\Utils\get_flag_value( $assoc_args, 'type', 'html' ); + $this->site['url'] = strtolower( \EE\Utils\remove_trailing_slash( $args[0] ) ); + $this->site['type'] = \EE\Utils\get_flag_value( $assoc_args, 'type', 'html' ); if ( 'html' !== $this->site['type'] ) { - EE::error( sprintf( 'Invalid site-type: %s', $this->site['type'] ) ); + \EE::error( sprintf( 'Invalid site-type: %s', $this->site['type'] ) ); } if ( Site::find( $this->site['url'] ) ) { - EE::error( sprintf( "Site %1\$s already exists. If you want to re-create it please delete the older one using:\n`ee site delete %1\$s`", $this->site['url'] ) ); + \EE::error( sprintf( "Site %1\$s already exists. If you want to re-create it please delete the older one using:\n`ee site delete %1\$s`", $this->site['url'] ) ); } - $this->ssl = EE\Utils\get_flag_value( $assoc_args, 'ssl' ); - $this->ssl_wildcard = EE\Utils\get_flag_value( $assoc_args, 'wildcard' ); - $this->skip_chk = EE\Utils\get_flag_value( $assoc_args, 'skip-status-check' ); + $this->ssl = \EE\Utils\get_flag_value( $assoc_args, 'ssl' ); + $this->ssl_wildcard = \EE\Utils\get_flag_value( $assoc_args, 'wildcard' ); + $this->skip_chk = \EE\Utils\get_flag_value( $assoc_args, 'skip-status-check' ); - EE\Site\Utils\init_checks(); + \EE\Site\Utils\init_checks(); - EE::log( 'Configuring project.' ); + \EE::log( 'Configuring project.' ); $this->create_site(); - EE\Utils\delem_log( 'site create end' ); + \EE\Utils\delem_log( 'site create end' ); } /** @@ -128,9 +129,9 @@ public function create( $args, $assoc_args ) { */ public function info( $args, $assoc_args ) { - EE\Utils\delem_log( 'site info start' ); + \EE\Utils\delem_log( 'site info start' ); if ( ! isset( $this->site['url'] ) ) { - $args = EE\Site\Utils\auto_site_name( $args, 'site', __FUNCTION__ ); + $args = \EE\Site\Utils\auto_site_name( $args, 'site', __FUNCTION__ ); $this->populate_site_info( $args ); } $ssl = $this->ssl ? 'Enabled' : 'Not Enabled'; @@ -145,9 +146,9 @@ public function info( $args, $assoc_args ) { $info[] = [ 'SSL Wildcard', $this->ssl_wildcard ? 'Yes': 'No' ]; } - EE\Utils\format_table( $info ); + \EE\Utils\format_table( $info ); - EE\Utils\delem_log( 'site info end' ); + \EE\Utils\delem_log( 'site info end' ); } @@ -163,21 +164,21 @@ private function configure_site_files() { $site_src_dir = $this->site['root'] . '/app/src'; $process_user = posix_getpwuid( posix_geteuid() ); - EE::log( sprintf( 'Creating site %s.', $this->site['url'] ) ); - EE::log( 'Copying configuration files.' ); + \EE::log( sprintf( 'Creating site %s.', $this->site['url'] ) ); + \EE::log( 'Copying configuration files.' ); $filter = []; $filter[] = $this->site['type']; $site_docker = new Site_HTML_Docker(); $docker_compose_content = $site_docker->generate_docker_compose_yml( $filter ); - $default_conf_content = $default_conf_content = EE\Utils\mustache_render( SITE_TEMPLATE_ROOT . '/config/nginx/default.conf.mustache', [ 'server_name' => $this->site['url'] ] ); + $default_conf_content = $default_conf_content = \EE\Utils\mustache_render( SITE_TEMPLATE_ROOT . '/config/nginx/default.conf.mustache', [ 'server_name' => $this->site['url'] ] ); $env_data = [ 'virtual_host' => $this->site['url'], 'user_id' => $process_user['uid'], 'group_id' => $process_user['gid'], ]; - $env_content = EE\Utils\mustache_render( SITE_TEMPLATE_ROOT . '/config/.env.mustache', $env_data ); + $env_content = \EE\Utils\mustache_render( SITE_TEMPLATE_ROOT . '/config/.env.mustache', $env_data ); try { $this->fs->dumpFile( $site_docker_yml, $docker_compose_content ); @@ -190,11 +191,11 @@ private function configure_site_files() { 'version' => 'v' . EE_VERSION, 'site_src_root' => $this->site['root'] . '/app/src', ]; - $index_html = EE\Utils\mustache_render( SITE_TEMPLATE_ROOT . '/index.html.mustache', $index_data ); + $index_html = \EE\Utils\mustache_render( SITE_TEMPLATE_ROOT . '/index.html.mustache', $index_data ); $this->fs->mkdir( $site_src_dir ); $this->fs->dumpFile( $site_src_dir . '/index.html', $index_html ); - EE::success( 'Configuration files copied.' ); + \EE::success( 'Configuration files copied.' ); } catch ( Exception $e ) { $this->catch_clean( $e ); } @@ -208,16 +209,16 @@ private function create_site() { $this->site['root'] = WEBROOT . $this->site['url']; $this->level = 1; try { - EE\Site\Utils\create_site_root( $this->site['root'], $this->site['url'] ); + \EE\Site\Utils\create_site_root( $this->site['root'], $this->site['url'] ); $this->level = 3; $this->configure_site_files(); - EE\Site\Utils\start_site_containers( $this->site['root'] ); + \EE\Site\Utils\start_site_containers( $this->site['root'] ); - EE\Site\Utils\create_etc_hosts_entry( $this->site['url'] ); + \EE\Site\Utils\create_etc_hosts_entry( $this->site['url'] ); if ( ! $this->skip_chk ) { $this->level = 4; - EE\Site\Utils\site_status_check( $this->site['url'] ); + \EE\Site\Utils\site_status_check( $this->site['url'] ); } /* @@ -230,13 +231,13 @@ private function create_site() { * We add redirection config two times in case of ssl as we need http redirection * when certs are being requested and http+https redirection after we have certs. */ - EE\Site\Utils\add_site_redirects( $this->site['url'], false, 'inherit' === $this->ssl ); - EE\Site\Utils\reload_proxy_configuration(); + \EE\Site\Utils\add_site_redirects( $this->site['url'], false, 'inherit' === $this->ssl ); + \EE\Site\Utils\reload_proxy_configuration(); if ( $this->ssl ) { $this->init_ssl( $this->site['url'], $this->site['root'], $this->ssl, $this->ssl_wildcard ); - EE\Site\Utils\add_site_redirects( $this->site['url'], true, 'inherit' === $this->ssl ); - EE\Site\Utils\reload_proxy_configuration(); + \EE\Site\Utils\add_site_redirects( $this->site['url'], true, 'inherit' === $this->ssl ); + \EE\Site\Utils\reload_proxy_configuration(); } } catch ( Exception $e ) { $this->catch_clean( $e ); @@ -265,7 +266,7 @@ private function create_site_db_entry() { try { if ( $site ) { - EE::log( 'Site entry created.' ); + \EE::log( 'Site entry created.' ); } else { throw new Exception( 'Error creating site entry in database.' ); } @@ -279,7 +280,7 @@ private function create_site_db_entry() { */ private function populate_site_info( $args ) { - $this->site['url'] = EE\Utils\remove_trailing_slash( $args[0] ); + $this->site['url'] = \EE\Utils\remove_trailing_slash( $args[0] ); $site = Site::find( $this->site['url'] ); @@ -289,7 +290,7 @@ private function populate_site_info( $args ) { $this->ssl = $site->site_ssl; $this->ssl_wildcard = $site->site_ssl_wildcard; } else { - EE::error( sprintf( 'Site %s does not exist.', $this->site['url'] ) ); + \EE::error( sprintf( 'Site %s does not exist.', $this->site['url'] ) ); } } @@ -316,11 +317,11 @@ public function reload( $args, $assoc_args, $whitelisted_containers = [], $reloa */ private function catch_clean( $e ) { - EE\Utils\delem_log( 'site cleanup start' ); - EE::warning( $e->getMessage() ); - EE::warning( 'Initiating clean-up.' ); + \EE\Utils\delem_log( 'site cleanup start' ); + \EE::warning( $e->getMessage() ); + \EE::warning( 'Initiating clean-up.' ); $this->delete_site( $this->level, $this->site['url'], $this->site['root'] ); - EE\Utils\delem_log( 'site cleanup end' ); + \EE\Utils\delem_log( 'site cleanup end' ); exit; } @@ -329,11 +330,11 @@ private function catch_clean( $e ) { */ private function rollback() { - EE::warning( 'Exiting gracefully after rolling back. This may take some time.' ); + \EE::warning( 'Exiting gracefully after rolling back. This may take some time.' ); if ( $this->level > 0 ) { $this->delete_site( $this->level, $this->site['url'], $this->site['root'] ); } - EE::success( 'Rollback complete. Exiting now.' ); + \EE::success( 'Rollback complete. Exiting now.' ); exit; } @@ -344,7 +345,7 @@ private function shutDownFunction() { $error = error_get_last(); if ( isset( $error ) && $error['type'] === E_ERROR ) { - EE::warning( 'An Error occurred. Initiating clean-up.' ); + \EE::warning( 'An Error occurred. Initiating clean-up.' ); $this->logger->error( 'Type: ' . $error['type'] ); $this->logger->error( 'Message: ' . $error['message'] ); $this->logger->error( 'File: ' . $error['file'] ); From a8af7f180b93e4b0f9b2460da33912221333c50a Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Tue, 28 Aug 2018 21:52:06 +0530 Subject: [PATCH 18/40] Update getting of args in help and set default type Signed-off-by: Riddhesh Sanghvi --- site-command.php | 17 ++++++++--------- 1 file changed, 8 insertions(+), 9 deletions(-) diff --git a/site-command.php b/site-command.php index 88482c5d..a672fcf8 100644 --- a/site-command.php +++ b/site-command.php @@ -18,18 +18,17 @@ // Load utility functions require_once 'src/site-utils.php'; -function Before_Help_Command() { +function Before_Help_Command( $args, $assoc_args ) { - $all_args = EE::get_runner()->get_args(); - $args = $all_args[0]; - $assoc_args = $all_args[1]; - - if ( isset( $args[1] ) && 'site' === $args[1] ) { + if ( isset( $args[0] ) && 'site' === $args[0] ) { $site_types = Site_Command::get_site_types(); - if ( ! isset( $assoc_args['type'] ) ) { - EE::error( 'No `--type` passed.' ); + if ( isset( $assoc_args['type'] ) ) { + $type = $assoc_args['type']; + } else { + //TODO: get from config. + $type = 'html'; } - $type = $assoc_args['type']; + if ( isset( $site_types[ $type ] ) ) { $callback = $site_types[ $type ]; From 5df0eafcc1afbc1cf3feb2c9f2dd480f002fc9e4 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Tue, 28 Aug 2018 21:56:01 +0530 Subject: [PATCH 19/40] Update EE usage to \EE for letsencrypt class Signed-off-by: Riddhesh Sanghvi --- src/Site_Letsencrypt.php | 108 +++++++++++++++++++-------------------- 1 file changed, 54 insertions(+), 54 deletions(-) diff --git a/src/Site_Letsencrypt.php b/src/Site_Letsencrypt.php index fb2afb39..56e426d2 100644 --- a/src/Site_Letsencrypt.php +++ b/src/Site_Letsencrypt.php @@ -59,15 +59,15 @@ function __construct() { private function setAcmeClient() { if ( ! $this->repository->hasAccountKeyPair() ) { - EE::debug( 'No account key pair was found, generating one.' ); - EE::debug( 'Generating a key pair' ); + \EE::debug( 'No account key pair was found, generating one.' ); + \EE::debug( 'Generating a key pair' ); $keygen = new KeyPairGenerator(); $accountKeyPair = $keygen->generateKeyPair(); - EE::debug( 'Key pair generated, storing' ); + \EE::debug( 'Key pair generated, storing' ); $this->repository->storeAccountKeyPair( $accountKeyPair ); } else { - EE::debug( 'Loading account keypair' ); + \EE::debug( 'Loading account keypair' ); $accountKeyPair = $this->repository->loadAccountKeyPair(); } @@ -114,12 +114,12 @@ public function register( $email ) { $this->client->registerAccount( null, $email ); } catch ( Exception $e ) { - EE::warning( $e->getMessage() ); - EE::warning( 'It seems you\'re in local environment or there is some issue with network, please check logs. Skipping letsencrypt.' ); + \EE::warning( $e->getMessage() ); + \EE::warning( 'It seems you\'re in local environment or there is some issue with network, please check logs. Skipping letsencrypt.' ); return false; } - EE::debug( "Account with email id: $email registered successfully!" ); + \EE::debug( "Account with email id: $email registered successfully!" ); return true; } @@ -130,8 +130,8 @@ public function authorize( Array $domains, $site_root, $wildcard = false ) { $order = $this->client->requestOrder( $domains ); } catch ( Exception $e ) { - EE::warning( $e->getMessage() ); - EE::warning( 'It seems you\'re in local environment or using non-public domain, please check logs. Skipping letsencrypt.' ); + \EE::warning( $e->getMessage() ); + \EE::warning( 'It seems you\'re in local environment or using non-public domain, please check logs. Skipping letsencrypt.' ); return false; } @@ -142,16 +142,16 @@ public function authorize( Array $domains, $site_root, $wildcard = false ) { foreach ( $authorizationChallenges as $candidate ) { if ( $solver->supports( $candidate ) ) { $authorizationChallenge = $candidate; - EE::debug( 'Authorization challenge supported by solver. Solver: ' . $solverName . ' Challenge: ' . $candidate->getType() ); + \EE::debug( 'Authorization challenge supported by solver. Solver: ' . $solverName . ' Challenge: ' . $candidate->getType() ); break; } // Should not get here as we are handling it. - EE::debug( 'Authorization challenge not supported by solver. Solver: ' . $solverName . ' Challenge: ' . $candidate->getType() ); + \EE::debug( 'Authorization challenge not supported by solver. Solver: ' . $solverName . ' Challenge: ' . $candidate->getType() ); } if ( null === $authorizationChallenge ) { throw new ChallengeNotSupportedException(); } - EE::debug( 'Storing authorization challenge. Domain: ' . $domainKey . ' Challenge: ' . print_r( $authorizationChallenge->toArray(), true ) ); + \EE::debug( 'Storing authorization challenge. Domain: ' . $domainKey . ' Challenge: ' . print_r( $authorizationChallenge->toArray(), true ) ); $this->repository->storeDomainAuthorizationChallenge( $domainKey, $authorizationChallenge ); $authorizationChallengesToSolve[] = $authorizationChallenge; @@ -159,16 +159,16 @@ public function authorize( Array $domains, $site_root, $wildcard = false ) { /** @var AuthorizationChallenge $authorizationChallenge */ foreach ( $authorizationChallengesToSolve as $authorizationChallenge ) { - EE::debug( 'Solving authorization challenge: Domain: ' . $authorizationChallenge->getDomain() . ' Challenge: ' . print_r( $authorizationChallenge->toArray(), true ) ); + \EE::debug( 'Solving authorization challenge: Domain: ' . $authorizationChallenge->getDomain() . ' Challenge: ' . print_r( $authorizationChallenge->toArray(), true ) ); $solver->solve( $authorizationChallenge ); if ( ! $wildcard ) { $token = $authorizationChallenge->toArray()['token']; $payload = $authorizationChallenge->toArray()['payload']; - EE::launch( "mkdir -p $site_root/app/src/.well-known/acme-challenge/" ); - EE::debug( "Creating challange file $site_root/app/src/.well-known/acme-challenge/$token" ); + \EE::launch( "mkdir -p $site_root/app/src/.well-known/acme-challenge/" ); + \EE::debug( "Creating challange file $site_root/app/src/.well-known/acme-challenge/$token" ); file_put_contents( "$site_root/app/src/.well-known/acme-challenge/$token", $payload ); - EE::launch( "chown www-data: $site_root/app/src/.well-known/acme-challenge/$token" ); + \EE::launch( "chown www-data: $site_root/app/src/.well-known/acme-challenge/$token" ); } } @@ -178,7 +178,7 @@ public function authorize( Array $domains, $site_root, $wildcard = false ) { } public function check( Array $domains, $wildcard = false ) { - EE::debug( ('Starting check with solver ') . ($wildcard ? 'dns' : 'http') ); + \EE::debug( ('Starting check with solver ') . ($wildcard ? 'dns' : 'http') ); $solver = $wildcard ? new SimpleDnsSolver( null, new ConsoleOutput() ) : new SimpleHttpSolver(); $validator = new ChainValidator( [ @@ -190,7 +190,7 @@ public function check( Array $domains, $wildcard = false ) { $order = null; if ( $this->repository->hasCertificateOrder( $domains ) ) { $order = $this->repository->loadCertificateOrder( $domains ); - EE::debug( sprintf( 'Loading the authorization token for domains %s ...', implode( ', ', $domains ) ) ); + \EE::debug( sprintf( 'Loading the authorization token for domains %s ...', implode( ', ', $domains ) ) ); } $authorizationChallengeToCleanup = []; @@ -209,31 +209,31 @@ public function check( Array $domains, $wildcard = false ) { } } else { if ( ! $this->repository->hasDomainAuthorizationChallenge( $domain ) ) { - EE::error( "Domain: $domain not yet authorized/has not been started of with EasyEngine letsencrypt site creation." ); + \EE::error( "Domain: $domain not yet authorized/has not been started of with EasyEngine letsencrypt site creation." ); } $authorizationChallenge = $this->repository->loadDomainAuthorizationChallenge( $domain ); if ( ! $solver->supports( $authorizationChallenge ) ) { throw new ChallengeNotSupportedException(); } } - EE::debug( 'Challenge loaded.' ); + \EE::debug( 'Challenge loaded.' ); $authorizationChallenge = $this->client->reloadAuthorization( $authorizationChallenge ); if ( ! $authorizationChallenge->isValid() ) { - EE::debug( sprintf( 'Testing the challenge for domain %s', $domain ) ); + \EE::debug( sprintf( 'Testing the challenge for domain %s', $domain ) ); if ( ! $validator->isValid( $authorizationChallenge ) ) { - EE::warning( sprintf( 'Can not valid challenge for domain %s', $domain ) ); + \EE::warning( sprintf( 'Can not valid challenge for domain %s', $domain ) ); } - EE::debug( sprintf( 'Requesting authorization check for domain %s', $domain ) ); + \EE::debug( sprintf( 'Requesting authorization check for domain %s', $domain ) ); try { $this->client->challengeAuthorization( $authorizationChallenge ); } catch ( Exception $e ) { - EE::debug( $e->getMessage() ); - EE::warning( 'Challange Authorization failed. Check logs and check if your domain is pointed correctly to this server.' ); + \EE::debug( $e->getMessage() ); + \EE::warning( 'Challange Authorization failed. Check logs and check if your domain is pointed correctly to this server.' ); $site_name = isset( $domains[1] ) ? $domains[1] : $domains[0]; - EE::log( "Re-run `ee site le $site_name` after fixing the issue." ); + \EE::log( "Re-run `ee site le $site_name` after fixing the issue." ); return false; } @@ -241,7 +241,7 @@ public function check( Array $domains, $wildcard = false ) { } } - EE::log( 'The authorization check was successful!' ); + \EE::log( 'The authorization check was successful!' ); if ( $solver instanceof MultipleChallengesSolverInterface ) { $solver->cleanupAll( $authorizationChallengeToCleanup ); @@ -260,12 +260,12 @@ public function request( $domain, $altNames = [], $email, $force=false ) { // Certificate renewal if ( $this->hasValidCertificate( $domain, $alternativeNames ) ) { - EE::debug( "Certificate found for $domain, executing renewal" ); + \EE::debug( "Certificate found for $domain, executing renewal" ); return $this->executeRenewal( $domain, $alternativeNames, $force ); } - EE::debug( "No certificate found, executing first request for $domain" ); + \EE::debug( "No certificate found, executing first request for $domain" ); // Certificate first request return $this->executeFirstRequest( $domain, $alternativeNames, $email ); @@ -278,36 +278,36 @@ public function request( $domain, $altNames = [], $email, $force=false ) { * @param array $alternativeNames */ private function executeFirstRequest( $domain, array $alternativeNames, $email ) { - EE::log( 'Executing first request.' ); + \EE::log( 'Executing first request.' ); // Generate domain key pair $keygen = new KeyPairGenerator(); $domainKeyPair = $keygen->generateKeyPair(); $this->repository->storeDomainKeyPair( $domain, $domainKeyPair ); - EE::debug( "$domain Domain key pair generated and stored" ); + \EE::debug( "$domain Domain key pair generated and stored" ); $distinguishedName = $this->getOrCreateDistinguishedName( $domain, $alternativeNames, $email ); // TODO: ask them ;) - EE::debug( 'Distinguished name informations have been stored locally for this domain (they won\'t be asked on renewal).' ); + \EE::debug( 'Distinguished name informations have been stored locally for this domain (they won\'t be asked on renewal).' ); // Order $domains = array_merge( [ $domain ], $alternativeNames ); - EE::debug( sprintf( 'Loading the order related to the domains %s .', implode( ', ', $domains ) ) ); + \EE::debug( sprintf( 'Loading the order related to the domains %s .', implode( ', ', $domains ) ) ); if ( ! $this->repository->hasCertificateOrder( $domains ) ) { - EE::error( "$domain has not yet been authorized." ); + \EE::error( "$domain has not yet been authorized." ); } $order = $this->repository->loadCertificateOrder( $domains ); // Request - EE::log( sprintf( 'Requesting first certificate for domain %s.', $domain ) ); + \EE::log( sprintf( 'Requesting first certificate for domain %s.', $domain ) ); $csr = new CertificateRequest( $distinguishedName, $domainKeyPair ); $response = $this->client->finalizeOrder( $order, $csr ); - EE::log( 'Certificate received' ); + \EE::log( 'Certificate received' ); // Store $this->repository->storeDomainCertificate( $domain, $response->getCertificate() ); - EE::log( 'Certificate stored' ); + \EE::log( 'Certificate stored' ); // Post-generate actions $this->moveCertsToNginxProxy( $domain ); @@ -338,7 +338,7 @@ private function moveCertsToNginxProxy( string $domain ) { private function executeRenewal( $domain, array $alternativeNames, $force = false ) { try { // Check expiration date to avoid too much renewal - EE::log( "Loading current certificate for $domain" ); + \EE::log( "Loading current certificate for $domain" ); $certificate = $this->repository->loadDomainCertificate( $domain ); @@ -348,7 +348,7 @@ private function executeRenewal( $domain, array $alternativeNames, $force = fals if ( $parsedCertificate->getValidTo()->format( 'U' ) - time() >= 604800 ) { - EE::log( + \EE::log( sprintf( 'Current certificate is valid until %s, renewal is not necessary.', $parsedCertificate->getValidTo()->format( 'Y-m-d H:i:s' ) @@ -358,55 +358,55 @@ private function executeRenewal( $domain, array $alternativeNames, $force = fals return; } - EE::log( + \EE::log( sprintf( 'Current certificate will expire in less than a week (%s), renewal is required.', $parsedCertificate->getValidTo()->format( 'Y-m-d H:i:s' ) ) ); } else { - EE::log( 'Forced renewal.' ); + \EE::log( 'Forced renewal.' ); } // Key pair - EE::debug( 'Loading domain key pair...' ); + \EE::debug( 'Loading domain key pair...' ); $domainKeyPair = $this->repository->loadDomainKeyPair( $domain ); // Distinguished name - EE::debug( 'Loading domain distinguished name...' ); + \EE::debug( 'Loading domain distinguished name...' ); $distinguishedName = $this->getOrCreateDistinguishedName( $domain, $alternativeNames ); // Order $domains = array_merge( [ $domain ], $alternativeNames ); - EE::debug( sprintf( 'Loading the order related to the domains %s.', implode( ', ', $domains ) ) ); + \EE::debug( sprintf( 'Loading the order related to the domains %s.', implode( ', ', $domains ) ) ); if ( ! $this->repository->hasCertificateOrder( $domains ) ) { - EE::error( "$domain has not yet been authorized." ); + \EE::error( "$domain has not yet been authorized." ); } $order = $this->repository->loadCertificateOrder( $domains ); // Renewal - EE::log( sprintf( 'Renewing certificate for domain %s.', $domain ) ); + \EE::log( sprintf( 'Renewing certificate for domain %s.', $domain ) ); $csr = new CertificateRequest( $distinguishedName, $domainKeyPair ); $response = $this->client->finalizeOrder( $order, $csr ); - EE::log( 'Certificate received' ); + \EE::log( 'Certificate received' ); $this->repository->storeDomainCertificate( $domain, $response->getCertificate() ); $this->log( 'Certificate stored' ); // Post-generate actions $this->moveCertsToNginxProxy( $domain ); - EE::log( 'Certificate renewed successfully!' ); + \EE::log( 'Certificate renewed successfully!' ); } catch ( \Exception $e ) { - EE::warning( 'A critical error occured during certificate renewal' ); - EE::debug( print_r( $e, true ) ); + \EE::warning( 'A critical error occured during certificate renewal' ); + \EE::debug( print_r( $e, true ) ); throw $e; } catch ( \Throwable $e ) { - EE::warning( 'A critical error occured during certificate renewal' ); - EE::debug( print_r( $e, true ) ); + \EE::warning( 'A critical error occured during certificate renewal' ); + \EE::debug( print_r( $e, true ) ); throw $e; } @@ -524,8 +524,8 @@ public function status() { public function cleanup( $site_root ) { $challange_dir = "$site_root/app/src/.well-known"; if ( file_exists( "$site_root/app/src/.well-known" ) ) { - EE::debug( 'Cleaning up webroot files.' ); - EE\Utils\delete_dir( $challange_dir ); + \EE::debug( 'Cleaning up webroot files.' ); + \EE\Utils\delete_dir( $challange_dir ); } } } From 000c41336b60884ff71156923e29705be99e2f38 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Tue, 28 Aug 2018 23:34:06 +0530 Subject: [PATCH 20/40] Update execeptions according to namespace changes Signed-off-by: Riddhesh Sanghvi --- src/HTML.php | 8 ++++---- src/Site_Letsencrypt.php | 6 +++--- src/class-ee-site.php | 6 +++--- 3 files changed, 10 insertions(+), 10 deletions(-) diff --git a/src/HTML.php b/src/HTML.php index f55dc371..182192bb 100644 --- a/src/HTML.php +++ b/src/HTML.php @@ -196,7 +196,7 @@ private function configure_site_files() { $this->fs->dumpFile( $site_src_dir . '/index.html', $index_html ); \EE::success( 'Configuration files copied.' ); - } catch ( Exception $e ) { + } catch ( \Exception $e ) { $this->catch_clean( $e ); } } @@ -239,7 +239,7 @@ private function create_site() { \EE\Site\Utils\add_site_redirects( $this->site['url'], true, 'inherit' === $this->ssl ); \EE\Site\Utils\reload_proxy_configuration(); } - } catch ( Exception $e ) { + } catch ( \Exception $e ) { $this->catch_clean( $e ); } @@ -270,7 +270,7 @@ private function create_site_db_entry() { } else { throw new Exception( 'Error creating site entry in database.' ); } - } catch ( Exception $e ) { + } catch ( \Exception $e ) { $this->catch_clean( $e ); } } @@ -313,7 +313,7 @@ public function reload( $args, $assoc_args, $whitelisted_containers = [], $reloa /** * Catch and clean exceptions. * - * @param Exception $e + * @param \Exception $e */ private function catch_clean( $e ) { diff --git a/src/Site_Letsencrypt.php b/src/Site_Letsencrypt.php index 56e426d2..af0204ec 100644 --- a/src/Site_Letsencrypt.php +++ b/src/Site_Letsencrypt.php @@ -113,7 +113,7 @@ public function register( $email ) { try { $this->client->registerAccount( null, $email ); } - catch ( Exception $e ) { + catch ( \Exception $e ) { \EE::warning( $e->getMessage() ); \EE::warning( 'It seems you\'re in local environment or there is some issue with network, please check logs. Skipping letsencrypt.' ); @@ -129,7 +129,7 @@ public function authorize( Array $domains, $site_root, $wildcard = false ) { try { $order = $this->client->requestOrder( $domains ); } - catch ( Exception $e ) { + catch ( \Exception $e ) { \EE::warning( $e->getMessage() ); \EE::warning( 'It seems you\'re in local environment or using non-public domain, please check logs. Skipping letsencrypt.' ); @@ -229,7 +229,7 @@ public function check( Array $domains, $wildcard = false ) { try { $this->client->challengeAuthorization( $authorizationChallenge ); } - catch ( Exception $e ) { + catch ( \Exception $e ) { \EE::debug( $e->getMessage() ); \EE::warning( 'Challange Authorization failed. Check logs and check if your domain is pointed correctly to this server.' ); $site_name = isset( $domains[1] ) ? $domains[1] : $domains[0]; diff --git a/src/class-ee-site.php b/src/class-ee-site.php index db3ce914..8f7147aa 100644 --- a/src/class-ee-site.php +++ b/src/class-ee-site.php @@ -155,7 +155,7 @@ protected function delete_site( $level, $site_name, $site_root ) { if ( $this->fs->exists( $site_root ) ) { try { $this->fs->remove( $site_root ); - } catch ( Exception $e ) { + } catch ( \Exception $e ) { \EE::debug( $e ); \EE::error( 'Could not remove site root. Please check if you have sufficient rights.' ); } @@ -167,7 +167,7 @@ protected function delete_site( $level, $site_name, $site_root ) { if ( $this->fs->exists( $config_file_path ) ) { try { $this->fs->remove( $config_file_path ); - } catch ( Exception $e ) { + } catch ( \Exception $e ) { \EE::debug( $e ); \EE::error( 'Could not remove site redirection file. Please check if you have sufficient rights.' ); } @@ -185,7 +185,7 @@ protected function delete_site( $level, $site_name, $site_root ) { $cert_files = [$conf_certs, $conf_var, $crt_file, $key_file]; try { $this->fs->remove( $cert_files ); - } catch ( Exception $e ) { + } catch ( \Exception $e ) { \EE::warning( $e ); } } From c9590a57c9d222a86eef985713dc4d99c8938d8c Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Wed, 29 Aug 2018 12:04:24 +0530 Subject: [PATCH 21/40] Move hooks to helper Signed-off-by: Riddhesh Sanghvi --- site-command.php | 35 ++++------------------------------- src/helper/hooks.php | 35 +++++++++++++++++++++++++++++++++++ 2 files changed, 39 insertions(+), 31 deletions(-) create mode 100644 src/helper/hooks.php diff --git a/site-command.php b/site-command.php index a672fcf8..54ec7df3 100644 --- a/site-command.php +++ b/site-command.php @@ -15,38 +15,11 @@ require_once $autoload; } -// Load utility functions -require_once 'src/site-utils.php'; +// Load utility functions. +require_once 'src/helper/site-utils.php'; -function Before_Help_Command( $args, $assoc_args ) { - - if ( isset( $args[0] ) && 'site' === $args[0] ) { - $site_types = Site_Command::get_site_types(); - if ( isset( $assoc_args['type'] ) ) { - $type = $assoc_args['type']; - } else { - //TODO: get from config. - $type = 'html'; - } - - if ( isset( $site_types[ $type ] ) ) { - $callback = $site_types[ $type ]; - - $command = EE::get_root_command(); - $leaf_command = CommandFactory::create( 'site', $callback, $command ); - $command->add_subcommand( 'site', $leaf_command ); - } else { - $error = sprintf( - "'%s' is not a registered site type of 'ee site --type=%s'. See 'ee help site --type=%s' for available subcommands.", - $type, - $type, - $type - ); - EE::error( $error ); - } - } -} +// Load hooks. +require_once 'src/helper/hooks.php'; EE::add_command( 'site', 'Site_Command' ); -EE::add_hook( 'before_invoke:help', 'Before_Help_Command' ); Site_Command::add_site_type( 'html', 'EE\Site\Type\HTML' ); diff --git a/src/helper/hooks.php b/src/helper/hooks.php new file mode 100644 index 00000000..d0254f96 --- /dev/null +++ b/src/helper/hooks.php @@ -0,0 +1,35 @@ +add_subcommand( 'site', $leaf_command ); + } else { + $error = sprintf( + "'%s' is not a registered site type of 'ee site --type=%s'. See 'ee help site --type=%s' for available subcommands.", + $type, + $type, + $type + ); + EE::error( $error ); + } + } +} + +EE::add_hook( 'before_invoke:help', 'Before_Help_Command' ); From f5d6ccee795ded81f7c2abfd381934d92ef97d8a Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Wed, 29 Aug 2018 12:05:15 +0530 Subject: [PATCH 22/40] Reorganize files to helper and site-type folders Signed-off-by: Riddhesh Sanghvi --- src/{ => helper}/Shutdown_Handler.php | 3 +- src/{ => helper}/Site_Letsencrypt.php | 32 ++++++------ src/{ => helper}/class-ee-site.php | 65 +++++++++++++----------- src/{ => helper}/site-utils.php | 19 +++---- src/{ => site-type}/HTML.php | 15 +++--- src/{ => site-type}/Site_HTML_Docker.php | 1 + 6 files changed, 70 insertions(+), 65 deletions(-) rename src/{ => helper}/Shutdown_Handler.php (74%) rename src/{ => helper}/Site_Letsencrypt.php (97%) rename src/{ => helper}/class-ee-site.php (89%) rename src/{ => helper}/site-utils.php (95%) rename src/{ => site-type}/HTML.php (96%) rename src/{ => site-type}/Site_HTML_Docker.php (99%) diff --git a/src/Shutdown_Handler.php b/src/helper/Shutdown_Handler.php similarity index 74% rename from src/Shutdown_Handler.php rename to src/helper/Shutdown_Handler.php index 7a7bdebf..1d320bcd 100644 --- a/src/Shutdown_Handler.php +++ b/src/helper/Shutdown_Handler.php @@ -7,7 +7,8 @@ class Shutdown_Handler { /** - * Handle fatal errors. This function was created as the register_shutdown_function requires the callable function to be public and any public function inside site-command would be callable directly through command-line. + * Handle fatal errors. This function was created as the register_shutdown_function requires the callable function + * to be public and any public function inside site-command would be callable directly through command-line. * * @param array $site_command having Site_Command object. */ diff --git a/src/Site_Letsencrypt.php b/src/helper/Site_Letsencrypt.php similarity index 97% rename from src/Site_Letsencrypt.php rename to src/helper/Site_Letsencrypt.php index af0204ec..6fd81117 100644 --- a/src/Site_Letsencrypt.php +++ b/src/helper/Site_Letsencrypt.php @@ -1,6 +1,7 @@ client->registerAccount( null, $email ); - } - catch ( \Exception $e ) { + } catch ( \Exception $e ) { \EE::warning( $e->getMessage() ); \EE::warning( 'It seems you\'re in local environment or there is some issue with network, please check logs. Skipping letsencrypt.' ); return false; } \EE::debug( "Account with email id: $email registered successfully!" ); + return true; } @@ -128,8 +129,7 @@ public function authorize( Array $domains, $site_root, $wildcard = false ) { $solverName = $wildcard ? 'dns-01' : 'http-01'; try { $order = $this->client->requestOrder( $domains ); - } - catch ( \Exception $e ) { + } catch ( \Exception $e ) { \EE::warning( $e->getMessage() ); \EE::warning( 'It seems you\'re in local environment or using non-public domain, please check logs. Skipping letsencrypt.' ); @@ -178,7 +178,7 @@ public function authorize( Array $domains, $site_root, $wildcard = false ) { } public function check( Array $domains, $wildcard = false ) { - \EE::debug( ('Starting check with solver ') . ($wildcard ? 'dns' : 'http') ); + \EE::debug( ( 'Starting check with solver ' ) . ( $wildcard ? 'dns' : 'http' ) ); $solver = $wildcard ? new SimpleDnsSolver( null, new ConsoleOutput() ) : new SimpleHttpSolver(); $validator = new ChainValidator( [ @@ -228,8 +228,7 @@ public function check( Array $domains, $wildcard = false ) { \EE::debug( sprintf( 'Requesting authorization check for domain %s', $domain ) ); try { $this->client->challengeAuthorization( $authorizationChallenge ); - } - catch ( \Exception $e ) { + } catch ( \Exception $e ) { \EE::debug( $e->getMessage() ); \EE::warning( 'Challange Authorization failed. Check logs and check if your domain is pointed correctly to this server.' ); $site_name = isset( $domains[1] ) ? $domains[1] : $domains[0]; @@ -251,10 +250,11 @@ public function check( Array $domains, $wildcard = false ) { $solver->cleanup( $authorizationChallenge ); } } + return true; } - public function request( $domain, $altNames = [], $email, $force=false ) { + public function request( $domain, $altNames = [], $email, $force = false ) { $alternativeNames = array_unique( $altNames ); sort( $alternativeNames ); @@ -275,7 +275,7 @@ public function request( $domain, $altNames = [], $email, $force=false ) { * Request a first certificate for the given domain. * * @param string $domain - * @param array $alternativeNames + * @param array $alternativeNames */ private function executeFirstRequest( $domain, array $alternativeNames, $email ) { \EE::log( 'Executing first request.' ); @@ -332,8 +332,8 @@ private function moveCertsToNginxProxy( string $domain ) { * Renew a given domain certificate. * * @param string $domain - * @param array $alternativeNames - * @param bool $force + * @param array $alternativeNames + * @param bool $force */ private function executeRenewal( $domain, array $alternativeNames, $force = false ) { try { @@ -397,14 +397,12 @@ private function executeRenewal( $domain, array $alternativeNames, $force = fals $this->moveCertsToNginxProxy( $domain ); \EE::log( 'Certificate renewed successfully!' ); - } - catch ( \Exception $e ) { + } catch ( \Exception $e ) { \EE::warning( 'A critical error occured during certificate renewal' ); \EE::debug( print_r( $e, true ) ); throw $e; - } - catch ( \Throwable $e ) { + } catch ( \Throwable $e ) { \EE::warning( 'A critical error occured during certificate renewal' ); \EE::debug( print_r( $e, true ) ); @@ -436,7 +434,7 @@ private function hasValidCertificate( $domain, array $alternativeNames ) { * Retrieve the stored distinguishedName or create a new one if needed. * * @param string $domain - * @param array $alternativeNames + * @param array $alternativeNames * * @return DistinguishedName */ @@ -521,7 +519,7 @@ public function status() { $table->render(); } - public function cleanup( $site_root ) { + public function cleanup( $site_root ) { $challange_dir = "$site_root/app/src/.well-known"; if ( file_exists( "$site_root/app/src/.well-known" ) ) { \EE::debug( 'Cleaning up webroot files.' ); diff --git a/src/class-ee-site.php b/src/helper/class-ee-site.php similarity index 89% rename from src/class-ee-site.php rename to src/helper/class-ee-site.php index 8f7147aa..4166c791 100644 --- a/src/class-ee-site.php +++ b/src/helper/class-ee-site.php @@ -1,6 +1,7 @@ site = $site->site_url; + $site->site = $site->site_url; $site->status = $site->site_enabled ? 'enabled' : 'disabled'; return $site; @@ -128,7 +129,7 @@ public function delete( $args, $assoc_args ) { /** * Function to delete the given site. * - * @param int $level Level of deletion. + * @param int $level Level of deletion. * Level - 0: No need of clean-up. * Level - 1: Clean-up only the site-root. * Level - 2: Try to remove network. The network may or may not have been created. @@ -139,7 +140,7 @@ public function delete( $args, $assoc_args ) { */ protected function delete_site( $level, $site_name, $site_root ) { - $this->fs = new Filesystem(); + $this->fs = new Filesystem(); $proxy_type = EE_PROXY_TYPE; if ( $level >= 3 ) { if ( \EE::docker()::docker_compose_down( $site_root ) ) { @@ -177,12 +178,12 @@ protected function delete_site( $level, $site_name, $site_root ) { if ( $level > 4 ) { if ( $this->ssl ) { \EE::log( 'Removing ssl certs.' ); - $crt_file = EE_CONF_ROOT . "/nginx/certs/$site_name.crt"; - $key_file = EE_CONF_ROOT . "/nginx/certs/$site_name.key"; + $crt_file = EE_CONF_ROOT . "/nginx/certs/$site_name.crt"; + $key_file = EE_CONF_ROOT . "/nginx/certs/$site_name.key"; $conf_certs = EE_CONF_ROOT . "/acme-conf/certs/$site_name"; - $conf_var = EE_CONF_ROOT . "/acme-conf/var/$site_name"; + $conf_var = EE_CONF_ROOT . "/acme-conf/var/$site_name"; - $cert_files = [$conf_certs, $conf_var, $crt_file, $key_file]; + $cert_files = [ $conf_certs, $conf_var, $crt_file, $key_file ]; try { $this->fs->remove( $cert_files ); } catch ( \Exception $e ) { @@ -214,9 +215,9 @@ public function up( $args, $assoc_args ) { \EE\Utils\delem_log( 'site enable start' ); $force = \EE\Utils\get_flag_value( $assoc_args, 'force' ); - $args = EE\SiteUtils\auto_site_name( $args, 'site', __FUNCTION__ ); + $args = EE\SiteUtils\auto_site_name( $args, 'site', __FUNCTION__ ); $this->populate_site_info( $args ); - $site = Site::find( $this->site['url'] ); + $site = Site::find( $this->site['url'] ); if ( $site->site_enabled && ! $force ) { \EE::error( sprintf( '%s is already enabled!', $site->site_url ) ); @@ -248,7 +249,7 @@ public function down( $args, $assoc_args ) { $args = EE\SiteUtils\auto_site_name( $args, 'site', __FUNCTION__ ); $this->populate_site_info( $args ); - $site = Site::find($this->site['url']); + $site = Site::find( $this->site['url'] ); \EE::log( sprintf( 'Disabling site %s.', $site->site_url ) ); @@ -279,8 +280,8 @@ public function down( $args, $assoc_args ) { public function restart( $args, $assoc_args, $whitelisted_containers = [] ) { \EE\Utils\delem_log( 'site restart start' ); - $args = EE\SiteUtils\auto_site_name( $args, 'site', __FUNCTION__ ); - $all = \EE\Utils\get_flag_value( $assoc_args, 'all' ); + $args = EE\SiteUtils\auto_site_name( $args, 'site', __FUNCTION__ ); + $all = \EE\Utils\get_flag_value( $assoc_args, 'all' ); $no_service_specified = count( $assoc_args ) === 0; $this->populate_site_info( $args ); @@ -317,8 +318,8 @@ public function reload( $args, $assoc_args, $whitelisted_containers = [], $reloa \EE\Utils\delem_log( 'site reload start' ); $args = EE\SiteUtils\auto_site_name( $args, 'site', __FUNCTION__ ); - $all = \EE\Utils\get_flag_value( $assoc_args, 'all' ); - if ( !array_key_exists( 'nginx', $reload_commands ) ) { + $all = \EE\Utils\get_flag_value( $assoc_args, 'all' ); + if ( ! array_key_exists( 'nginx', $reload_commands ) ) { $reload_commands['nginx'] = 'nginx sh -c \'nginx -t && service openresty reload\''; } $no_service_specified = count( $assoc_args ) === 0; @@ -344,14 +345,14 @@ public function reload( $args, $assoc_args, $whitelisted_containers = [], $reloa private function reload_services( $services, $reload_commands ) { foreach ( $services as $service ) { - EE\SiteUtils\run_compose_command( 'exec', $reload_commands[$service], 'reload', $service ); + EE\SiteUtils\run_compose_command( 'exec', $reload_commands[ $service ], 'reload', $service ); } } /** * Runs the acme le registration and authorization. * - * @param string $site_name Name of the site for ssl. + * @param string $site_name Name of the site for ssl. * * @throws Exception */ @@ -381,7 +382,7 @@ protected function inherit_certs( $site_name ) { * @param string $site_name Name of the site for ssl. * @param string $site_root Webroot of the site. * @param string $ssl_type Type of ssl cert to issue. - * @param bool $wildcard SSL with wildcard or not. + * @param bool $wildcard SSL with wildcard or not. * * @throws \EE\ExitException If --ssl flag has unrecognized value */ @@ -406,16 +407,16 @@ protected function init_ssl( $site_name, $site_root, $ssl_type, $wildcard = fals * * @param string $site_name Name of the site for ssl. * @param string $site_root Webroot of the site. - * @param bool $wildcard SSL with wildcard or not. + * @param bool $wildcard SSL with wildcard or not. */ protected function init_le( $site_name, $site_root, $wildcard = false ) { \EE::debug( "Wildcard in init_le: $wildcard" ); - $this->site['url'] = $site_name; + $this->site['url'] = $site_name; $this->site['root'] = $site_root; - $this->wildcard = $wildcard; - $client = new Site_Letsencrypt(); - $this->le_mail = \EE::get_runner()->config['le-mail'] ?? \EE::input( 'Enter your mail id: ' ); + $this->wildcard = $wildcard; + $client = new Site_Letsencrypt(); + $this->le_mail = \EE::get_runner()->config['le-mail'] ?? \EE::input( 'Enter your mail id: ' ); \EE::get_runner()->ensure_present_in_config( 'le-mail', $this->le_mail ); if ( ! $client->register( $this->le_mail ) ) { $this->ssl = null; @@ -441,11 +442,11 @@ protected function init_le( $site_name, $site_root, $wildcard = false ) { * Returns all domains required by cert * * @param string $site_name Name of site - * @param $wildcard Wildcard cert required? + * @param $wildcard Wildcard cert required? * * @return array */ - private function get_cert_domains( string $site_name, $wildcard ) : array { + private function get_cert_domains( string $site_name, $wildcard ): array { $domains = [ $site_name ]; $has_www = ( strpos( $site_name, 'www.' ) === 0 ); @@ -454,6 +455,7 @@ private function get_cert_domains( string $site_name, $wildcard ) : array { } else { $domains[] = $this->get_www_domain( $site_name ); } + return $domains; } @@ -465,13 +467,13 @@ private function get_cert_domains( string $site_name, $wildcard ) : array { * * @return string Domain name with or without www */ - private function get_www_domain( string $site_name ) : string { + private function get_www_domain( string $site_name ): string { $has_www = ( strpos( $site_name, 'www.' ) === 0 ); if ( $has_www ) { return ltrim( $site_name, 'www.' ); } else { - return 'www.' . $site_name; + return 'www.' . $site_name; } } @@ -489,11 +491,11 @@ private function get_www_domain( string $site_name ) : string { */ public function le( $args = [], $assoc_args = [] ) { - if ( !isset( $this->site['url'] ) ) { + if ( ! isset( $this->site['url'] ) ) { $this->populate_site_info( $args ); } - if ( !isset( $this->le_mail ) ) { + if ( ! isset( $this->le_mail ) ) { $this->le_mail = \EE::get_config( 'le-mail' ) ?? \EE::input( 'Enter your mail id: ' ); } @@ -503,6 +505,7 @@ public function le( $args = [], $assoc_args = [] ) { if ( ! $client->check( $domains, $this->wildcard ) ) { $this->ssl = null; + return; } @@ -521,7 +524,7 @@ public function le( $args = [], $assoc_args = [] ) { private function populate_site_info( $args ) { $this->site['url'] = \EE\Utils\remove_trailing_slash( $args[0] ); - $site = Site::find( $this->site['url'] ); + $site = Site::find( $this->site['url'] ); if ( $site ) { $db_select = $site->site_url; diff --git a/src/site-utils.php b/src/helper/site-utils.php similarity index 95% rename from src/site-utils.php rename to src/helper/site-utils.php index e912cb02..0092c19b 100644 --- a/src/site-utils.php +++ b/src/helper/site-utils.php @@ -15,7 +15,7 @@ function get_site_name() { $sites = Site::all( [ 'site_url' ] ); if ( ! empty( $sites ) ) { - $cwd = getcwd(); + $cwd = getcwd(); $name_in_path = explode( '/', $cwd ); $site_name = array_intersect( array_column( $sites, 'site_url' ), $name_in_path ); @@ -50,10 +50,10 @@ function auto_site_name( $args, $command, $function, $arg_pos = 0 ) { if ( isset( $args[ $arg_pos ] ) ) { $possible_site_name = $args[ $arg_pos ]; - if( substr( $possible_site_name, 0, 4 ) === 'http' ) { - $possible_site_name = str_replace(['https','http'],'',$possible_site_name); + if ( substr( $possible_site_name, 0, 4 ) === 'http' ) { + $possible_site_name = str_replace( [ 'https', 'http' ], '', $possible_site_name ); } - $url_path = parse_url(EE\Utils\remove_trailing_slash($possible_site_name), PHP_URL_PATH); + $url_path = parse_url( EE\Utils\remove_trailing_slash( $possible_site_name ), PHP_URL_PATH ); if ( Site::find( $url_path ) ) { return $args; } @@ -122,7 +122,7 @@ function init_checks() { function generate_global_docker_compose_yml( Filesystem $fs ) { $img_versions = EE\Utils\get_image_versions(); - $data = [ + $data = [ 'services' => [ 'name' => 'nginx-proxy', 'container_name' => EE_PROXY_TYPE, @@ -189,8 +189,8 @@ function reload_proxy_configuration() { * Adds www to non-www redirection to site * * @param string $site_name name of the site. - * @param bool $ssl enable ssl or not. - * @param bool $inherit inherit cert or not. + * @param bool $ssl enable ssl or not. + * @param bool $inherit inherit cert or not. */ function add_site_redirects( string $site_name, bool $ssl, bool $inherit ) { @@ -297,8 +297,9 @@ function get_curl_info( $url, $port = 80, $port_info = false ) { /** * Function to pull the latest images and bring up the site containers. * - * @param string $site_root Root directory of the site. - * @param array $containers The minimum required conatainers to start the site. Default null, leads to starting of all containers. + * @param string $site_root Root directory of the site. + * @param array $containers The minimum required conatainers to start the site. Default null, leads to starting of all + * containers. * * @throws \Exception when docker-compose up fails. */ diff --git a/src/HTML.php b/src/site-type/HTML.php similarity index 96% rename from src/HTML.php rename to src/site-type/HTML.php index 182192bb..57d0438d 100644 --- a/src/HTML.php +++ b/src/site-type/HTML.php @@ -14,6 +14,7 @@ */ namespace EE\Site\Type; + use \EE\Model\Site; use \Symfony\Component\Filesystem\Filesystem; @@ -62,7 +63,7 @@ class HTML extends EE_Site_Command { public function __construct() { - $this->level = 0; + $this->level = 0; pcntl_signal( SIGTERM, [ $this, "rollback" ] ); pcntl_signal( SIGHUP, [ $this, "rollback" ] ); pcntl_signal( SIGUSR1, [ $this, "rollback" ] ); @@ -99,7 +100,7 @@ public function create( $args, $assoc_args ) { \EE::warning( 'This is a beta version. Please don\'t use it in production.' ); $this->logger->debug( 'args:', $args ); $this->logger->debug( 'assoc_args:', empty( $assoc_args ) ? [ 'NULL' ] : $assoc_args ); - $this->site['url'] = strtolower( \EE\Utils\remove_trailing_slash( $args[0] ) ); + $this->site['url'] = strtolower( \EE\Utils\remove_trailing_slash( $args[0] ) ); $this->site['type'] = \EE\Utils\get_flag_value( $assoc_args, 'type', 'html' ); if ( 'html' !== $this->site['type'] ) { \EE::error( sprintf( 'Invalid site-type: %s', $this->site['type'] ) ); @@ -111,7 +112,7 @@ public function create( $args, $assoc_args ) { $this->ssl = \EE\Utils\get_flag_value( $assoc_args, 'ssl' ); $this->ssl_wildcard = \EE\Utils\get_flag_value( $assoc_args, 'wildcard' ); - $this->skip_chk = \EE\Utils\get_flag_value( $assoc_args, 'skip-status-check' ); + $this->skip_chk = \EE\Utils\get_flag_value( $assoc_args, 'skip-status-check' ); \EE\Site\Utils\init_checks(); @@ -143,7 +144,7 @@ public function info( $args, $assoc_args ) { ]; if ( $this->ssl ) { - $info[] = [ 'SSL Wildcard', $this->ssl_wildcard ? 'Yes': 'No' ]; + $info[] = [ 'SSL Wildcard', $this->ssl_wildcard ? 'Yes' : 'No' ]; } \EE\Utils\format_table( $info ); @@ -252,17 +253,17 @@ private function create_site() { */ private function create_site_db_entry() { - $ssl = $this->ssl ? 1 : 0; + $ssl = $this->ssl ? 1 : 0; $ssl_wildcard = $this->ssl_wildcard ? 1 : 0; - $site = Site::create([ + $site = Site::create( [ 'site_url' => $this->site['url'], 'site_type' => $this->site['type'], 'site_fs_path' => $this->site['root'], 'site_ssl' => $ssl, 'site_ssl_wildcard' => $ssl_wildcard, 'created_on' => date( 'Y-m-d H:i:s', time() ), - ]); + ] ); try { if ( $site ) { diff --git a/src/Site_HTML_Docker.php b/src/site-type/Site_HTML_Docker.php similarity index 99% rename from src/Site_HTML_Docker.php rename to src/site-type/Site_HTML_Docker.php index 8ce8dfc2..8f4e5b39 100644 --- a/src/Site_HTML_Docker.php +++ b/src/site-type/Site_HTML_Docker.php @@ -1,6 +1,7 @@ Date: Wed, 29 Aug 2018 14:06:55 +0530 Subject: [PATCH 23/40] Move use statement to hooks file Signed-off-by: Riddhesh Sanghvi --- site-command.php | 2 -- src/helper/hooks.php | 2 ++ 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/site-command.php b/site-command.php index 54ec7df3..cc57adac 100644 --- a/site-command.php +++ b/site-command.php @@ -1,7 +1,5 @@ Date: Wed, 29 Aug 2018 14:07:38 +0530 Subject: [PATCH 24/40] Update doc comment Signed-off-by: Riddhesh Sanghvi --- src/site-type/HTML.php | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/site-type/HTML.php b/src/site-type/HTML.php index 57d0438d..590e8217 100644 --- a/src/site-type/HTML.php +++ b/src/site-type/HTML.php @@ -76,7 +76,7 @@ public function __construct() { } /** - * Runs the standard WordPress site installation. + * Runs the standard HTML site installation. * * ## OPTIONS * @@ -89,7 +89,7 @@ public function __construct() { * [--wildcard] * : Gets wildcard SSL . * [--type=] - * : Type of the site to be created. Values: html,php,wp. + * : Type of the site to be created. Values: html,php,wp etc. * * [--skip-status-check] * : Skips site status check. From ed5c6e094d2b7643f406632130846c0154b03ac4 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Wed, 29 Aug 2018 16:19:41 +0530 Subject: [PATCH 25/40] Update warning message Signed-off-by: Riddhesh Sanghvi --- src/Site_Command.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Site_Command.php b/src/Site_Command.php index 636454d6..eaf906b8 100644 --- a/src/Site_Command.php +++ b/src/Site_Command.php @@ -38,7 +38,7 @@ public static function instance() { public static function add_site_type( $name, $callback ) { if ( isset( self::$instance->site_types[ $name ] ) ) { - EE::warning( sprintf( '%s site-type has already been previously registered by %s. It will be over-written by the new package class %s. Please update your packages to resolve this.', $name, self::$instance->site_types[ $name ], $callback ) ); + EE::warning( sprintf( '%s site-type had already been previously registered by %s. It is overridden by the new package class %s. Please update your packages to resolve this.', $name, self::$instance->site_types[ $name ], $callback ) ); } self::$instance->site_types[ $name ] = $callback; } From eb2e1e42196da1345e022e7bc0efe11b38814512 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Wed, 29 Aug 2018 16:21:42 +0530 Subject: [PATCH 26/40] Update sprintf Signed-off-by: Riddhesh Sanghvi --- src/Site_Command.php | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/src/Site_Command.php b/src/Site_Command.php index eaf906b8..12bf25d1 100644 --- a/src/Site_Command.php +++ b/src/Site_Command.php @@ -73,9 +73,7 @@ public function __invoke( $args, $assoc_args ) { if ( ! isset( $site_types[ $type ] ) ) { $error = sprintf( - "'%s' is not a registered site type of 'ee site --type=%s'. See 'ee help site --type=%s' for available subcommands.", - $type, - $type, + '\'%1$s\' is not a registered site type of \'ee site --type=%1$s\'. See \'ee help site --type=%1$s\' for available subcommands.', $type ); EE::error( $error ); From 69b69764b36272ecbeb8e33bd2e149135bffff56 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Wed, 29 Aug 2018 16:29:09 +0530 Subject: [PATCH 27/40] Remove TODO from code to track in seperate issue Signed-off-by: Riddhesh Sanghvi --- src/Site_Command.php | 4 ---- src/helper/hooks.php | 1 - 2 files changed, 5 deletions(-) diff --git a/src/Site_Command.php b/src/Site_Command.php index 12bf25d1..a6495a67 100644 --- a/src/Site_Command.php +++ b/src/Site_Command.php @@ -104,10 +104,6 @@ private function determine_type( $args ) { // default site-type $type = 'html'; - // TODO: get type from config file as below - // $config_type = EE::get_config('type'); - // $type = empty( $config_type ) ? 'html' : $config_type; - $last_arg = array_pop( $args ); if ( substr( $last_arg, 0, 4 ) === 'http' ) { $last_arg = str_replace( [ 'https://', 'http://' ], '', $last_arg ); diff --git a/src/helper/hooks.php b/src/helper/hooks.php index 16af8399..37609c1b 100644 --- a/src/helper/hooks.php +++ b/src/helper/hooks.php @@ -12,7 +12,6 @@ function Before_Help_Command( $args, $assoc_args ) { if ( isset( $assoc_args['type'] ) ) { $type = $assoc_args['type']; } else { - //TODO: get from config. $type = 'html'; } From 4d9dc92219bceddae042d763dda62047b84572ac Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Wed, 29 Aug 2018 16:31:39 +0530 Subject: [PATCH 28/40] Update hook name and doc comment Signed-off-by: Riddhesh Sanghvi --- src/helper/hooks.php | 9 ++++++--- 1 file changed, 6 insertions(+), 3 deletions(-) diff --git a/src/helper/hooks.php b/src/helper/hooks.php index 37609c1b..8e67645d 100644 --- a/src/helper/hooks.php +++ b/src/helper/hooks.php @@ -3,9 +3,12 @@ use EE\Dispatcher\CommandFactory; /** - * Add hook before the invocation of help command to appropriately handle the help for given site-type. + * Callback function of `before_invoke:help` hook: Add routing for "ee help site" command before the invocation of help command. + * + * @param array $args Commandline arguments passed to help command. + * @param array $assoc_args Associative arguments passed to help command. */ -function Before_Help_Command( $args, $assoc_args ) { +function ee_site_help_cmd_routing( $args, $assoc_args ) { if ( isset( $args[0] ) && 'site' === $args[0] ) { $site_types = Site_Command::get_site_types(); @@ -33,4 +36,4 @@ function Before_Help_Command( $args, $assoc_args ) { } } -EE::add_hook( 'before_invoke:help', 'Before_Help_Command' ); +EE::add_hook( 'before_invoke:help', 'ee_site_help_cmd_routing' ); From 4f66fc202a227d2e7d63ecb660669f55ba53e269 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Wed, 29 Aug 2018 16:35:53 +0530 Subject: [PATCH 29/40] Add early return in help hook Signed-off-by: Riddhesh Sanghvi --- src/helper/hooks.php | 52 ++++++++++++++++++++++++-------------------- 1 file changed, 28 insertions(+), 24 deletions(-) diff --git a/src/helper/hooks.php b/src/helper/hooks.php index 8e67645d..043778ff 100644 --- a/src/helper/hooks.php +++ b/src/helper/hooks.php @@ -3,37 +3,41 @@ use EE\Dispatcher\CommandFactory; /** - * Callback function of `before_invoke:help` hook: Add routing for "ee help site" command before the invocation of help command. + * Callback function of `before_invoke:help` hook: Add routing for "ee help site" command before the invocation of help + * command. * * @param array $args Commandline arguments passed to help command. * @param array $assoc_args Associative arguments passed to help command. */ function ee_site_help_cmd_routing( $args, $assoc_args ) { - if ( isset( $args[0] ) && 'site' === $args[0] ) { - $site_types = Site_Command::get_site_types(); - if ( isset( $assoc_args['type'] ) ) { - $type = $assoc_args['type']; - } else { - $type = 'html'; - } - - if ( isset( $site_types[ $type ] ) ) { - $callback = $site_types[ $type ]; - - $command = EE::get_root_command(); - $leaf_command = CommandFactory::create( 'site', $callback, $command ); - $command->add_subcommand( 'site', $leaf_command ); - } else { - $error = sprintf( - "'%s' is not a registered site type of 'ee site --type=%s'. See 'ee help site --type=%s' for available subcommands.", - $type, - $type, - $type - ); - EE::error( $error ); - } + if ( ( ! isset( $args[0] ) ) || ( 'site' !== $args[0] ) ) { + return; } + + $site_types = Site_Command::get_site_types(); + if ( isset( $assoc_args['type'] ) ) { + $type = $assoc_args['type']; + } else { + $type = 'html'; + } + + if ( isset( $site_types[ $type ] ) ) { + $callback = $site_types[ $type ]; + + $command = EE::get_root_command(); + $leaf_command = CommandFactory::create( 'site', $callback, $command ); + $command->add_subcommand( 'site', $leaf_command ); + } else { + $error = sprintf( + "'%s' is not a registered site type of 'ee site --type=%s'. See 'ee help site --type=%s' for available subcommands.", + $type, + $type, + $type + ); + EE::error( $error ); + } + } EE::add_hook( 'before_invoke:help', 'ee_site_help_cmd_routing' ); From 75e55f7f3754ba3b9e010f0a39383312095d900a Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Wed, 29 Aug 2018 16:37:51 +0530 Subject: [PATCH 30/40] Rename html class file Signed-off-by: Riddhesh Sanghvi --- src/site-type/{HTML.php => html.php} | 0 1 file changed, 0 insertions(+), 0 deletions(-) rename src/site-type/{HTML.php => html.php} (100%) diff --git a/src/site-type/HTML.php b/src/site-type/html.php similarity index 100% rename from src/site-type/HTML.php rename to src/site-type/html.php From 6dd18f838b470aa5bb46bc5beddb87760279dde0 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Wed, 29 Aug 2018 16:46:24 +0530 Subject: [PATCH 31/40] Update help info in doc block Signed-off-by: Riddhesh Sanghvi --- src/Site_Command.php | 1 + 1 file changed, 1 insertion(+) diff --git a/src/Site_Command.php b/src/Site_Command.php index a6495a67..fbd95a9b 100644 --- a/src/Site_Command.php +++ b/src/Site_Command.php @@ -53,6 +53,7 @@ public static function get_site_types() { } /** + * Performs site operations. Check `ee help site` for more info. * Invoked function of site-type routing. Called when `ee site` is invoked. * Performs the routing to respective site-type passed using either `--type=`, * Or discovers the type from the site-name and fetches the type from it, From 18c462fe5fac0ef803f2c5486abfade4dd29b90d Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Wed, 29 Aug 2018 17:03:51 +0530 Subject: [PATCH 32/40] Update path for require Signed-off-by: Riddhesh Sanghvi --- site-command.php | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/site-command.php b/site-command.php index cc57adac..4fcd4336 100644 --- a/site-command.php +++ b/site-command.php @@ -14,10 +14,10 @@ } // Load utility functions. -require_once 'src/helper/site-utils.php'; +require_once __DIR__ . 'src/helper/site-utils.php'; // Load hooks. -require_once 'src/helper/hooks.php'; +require_once __DIR__ . 'src/helper/hooks.php'; EE::add_command( 'site', 'Site_Command' ); Site_Command::add_site_type( 'html', 'EE\Site\Type\HTML' ); From 077316c1afcfee793d07524fcb9c313075dfeb61 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Wed, 29 Aug 2018 17:31:19 +0530 Subject: [PATCH 33/40] Move shutdownfunction and common rollback to abstract class Signed-off-by: Riddhesh Sanghvi --- src/helper/class-ee-site.php | 26 ++++++++++++++++++++++++++ src/site-type/html.php | 30 +++++------------------------- 2 files changed, 31 insertions(+), 25 deletions(-) diff --git a/src/helper/class-ee-site.php b/src/helper/class-ee-site.php index 4166c791..c55c0005 100644 --- a/src/helper/class-ee-site.php +++ b/src/helper/class-ee-site.php @@ -37,6 +37,13 @@ abstract class EE_Site_Command { private $site; public function __construct() { + + pcntl_signal( SIGTERM, [ $this, "rollback" ] ); + pcntl_signal( SIGHUP, [ $this, "rollback" ] ); + pcntl_signal( SIGUSR1, [ $this, "rollback" ] ); + pcntl_signal( SIGINT, [ $this, "rollback" ] ); + $shutdown_handler = new Shutdown_Handler(); + register_shutdown_function( [ $shutdown_handler, "cleanup" ], [ &$this ] ); } /** @@ -538,6 +545,25 @@ private function populate_site_info( $args ) { } } + /** + * Shutdown function to catch and rollback from fatal errors. + */ + protected function shutDownFunction() { + + $logger = \EE::get_file_logger()->withName( 'site-command' ); + $error = error_get_last(); + if ( isset( $error ) && $error['type'] === E_ERROR ) { + \EE::warning( 'An Error occurred. Initiating clean-up.' ); + $logger->error( 'Type: ' . $error['type'] ); + $logger->error( 'Message: ' . $error['message'] ); + $logger->error( 'File: ' . $error['file'] ); + $logger->error( 'Line: ' . $error['line'] ); + $this->rollback(); + } + } + abstract public function create( $args, $assoc_args ); + abstract protected function rollback(); + } diff --git a/src/site-type/html.php b/src/site-type/html.php index 590e8217..396c1c0b 100644 --- a/src/site-type/html.php +++ b/src/site-type/html.php @@ -63,15 +63,10 @@ class HTML extends EE_Site_Command { public function __construct() { - $this->level = 0; - pcntl_signal( SIGTERM, [ $this, "rollback" ] ); - pcntl_signal( SIGHUP, [ $this, "rollback" ] ); - pcntl_signal( SIGUSR1, [ $this, "rollback" ] ); - pcntl_signal( SIGINT, [ $this, "rollback" ] ); - $shutdown_handler = new Shutdown_Handler(); - register_shutdown_function( [ $shutdown_handler, "cleanup" ], [ &$this ] ); + parent::__construct(); + $this->level = 0; $this->docker = \EE::docker(); - $this->logger = \EE::get_file_logger()->withName( 'site_command' ); + $this->logger = \EE::get_file_logger()->withName( 'html_type' ); $this->fs = new Filesystem(); } @@ -95,7 +90,7 @@ public function __construct() { * : Skips site status check. */ public function create( $args, $assoc_args ) { - + trigger_error("Cannot divide by zero", E_USER_ERROR); \EE\Utils\delem_log( 'site create start' ); \EE::warning( 'This is a beta version. Please don\'t use it in production.' ); $this->logger->debug( 'args:', $args ); @@ -329,7 +324,7 @@ private function catch_clean( $e ) { /** * Roll back on interrupt. */ - private function rollback() { + protected function rollback() { \EE::warning( 'Exiting gracefully after rolling back. This may take some time.' ); if ( $this->level > 0 ) { @@ -339,19 +334,4 @@ private function rollback() { exit; } - /** - * Shutdown function to catch and rollback from fatal errors. - */ - private function shutDownFunction() { - - $error = error_get_last(); - if ( isset( $error ) && $error['type'] === E_ERROR ) { - \EE::warning( 'An Error occurred. Initiating clean-up.' ); - $this->logger->error( 'Type: ' . $error['type'] ); - $this->logger->error( 'Message: ' . $error['message'] ); - $this->logger->error( 'File: ' . $error['file'] ); - $this->logger->error( 'Line: ' . $error['line'] ); - $this->rollback(); - } - } } From b5c167d85543141931c2841259f5fdd322fd5648 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Wed, 29 Aug 2018 17:33:22 +0530 Subject: [PATCH 34/40] Move file loading to composer Signed-off-by: Riddhesh Sanghvi --- composer.json | 6 +++++- site-command.php | 6 ------ 2 files changed, 5 insertions(+), 7 deletions(-) diff --git a/composer.json b/composer.json index 2303c5bc..90c612bd 100644 --- a/composer.json +++ b/composer.json @@ -12,7 +12,11 @@ "": "src/", "AcmePhp\\Cli\\": "AcmePhp/Cli" }, - "files": [ "site-command.php" ] + "files": [ + "site-command.php", + "src/helper/site-utils.php", + "src/helper/hooks.php" + ] }, "extra": { "branch-alias": { diff --git a/site-command.php b/site-command.php index 4fcd4336..5b78c171 100644 --- a/site-command.php +++ b/site-command.php @@ -13,11 +13,5 @@ require_once $autoload; } -// Load utility functions. -require_once __DIR__ . 'src/helper/site-utils.php'; - -// Load hooks. -require_once __DIR__ . 'src/helper/hooks.php'; - EE::add_command( 'site', 'Site_Command' ); Site_Command::add_site_type( 'html', 'EE\Site\Type\HTML' ); From e408a92b1f277df1593aa41ea64814b9a4ac372a Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Wed, 29 Aug 2018 17:36:57 +0530 Subject: [PATCH 35/40] Update loading sequence Signed-off-by: Riddhesh Sanghvi --- composer.json | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/composer.json b/composer.json index 90c612bd..81133af2 100644 --- a/composer.json +++ b/composer.json @@ -13,9 +13,9 @@ "AcmePhp\\Cli\\": "AcmePhp/Cli" }, "files": [ - "site-command.php", + "src/helper/hooks.php", "src/helper/site-utils.php", - "src/helper/hooks.php" + "site-command.php" ] }, "extra": { From 72444d874c13c2b7516de65e55f5f88539d4adc0 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Wed, 29 Aug 2018 17:45:36 +0530 Subject: [PATCH 36/40] Change hook addition Signed-off-by: Riddhesh Sanghvi --- site-command.php | 1 + src/helper/hooks.php | 2 -- 2 files changed, 1 insertion(+), 2 deletions(-) diff --git a/site-command.php b/site-command.php index 5b78c171..b0127971 100644 --- a/site-command.php +++ b/site-command.php @@ -14,4 +14,5 @@ } EE::add_command( 'site', 'Site_Command' ); +EE::add_hook( 'before_invoke:help', 'ee_site_help_cmd_routing' ); Site_Command::add_site_type( 'html', 'EE\Site\Type\HTML' ); diff --git a/src/helper/hooks.php b/src/helper/hooks.php index 043778ff..f2621481 100644 --- a/src/helper/hooks.php +++ b/src/helper/hooks.php @@ -39,5 +39,3 @@ function ee_site_help_cmd_routing( $args, $assoc_args ) { } } - -EE::add_hook( 'before_invoke:help', 'ee_site_help_cmd_routing' ); From eb15ce54c1869eeb905ab1d05a8efe494d045c64 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Wed, 29 Aug 2018 18:00:20 +0530 Subject: [PATCH 37/40] Remove test trigger of error Signed-off-by: Riddhesh Sanghvi --- src/site-type/html.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/site-type/html.php b/src/site-type/html.php index 396c1c0b..5c279b5a 100644 --- a/src/site-type/html.php +++ b/src/site-type/html.php @@ -90,7 +90,7 @@ public function __construct() { * : Skips site status check. */ public function create( $args, $assoc_args ) { - trigger_error("Cannot divide by zero", E_USER_ERROR); + \EE\Utils\delem_log( 'site create start' ); \EE::warning( 'This is a beta version. Please don\'t use it in production.' ); $this->logger->debug( 'args:', $args ); From a313af67a770f41139b202b882e49a090d5251c7 Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Wed, 29 Aug 2018 22:41:36 +0530 Subject: [PATCH 38/40] Update hooks file to add hooks Signed-off-by: Riddhesh Sanghvi --- site-command.php | 1 - src/helper/hooks.php | 6 ++++++ 2 files changed, 6 insertions(+), 1 deletion(-) diff --git a/site-command.php b/site-command.php index b0127971..5b78c171 100644 --- a/site-command.php +++ b/site-command.php @@ -14,5 +14,4 @@ } EE::add_command( 'site', 'Site_Command' ); -EE::add_hook( 'before_invoke:help', 'ee_site_help_cmd_routing' ); Site_Command::add_site_type( 'html', 'EE\Site\Type\HTML' ); diff --git a/src/helper/hooks.php b/src/helper/hooks.php index f2621481..06d339f2 100644 --- a/src/helper/hooks.php +++ b/src/helper/hooks.php @@ -1,5 +1,9 @@ Date: Thu, 30 Aug 2018 13:44:59 +0530 Subject: [PATCH 39/40] Update sprintf usage Signed-off-by: Riddhesh Sanghvi --- src/helper/hooks.php | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/src/helper/hooks.php b/src/helper/hooks.php index 06d339f2..256709bc 100644 --- a/src/helper/hooks.php +++ b/src/helper/hooks.php @@ -34,9 +34,7 @@ function ee_site_help_cmd_routing( $args, $assoc_args ) { $command->add_subcommand( 'site', $leaf_command ); } else { $error = sprintf( - "'%s' is not a registered site type of 'ee site --type=%s'. See 'ee help site --type=%s' for available subcommands.", - $type, - $type, + '\'%1$s\' is not a registered site type of \'ee site --type=%1$s\'. See \'ee help site --type=%1$s\' for available subcommands.', $type ); EE::error( $error ); From cd38bc779f2fbc094ca33b7c0421eb0e15817a5c Mon Sep 17 00:00:00 2001 From: Riddhesh Sanghvi Date: Thu, 30 Aug 2018 13:46:39 +0530 Subject: [PATCH 40/40] Update function name Signed-off-by: Riddhesh Sanghvi --- src/helper/Shutdown_Handler.php | 2 +- src/helper/class-ee-site.php | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/helper/Shutdown_Handler.php b/src/helper/Shutdown_Handler.php index 1d320bcd..db032575 100644 --- a/src/helper/Shutdown_Handler.php +++ b/src/helper/Shutdown_Handler.php @@ -14,7 +14,7 @@ class Shutdown_Handler { */ public function cleanup( $site_command ) { $reflector = new \ReflectionObject( $site_command[0] ); - $method = $reflector->getMethod( 'shutDownFunction' ); + $method = $reflector->getMethod( 'shut_down_function' ); $method->setAccessible( true ); $method->invoke( $site_command[0] ); } diff --git a/src/helper/class-ee-site.php b/src/helper/class-ee-site.php index c55c0005..8e15961c 100644 --- a/src/helper/class-ee-site.php +++ b/src/helper/class-ee-site.php @@ -548,7 +548,7 @@ private function populate_site_info( $args ) { /** * Shutdown function to catch and rollback from fatal errors. */ - protected function shutDownFunction() { + protected function shut_down_function() { $logger = \EE::get_file_logger()->withName( 'site-command' ); $error = error_get_last();