2023-08-16 12:21:15 -04:00
|
|
|
package node
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto"
|
|
|
|
"crypto/ecdsa"
|
|
|
|
"crypto/elliptic"
|
|
|
|
"crypto/rand"
|
|
|
|
"crypto/x509"
|
|
|
|
"encoding/pem"
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"path"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/go-acme/lego/v4/certificate"
|
|
|
|
"github.com/go-acme/lego/v4/challenge/http01"
|
|
|
|
"github.com/go-acme/lego/v4/providers/dns"
|
|
|
|
"github.com/go-acme/lego/v4/registration"
|
|
|
|
"github.com/goccy/go-json"
|
|
|
|
|
|
|
|
"github.com/InazumaV/V2bX/common/file"
|
|
|
|
"github.com/InazumaV/V2bX/conf"
|
|
|
|
"github.com/go-acme/lego/v4/certcrypto"
|
|
|
|
"github.com/go-acme/lego/v4/lego"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Lego struct {
|
|
|
|
client *lego.Client
|
|
|
|
config *conf.CertConfig
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewLego(config *conf.CertConfig) (*Lego, error) {
|
|
|
|
user, err := NewLegoUser(path.Join(path.Dir(config.CertFile),
|
|
|
|
"user",
|
|
|
|
fmt.Sprintf("user-%s.json", config.Email)),
|
|
|
|
config.Email)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("create user error: %s", err)
|
|
|
|
}
|
|
|
|
c := lego.NewConfig(user)
|
|
|
|
//c.CADirURL = "http://192.168.99.100:4000/directory"
|
|
|
|
c.Certificate.KeyType = certcrypto.RSA2048
|
|
|
|
client, err := lego.NewClient(c)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
l := Lego{
|
|
|
|
client: client,
|
|
|
|
config: config,
|
|
|
|
}
|
|
|
|
err = l.SetProvider()
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("set provider error: %s", err)
|
|
|
|
}
|
|
|
|
return &l, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func checkPath(p string) error {
|
|
|
|
if !file.IsExist(path.Dir(p)) {
|
|
|
|
err := os.MkdirAll(path.Dir(p), 0755)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("create dir error: %s", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *Lego) SetProvider() error {
|
|
|
|
switch l.config.CertMode {
|
|
|
|
case "http":
|
|
|
|
err := l.client.Challenge.SetHTTP01Provider(http01.NewProviderServer("", "80"))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
case "dns":
|
|
|
|
for k, v := range l.config.DNSEnv {
|
|
|
|
os.Setenv(k, v)
|
|
|
|
}
|
|
|
|
p, err := dns.NewDNSChallengeProviderByName(l.config.Provider)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("create dns challenge provider error: %s", err)
|
|
|
|
}
|
|
|
|
err = l.client.Challenge.SetDNS01Provider(p)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("set dns provider error: %s", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *Lego) CreateCert() (err error) {
|
|
|
|
request := certificate.ObtainRequest{
|
|
|
|
Domains: []string{l.config.CertDomain},
|
2023-09-06 11:28:38 -04:00
|
|
|
Bundle: true,
|
2023-08-16 12:21:15 -04:00
|
|
|
}
|
|
|
|
certificates, err := l.client.Certificate.Obtain(request)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("obtain certificate error: %s", err)
|
|
|
|
}
|
|
|
|
err = l.writeCert(certificates)
|
2023-11-17 17:05:28 -05:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("write certificate error: %s", err)
|
|
|
|
}
|
2023-08-16 12:21:15 -04:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *Lego) RenewCert() error {
|
|
|
|
file, err := os.ReadFile(l.config.CertFile)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("read cert file error: %s", err)
|
|
|
|
}
|
|
|
|
if e, err := l.CheckCert(file); !e {
|
|
|
|
return nil
|
|
|
|
} else if err != nil {
|
|
|
|
return fmt.Errorf("check cert error: %s", err)
|
|
|
|
}
|
|
|
|
res, err := l.client.Certificate.Renew(certificate.Resource{
|
|
|
|
Domain: l.config.CertDomain,
|
|
|
|
Certificate: file,
|
2023-09-06 11:28:38 -04:00
|
|
|
}, true, false, "")
|
2023-08-16 12:21:15 -04:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
err = l.writeCert(res)
|
2023-11-17 17:05:28 -05:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("write certificate error: %s", err)
|
|
|
|
}
|
2023-08-16 12:21:15 -04:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *Lego) CheckCert(file []byte) (bool, error) {
|
|
|
|
cert, err := certcrypto.ParsePEMCertificate(file)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
notAfter := int(time.Until(cert.NotAfter).Hours() / 24.0)
|
|
|
|
if notAfter > 30 {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
func (l *Lego) parseParams(path string) string {
|
|
|
|
r := strings.NewReplacer("{domain}", l.config.CertDomain,
|
|
|
|
"{email}", l.config.Email)
|
|
|
|
return r.Replace(path)
|
|
|
|
}
|
|
|
|
func (l *Lego) writeCert(certificates *certificate.Resource) error {
|
|
|
|
err := checkPath(l.config.CertFile)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("check path error: %s", err)
|
|
|
|
}
|
|
|
|
err = os.WriteFile(l.parseParams(l.config.CertFile), certificates.Certificate, 0644)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
err = checkPath(l.config.KeyFile)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("check path error: %s", err)
|
|
|
|
}
|
|
|
|
err = os.WriteFile(l.parseParams(l.config.KeyFile), certificates.PrivateKey, 0644)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type User struct {
|
|
|
|
Email string `json:"Email"`
|
|
|
|
Registration *registration.Resource `json:"Registration"`
|
|
|
|
key crypto.PrivateKey
|
|
|
|
KeyEncoded string `json:"Key"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func (u *User) GetEmail() string {
|
|
|
|
return u.Email
|
|
|
|
}
|
|
|
|
func (u *User) GetRegistration() *registration.Resource {
|
|
|
|
return u.Registration
|
|
|
|
}
|
|
|
|
func (u *User) GetPrivateKey() crypto.PrivateKey {
|
|
|
|
return u.key
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewLegoUser(path string, email string) (*User, error) {
|
|
|
|
var user User
|
|
|
|
if file.IsExist(path) {
|
|
|
|
err := user.Load(path)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if user.Email != email {
|
|
|
|
user.Registration = nil
|
|
|
|
user.Email = email
|
|
|
|
err := registerUser(&user, path)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
user.Email = email
|
|
|
|
err := registerUser(&user, path)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return &user, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func registerUser(user *User, path string) error {
|
|
|
|
privateKey, err := ecdsa.GenerateKey(elliptic.P256(), rand.Reader)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("generate key error: %s", err)
|
|
|
|
}
|
|
|
|
user.key = privateKey
|
|
|
|
c := lego.NewConfig(user)
|
|
|
|
client, err := lego.NewClient(c)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("create lego client error: %s", err)
|
|
|
|
}
|
|
|
|
reg, err := client.Registration.Register(registration.RegisterOptions{TermsOfServiceAgreed: true})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
user.Registration = reg
|
|
|
|
err = user.Save(path)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("save user error: %s", err)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func EncodePrivate(privKey *ecdsa.PrivateKey) (string, error) {
|
|
|
|
encoded, err := x509.MarshalECPrivateKey(privKey)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
pemEncoded := pem.EncodeToMemory(&pem.Block{Type: "EC PRIVATE KEY", Bytes: encoded})
|
|
|
|
return string(pemEncoded), nil
|
|
|
|
}
|
|
|
|
func (u *User) Save(path string) error {
|
|
|
|
err := checkPath(path)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("check path error: %s", err)
|
|
|
|
}
|
|
|
|
u.KeyEncoded, _ = EncodePrivate(u.key.(*ecdsa.PrivateKey))
|
|
|
|
f, err := os.OpenFile(path, os.O_CREATE|os.O_TRUNC|os.O_WRONLY, 0644)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
err = json.NewEncoder(f).Encode(u)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("marshal json error: %s", err)
|
|
|
|
}
|
|
|
|
u.KeyEncoded = ""
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (u *User) DecodePrivate(pemEncodedPriv string) (*ecdsa.PrivateKey, error) {
|
|
|
|
blockPriv, _ := pem.Decode([]byte(pemEncodedPriv))
|
|
|
|
x509EncodedPriv := blockPriv.Bytes
|
|
|
|
privateKey, err := x509.ParseECPrivateKey(x509EncodedPriv)
|
|
|
|
return privateKey, err
|
|
|
|
}
|
2024-01-25 09:06:37 -05:00
|
|
|
|
2023-08-16 12:21:15 -04:00
|
|
|
func (u *User) Load(path string) error {
|
2024-01-25 09:06:37 -05:00
|
|
|
data, err := os.ReadFile(path)
|
2023-08-16 12:21:15 -04:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("open file error: %s", err)
|
|
|
|
}
|
2024-01-25 09:06:37 -05:00
|
|
|
|
|
|
|
err = json.Unmarshal(data, u)
|
2023-08-16 12:21:15 -04:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("unmarshal json error: %s", err)
|
|
|
|
}
|
|
|
|
u.key, err = u.DecodePrivate(u.KeyEncoded)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("decode private key error: %s", err)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|