"""ApacheDVSNI""" import os from letsencrypt.plugins import common from letsencrypt_apache import obj from letsencrypt_apache import parser class ApacheDvsni(common.Dvsni): """Class performs DVSNI challenges within the Apache configurator. :ivar configurator: ApacheConfigurator object :type configurator: :class:`~apache.configurator.ApacheConfigurator` :ivar list achalls: Annotated :class:`~letsencrypt.achallenges.DVSNI` challenges. :param list indices: Meant to hold indices of challenges in a larger array. ApacheDvsni is capable of solving many challenges at once which causes an indexing issue within ApacheConfigurator who must return all responses in order. Imagine ApacheConfigurator maintaining state about where all of the http-01 Challenges, Dvsni Challenges belong in the response array. This is an optional utility. :param str challenge_conf: location of the challenge config file """ VHOST_TEMPLATE = """\ ServerName {server_name} UseCanonicalName on SSLStrictSNIVHostCheck on LimitRequestBody 1048576 Include {ssl_options_conf_path} SSLCertificateFile {cert_path} SSLCertificateKeyFile {key_path} DocumentRoot {document_root} """ def __init__(self, *args, **kwargs): super(ApacheDvsni, self).__init__(*args, **kwargs) self.challenge_conf = os.path.join( self.configurator.conf("server-root"), "le_dvsni_cert_challenge.conf") def perform(self): """Perform a DVSNI challenge.""" if not self.achalls: return [] # Save any changes to the configuration as a precaution # About to make temporary changes to the config self.configurator.save() # Prepare the server for HTTPS self.configurator.prepare_server_https( str(self.configurator.config.dvsni_port), True) responses = [] # Create all of the challenge certs for achall in self.achalls: responses.append(self._setup_challenge_cert(achall)) # Setup the configuration dvsni_addrs = self._mod_config() self.configurator.make_addrs_sni_ready(dvsni_addrs) # Save reversible changes self.configurator.save("SNI Challenge", True) return responses def _mod_config(self): """Modifies Apache config files to include challenge vhosts. Result: Apache config includes virtual servers for issued challs :returns: All DVSNI addresses used :rtype: set """ dvsni_addrs = set() config_text = "\n" for achall in self.achalls: achall_addrs = self.get_dvsni_addrs(achall) dvsni_addrs.update(achall_addrs) config_text += self._get_config_text(achall, achall_addrs) config_text += "\n" self._conf_include_check(self.configurator.parser.loc["default"]) self.configurator.reverter.register_file_creation( True, self.challenge_conf) with open(self.challenge_conf, "w") as new_conf: new_conf.write(config_text) return dvsni_addrs def get_dvsni_addrs(self, achall): """Return the Apache addresses needed for DVSNI.""" vhost = self.configurator.choose_vhost(achall.domain) # TODO: Checkout _default_ rules. dvsni_addrs = set() default_addr = obj.Addr(("*", str(self.configurator.config.dvsni_port))) for addr in vhost.addrs: if "_default_" == addr.get_addr(): dvsni_addrs.add(default_addr) else: dvsni_addrs.add( addr.get_sni_addr(self.configurator.config.dvsni_port)) return dvsni_addrs def _conf_include_check(self, main_config): """Adds DVSNI challenge conf file into configuration. Adds DVSNI challenge include file if it does not already exist within mainConfig :param str main_config: file path to main user apache config file """ if len(self.configurator.parser.find_dir( parser.case_i("Include"), self.challenge_conf)) == 0: # print "Including challenge virtual host(s)" self.configurator.parser.add_dir( parser.get_aug_path(main_config), "Include", self.challenge_conf) def _get_config_text(self, achall, ip_addrs): """Chocolate virtual server configuration text :param achall: Annotated DVSNI challenge. :type achall: :class:`letsencrypt.achallenges.DVSNI` :param list ip_addrs: addresses of challenged domain :class:`list` of type `~.obj.Addr` :returns: virtual host configuration text :rtype: str """ ips = " ".join(str(i) for i in ip_addrs) document_root = os.path.join( self.configurator.config.work_dir, "dvsni_page/") # TODO: Python docs is not clear how mutliline string literal # newlines are parsed on different platforms. At least on # Linux (Debian sid), when source file uses CRLF, Python still # parses it as "\n"... c.f.: # https://docs.python.org/2.7/reference/lexical_analysis.html return self.VHOST_TEMPLATE.format( vhost=ips, server_name=achall.gen_response(achall.account_key).z_domain, ssl_options_conf_path=self.configurator.mod_ssl_conf, cert_path=self.get_cert_path(achall), key_path=self.get_key_path(achall), document_root=document_root).replace("\n", os.linesep)