diff --git a/src/mod/acme/acme.go b/src/mod/acme/acme.go index 1178168..d82cf05 100644 --- a/src/mod/acme/acme.go +++ b/src/mod/acme/acme.go @@ -5,6 +5,7 @@ import ( "crypto/ecdsa" "crypto/elliptic" "crypto/rand" + "crypto/tls" "crypto/x509" "encoding/json" "encoding/pem" @@ -30,6 +31,7 @@ import ( type CertificateInfoJSON struct { AcmeName string `json:"acme_name"` AcmeUrl string `json:"acme_url"` + SkipTLS bool `json:"skip_tls"` } // ACMEUser represents a user in the ACME system. @@ -69,7 +71,7 @@ func NewACME(acmeServer string, port string) *ACMEHandler { } // ObtainCert obtains a certificate for the specified domains. -func (a *ACMEHandler) ObtainCert(domains []string, certificateName string, email string, caName string, caUrl string) (bool, error) { +func (a *ACMEHandler) ObtainCert(domains []string, certificateName string, email string, caName string, caUrl string, skipTLS bool) (bool, error) { log.Println("[ACME] Obtaining certificate...") // generate private key @@ -88,6 +90,24 @@ func (a *ACMEHandler) ObtainCert(domains []string, certificateName string, email // create config config := lego.NewConfig(&adminUser) + // skip TLS verify if need + // Ref: https://github.com/go-acme/lego/blob/6af2c756ac73a9cb401621afca722d0f4112b1b8/lego/client_config.go#L74 + if skipTLS { + log.Println("[INFO] Ignore TLS/SSL Verification Error for ACME Server") + config.HTTPClient.Transport = &http.Transport{ + Proxy: http.ProxyFromEnvironment, + DialContext: (&net.Dialer{ + Timeout: 30 * time.Second, + KeepAlive: 30 * time.Second, + }).DialContext, + TLSHandshakeTimeout: 30 * time.Second, + ResponseHeaderTimeout: 30 * time.Second, + TLSClientConfig: &tls.Config{ + InsecureSkipVerify: true, + }, + } + } + // setup the custom ACME url endpoint. if caUrl != "" { config.CADirURL = caUrl @@ -159,6 +179,7 @@ func (a *ACMEHandler) ObtainCert(domains []string, certificateName string, email certInfo := &CertificateInfoJSON{ AcmeName: caName, AcmeUrl: caUrl, + SkipTLS: skipTLS, } certInfoBytes, err := json.Marshal(certInfo) @@ -287,15 +308,25 @@ func (a *ACMEHandler) HandleRenewCertificate(w http.ResponseWriter, r *http.Requ } if ca == "custom" { - caUrl, err = utils.PostPara(r, "ca_url") + caUrl, err = utils.PostPara(r, "caURL") if err != nil { log.Println("Custom CA set but no URL provide, Using default") ca, caUrl = "", "" } } + var skipTLS bool + + if skipTLSString, err := utils.PostPara(r, "skipTLS"); err != nil { + skipTLS = false + } else if skipTLSString != "true" { + skipTLS = false + } else { + skipTLS = true + } + domains := strings.Split(domainPara, ",") - result, err := a.ObtainCert(domains, filename, email, ca, caUrl) + result, err := a.ObtainCert(domains, filename, email, ca, caUrl, skipTLS) if err != nil { utils.SendErrorResponse(w, jsonEscape(err.Error())) return diff --git a/src/mod/acme/autorenew.go b/src/mod/acme/autorenew.go index 6ed7433..13872fd 100644 --- a/src/mod/acme/autorenew.go +++ b/src/mod/acme/autorenew.go @@ -365,7 +365,7 @@ func (a *AutoRenewer) renewExpiredDomains(certs []*ExpiredCerts) ([]string, erro certInfo = &CertificateInfoJSON{} } - _, err = a.AcmeHandler.ObtainCert(expiredCert.Domains, certName, a.RenewerConfig.Email, certInfo.AcmeName, certInfo.AcmeUrl) + _, err = a.AcmeHandler.ObtainCert(expiredCert.Domains, certName, a.RenewerConfig.Email, certInfo.AcmeName, certInfo.AcmeUrl, certInfo.SkipTLS) if err != nil { log.Println("Renew " + fileName + "(" + strings.Join(expiredCert.Domains, ",") + ") failed: " + err.Error()) } else { diff --git a/src/web/snippet/acme.html b/src/web/snippet/acme.html index 24e1d8c..554fb24 100644 --- a/src/web/snippet/acme.html +++ b/src/web/snippet/acme.html @@ -114,9 +114,15 @@

Generate New Certificate

- @@ -302,9 +308,11 @@

Generate New Certificate

$("input[name=ca]").on('change', function() { if(this.value == "Custom ACME Server") { - $("#customca").show(); + $("#caInput").show(); + $("#skipTLS").show(); } else { - $("#customca").hide(); + $("#caInput").hide(); + $("#skipTLS").hide(); } }) @@ -331,12 +339,14 @@

Generate New Certificate

} var ca = $("#ca").dropdown("get value"); - var ca_url = ""; + var caURL = ""; if (ca == "Custom ACME Server") { ca = "custom"; - ca_url = $("#caurl").val(); + caURL = $("#caURL").val(); } + var skipTLSValue = $("#skipTLSCheckbox")[0].checked; + $.ajax({ url: "/api/acme/obtainCert", method: "GET", @@ -345,7 +355,8 @@

Generate New Certificate

filename: filename, email: email, ca: ca, - ca_url: ca_url, + caURL: caURL, + skipTLS: skipTLSValue, }, success: function(response) { $("#obtainButton").removeClass("loading").removeClass("disabled");