mirror of
https://github.com/songquanpeng/one-api.git
synced 2026-02-19 12:24:25 +08:00
Compare commits
4 Commits
v0.6.8-alp
...
43368e68c8
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
43368e68c8 | ||
|
|
c4fe57c165 | ||
|
|
274fcf3d76 | ||
|
|
3462bd538c |
@@ -63,6 +63,7 @@ var SMTPPort = 587
|
|||||||
var SMTPAccount = ""
|
var SMTPAccount = ""
|
||||||
var SMTPFrom = ""
|
var SMTPFrom = ""
|
||||||
var SMTPToken = ""
|
var SMTPToken = ""
|
||||||
|
var SMTPAuthLoginEnabled = false
|
||||||
|
|
||||||
var GitHubClientId = ""
|
var GitHubClientId = ""
|
||||||
var GitHubClientSecret = ""
|
var GitHubClientSecret = ""
|
||||||
@@ -145,6 +146,9 @@ var InitialRootToken = os.Getenv("INITIAL_ROOT_TOKEN")
|
|||||||
|
|
||||||
var GeminiVersion = env.String("GEMINI_VERSION", "v1")
|
var GeminiVersion = env.String("GEMINI_VERSION", "v1")
|
||||||
|
|
||||||
|
|
||||||
|
var OnlyOneLogFile = env.Bool("ONLY_ONE_LOG_FILE", false)
|
||||||
|
|
||||||
var RelayProxy = env.String("RELAY_PROXY", "")
|
var RelayProxy = env.String("RELAY_PROXY", "")
|
||||||
var UserContentRequestProxy = env.String("USER_CONTENT_REQUEST_PROXY", "")
|
var UserContentRequestProxy = env.String("USER_CONTENT_REQUEST_PROXY", "")
|
||||||
var UserContentRequestTimeout = env.Int("USER_CONTENT_REQUEST_TIMEOUT", 30)
|
var UserContentRequestTimeout = env.Int("USER_CONTENT_REQUEST_TIMEOUT", 30)
|
||||||
|
|||||||
@@ -27,7 +27,12 @@ var setupLogOnce sync.Once
|
|||||||
func SetupLogger() {
|
func SetupLogger() {
|
||||||
setupLogOnce.Do(func() {
|
setupLogOnce.Do(func() {
|
||||||
if LogDir != "" {
|
if LogDir != "" {
|
||||||
logPath := filepath.Join(LogDir, fmt.Sprintf("oneapi-%s.log", time.Now().Format("20060102")))
|
var logPath string
|
||||||
|
if config.OnlyOneLogFile {
|
||||||
|
logPath = filepath.Join(LogDir, "oneapi.log")
|
||||||
|
} else {
|
||||||
|
logPath = filepath.Join(LogDir, fmt.Sprintf("oneapi-%s.log", time.Now().Format("20060102")))
|
||||||
|
}
|
||||||
fd, err := os.OpenFile(logPath, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644)
|
fd, err := os.OpenFile(logPath, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal("failed to open log file")
|
log.Fatal("failed to open log file")
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import (
|
|||||||
"crypto/rand"
|
"crypto/rand"
|
||||||
"crypto/tls"
|
"crypto/tls"
|
||||||
"encoding/base64"
|
"encoding/base64"
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/songquanpeng/one-api/common/config"
|
"github.com/songquanpeng/one-api/common/config"
|
||||||
"net/smtp"
|
"net/smtp"
|
||||||
@@ -11,6 +12,32 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type loginAuth struct {
|
||||||
|
username, password string
|
||||||
|
}
|
||||||
|
|
||||||
|
func LoginAuth(username, password string) smtp.Auth {
|
||||||
|
return &loginAuth{username, password}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *loginAuth) Start(_ *smtp.ServerInfo) (string, []byte, error) {
|
||||||
|
return "LOGIN", []byte(a.username), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *loginAuth) Next(fromServer []byte, more bool) ([]byte, error) {
|
||||||
|
if more {
|
||||||
|
switch string(fromServer) {
|
||||||
|
case "Username:":
|
||||||
|
return []byte(a.username), nil
|
||||||
|
case "Password:":
|
||||||
|
return []byte(a.password), nil
|
||||||
|
default:
|
||||||
|
return nil, errors.New("unknown command from server during login auth")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
func SendEmail(subject string, receiver string, content string) error {
|
func SendEmail(subject string, receiver string, content string) error {
|
||||||
if receiver == "" {
|
if receiver == "" {
|
||||||
return fmt.Errorf("receiver is empty")
|
return fmt.Errorf("receiver is empty")
|
||||||
@@ -41,7 +68,12 @@ func SendEmail(subject string, receiver string, content string) error {
|
|||||||
"Date: %s\r\n"+
|
"Date: %s\r\n"+
|
||||||
"Content-Type: text/html; charset=UTF-8\r\n\r\n%s\r\n",
|
"Content-Type: text/html; charset=UTF-8\r\n\r\n%s\r\n",
|
||||||
receiver, config.SystemName, config.SMTPFrom, encodedSubject, messageId, time.Now().Format(time.RFC1123Z), content))
|
receiver, config.SystemName, config.SMTPFrom, encodedSubject, messageId, time.Now().Format(time.RFC1123Z), content))
|
||||||
auth := smtp.PlainAuth("", config.SMTPAccount, config.SMTPToken, config.SMTPServer)
|
var auth smtp.Auth
|
||||||
|
if config.SMTPAuthLoginEnabled {
|
||||||
|
auth = LoginAuth(config.SMTPAccount, config.SMTPToken)
|
||||||
|
} else {
|
||||||
|
auth = smtp.PlainAuth("", config.SMTPAccount, config.SMTPToken, config.SMTPServer)
|
||||||
|
}
|
||||||
addr := fmt.Sprintf("%s:%d", config.SMTPServer, config.SMTPPort)
|
addr := fmt.Sprintf("%s:%d", config.SMTPServer, config.SMTPPort)
|
||||||
to := strings.Split(receiver, ";")
|
to := strings.Split(receiver, ";")
|
||||||
|
|
||||||
|
|||||||
@@ -330,6 +330,7 @@
|
|||||||
"通常和邮箱地址保持一致": "Usually consistent with the email address",
|
"通常和邮箱地址保持一致": "Usually consistent with the email address",
|
||||||
"SMTP 访问凭证": "SMTP Access Credential",
|
"SMTP 访问凭证": "SMTP Access Credential",
|
||||||
"敏感信息不会发送到前端显示": "Sensitive information will not be displayed in the frontend",
|
"敏感信息不会发送到前端显示": "Sensitive information will not be displayed in the frontend",
|
||||||
|
"使用 SMTP LOGIN 认证方式": "Use LOGIN as SMTP authentication method",
|
||||||
"保存 SMTP 设置": "Save SMTP Settings",
|
"保存 SMTP 设置": "Save SMTP Settings",
|
||||||
"配置 GitHub OAuth App": "Configure GitHub OAuth App",
|
"配置 GitHub OAuth App": "Configure GitHub OAuth App",
|
||||||
"用以支持通过 GitHub 进行登录注册": "To support login & registration via GitHub",
|
"用以支持通过 GitHub 进行登录注册": "To support login & registration via GitHub",
|
||||||
|
|||||||
22
main.go
22
main.go
@@ -27,27 +27,19 @@ func main() {
|
|||||||
common.Init()
|
common.Init()
|
||||||
logger.SetupLogger()
|
logger.SetupLogger()
|
||||||
logger.SysLogf("One API %s started", common.Version)
|
logger.SysLogf("One API %s started", common.Version)
|
||||||
if os.Getenv("GIN_MODE") != "debug" {
|
|
||||||
|
if os.Getenv("GIN_MODE") != gin.DebugMode {
|
||||||
gin.SetMode(gin.ReleaseMode)
|
gin.SetMode(gin.ReleaseMode)
|
||||||
}
|
}
|
||||||
if config.DebugEnabled {
|
if config.DebugEnabled {
|
||||||
logger.SysLog("running in debug mode")
|
logger.SysLog("running in debug mode")
|
||||||
}
|
}
|
||||||
var err error
|
|
||||||
// Initialize SQL Database
|
// Initialize SQL Database
|
||||||
model.DB, err = model.InitDB("SQL_DSN")
|
model.InitDB()
|
||||||
if err != nil {
|
model.InitLogDB()
|
||||||
logger.FatalLog("failed to initialize database: " + err.Error())
|
|
||||||
}
|
var err error
|
||||||
if os.Getenv("LOG_SQL_DSN") != "" {
|
|
||||||
logger.SysLog("using secondary database for table logs")
|
|
||||||
model.LOG_DB, err = model.InitDB("LOG_SQL_DSN")
|
|
||||||
if err != nil {
|
|
||||||
logger.FatalLog("failed to initialize secondary database: " + err.Error())
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
model.LOG_DB = model.DB
|
|
||||||
}
|
|
||||||
err = model.CreateRootAccountIfNeed()
|
err = model.CreateRootAccountIfNeed()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.FatalLog("database init error: " + err.Error())
|
logger.FatalLog("database init error: " + err.Error())
|
||||||
|
|||||||
219
model/main.go
219
model/main.go
@@ -1,6 +1,7 @@
|
|||||||
package model
|
package model
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"database/sql"
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/songquanpeng/one-api/common"
|
"github.com/songquanpeng/one-api/common"
|
||||||
"github.com/songquanpeng/one-api/common/config"
|
"github.com/songquanpeng/one-api/common/config"
|
||||||
@@ -60,90 +61,156 @@ func CreateRootAccountIfNeed() error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func chooseDB(envName string) (*gorm.DB, error) {
|
func chooseDB(envName string) (*gorm.DB, error) {
|
||||||
if os.Getenv(envName) != "" {
|
dsn := os.Getenv(envName)
|
||||||
dsn := os.Getenv(envName)
|
|
||||||
if strings.HasPrefix(dsn, "postgres://") {
|
switch {
|
||||||
// Use PostgreSQL
|
case strings.HasPrefix(dsn, "postgres://"):
|
||||||
logger.SysLog("using PostgreSQL as database")
|
// Use PostgreSQL
|
||||||
common.UsingPostgreSQL = true
|
return openPostgreSQL(dsn)
|
||||||
return gorm.Open(postgres.New(postgres.Config{
|
case dsn != "":
|
||||||
DSN: dsn,
|
|
||||||
PreferSimpleProtocol: true, // disables implicit prepared statement usage
|
|
||||||
}), &gorm.Config{
|
|
||||||
PrepareStmt: true, // precompile SQL
|
|
||||||
})
|
|
||||||
}
|
|
||||||
// Use MySQL
|
// Use MySQL
|
||||||
logger.SysLog("using MySQL as database")
|
return openMySQL(dsn)
|
||||||
common.UsingMySQL = true
|
default:
|
||||||
return gorm.Open(mysql.Open(dsn), &gorm.Config{
|
// Use SQLite
|
||||||
PrepareStmt: true, // precompile SQL
|
return openSQLite()
|
||||||
})
|
|
||||||
}
|
}
|
||||||
// Use SQLite
|
}
|
||||||
logger.SysLog("SQL_DSN not set, using SQLite as database")
|
|
||||||
common.UsingSQLite = true
|
func openPostgreSQL(dsn string) (*gorm.DB, error) {
|
||||||
config := fmt.Sprintf("?_busy_timeout=%d", common.SQLiteBusyTimeout)
|
logger.SysLog("using PostgreSQL as database")
|
||||||
return gorm.Open(sqlite.Open(common.SQLitePath+config), &gorm.Config{
|
common.UsingPostgreSQL = true
|
||||||
|
return gorm.Open(postgres.New(postgres.Config{
|
||||||
|
DSN: dsn,
|
||||||
|
PreferSimpleProtocol: true, // disables implicit prepared statement usage
|
||||||
|
}), &gorm.Config{
|
||||||
PrepareStmt: true, // precompile SQL
|
PrepareStmt: true, // precompile SQL
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func InitDB(envName string) (db *gorm.DB, err error) {
|
func openMySQL(dsn string) (*gorm.DB, error) {
|
||||||
db, err = chooseDB(envName)
|
logger.SysLog("using MySQL as database")
|
||||||
if err == nil {
|
common.UsingMySQL = true
|
||||||
if config.DebugSQLEnabled {
|
return gorm.Open(mysql.Open(dsn), &gorm.Config{
|
||||||
db = db.Debug()
|
PrepareStmt: true, // precompile SQL
|
||||||
}
|
})
|
||||||
sqlDB, err := db.DB()
|
}
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
sqlDB.SetMaxIdleConns(env.Int("SQL_MAX_IDLE_CONNS", 100))
|
|
||||||
sqlDB.SetMaxOpenConns(env.Int("SQL_MAX_OPEN_CONNS", 1000))
|
|
||||||
sqlDB.SetConnMaxLifetime(time.Second * time.Duration(env.Int("SQL_MAX_LIFETIME", 60)))
|
|
||||||
|
|
||||||
if !config.IsMasterNode {
|
func openSQLite() (*gorm.DB, error) {
|
||||||
return db, err
|
logger.SysLog("SQL_DSN not set, using SQLite as database")
|
||||||
}
|
common.UsingSQLite = true
|
||||||
if common.UsingMySQL {
|
dsn := fmt.Sprintf("%s?_busy_timeout=%d", common.SQLitePath, common.SQLiteBusyTimeout)
|
||||||
_, _ = sqlDB.Exec("DROP INDEX idx_channels_key ON channels;") // TODO: delete this line when most users have upgraded
|
return gorm.Open(sqlite.Open(dsn), &gorm.Config{
|
||||||
}
|
PrepareStmt: true, // precompile SQL
|
||||||
logger.SysLog("database migration started")
|
})
|
||||||
err = db.AutoMigrate(&Channel{})
|
}
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
func InitDB() {
|
||||||
}
|
var err error
|
||||||
err = db.AutoMigrate(&Token{})
|
DB, err = chooseDB("SQL_DSN")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
logger.FatalLog("failed to initialize database: " + err.Error())
|
||||||
}
|
return
|
||||||
err = db.AutoMigrate(&User{})
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
err = db.AutoMigrate(&Option{})
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
err = db.AutoMigrate(&Redemption{})
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
err = db.AutoMigrate(&Ability{})
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
err = db.AutoMigrate(&Log{})
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
logger.SysLog("database migrated")
|
|
||||||
return db, err
|
|
||||||
} else {
|
|
||||||
logger.FatalLog(err)
|
|
||||||
}
|
}
|
||||||
return db, err
|
|
||||||
|
sqlDB := setDBConns(DB)
|
||||||
|
|
||||||
|
if !config.IsMasterNode {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if common.UsingMySQL {
|
||||||
|
_, _ = sqlDB.Exec("DROP INDEX idx_channels_key ON channels;") // TODO: delete this line when most users have upgraded
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.SysLog("database migration started")
|
||||||
|
if err = migrateDB(); err != nil {
|
||||||
|
logger.FatalLog("failed to migrate database: " + err.Error())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
logger.SysLog("database migrated")
|
||||||
|
}
|
||||||
|
|
||||||
|
func migrateDB() error {
|
||||||
|
var err error
|
||||||
|
if err = DB.AutoMigrate(&Channel{}); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err = DB.AutoMigrate(&Token{}); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err = DB.AutoMigrate(&User{}); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err = DB.AutoMigrate(&Option{}); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err = DB.AutoMigrate(&Redemption{}); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err = DB.AutoMigrate(&Ability{}); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err = DB.AutoMigrate(&Log{}); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err = DB.AutoMigrate(&Channel{}); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func InitLogDB() {
|
||||||
|
if os.Getenv("LOG_SQL_DSN") == "" {
|
||||||
|
LOG_DB = DB
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.SysLog("using secondary database for table logs")
|
||||||
|
var err error
|
||||||
|
LOG_DB, err = chooseDB("LOG_SQL_DSN")
|
||||||
|
if err != nil {
|
||||||
|
logger.FatalLog("failed to initialize secondary database: " + err.Error())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
setDBConns(LOG_DB)
|
||||||
|
|
||||||
|
if !config.IsMasterNode {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.SysLog("secondary database migration started")
|
||||||
|
err = migrateLOGDB()
|
||||||
|
if err != nil {
|
||||||
|
logger.FatalLog("failed to migrate secondary database: " + err.Error())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
logger.SysLog("secondary database migrated")
|
||||||
|
}
|
||||||
|
|
||||||
|
func migrateLOGDB() error {
|
||||||
|
var err error
|
||||||
|
if err = LOG_DB.AutoMigrate(&Log{}); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func setDBConns(db *gorm.DB) *sql.DB {
|
||||||
|
if config.DebugSQLEnabled {
|
||||||
|
db = db.Debug()
|
||||||
|
}
|
||||||
|
|
||||||
|
sqlDB, err := db.DB()
|
||||||
|
if err != nil {
|
||||||
|
logger.FatalLog("failed to connect database: " + err.Error())
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
sqlDB.SetMaxIdleConns(env.Int("SQL_MAX_IDLE_CONNS", 100))
|
||||||
|
sqlDB.SetMaxOpenConns(env.Int("SQL_MAX_OPEN_CONNS", 1000))
|
||||||
|
sqlDB.SetConnMaxLifetime(time.Second * time.Duration(env.Int("SQL_MAX_LIFETIME", 60)))
|
||||||
|
return sqlDB
|
||||||
}
|
}
|
||||||
|
|
||||||
func closeDB(db *gorm.DB) error {
|
func closeDB(db *gorm.DB) error {
|
||||||
|
|||||||
@@ -45,6 +45,7 @@ func InitOptionMap() {
|
|||||||
config.OptionMap["SMTPPort"] = strconv.Itoa(config.SMTPPort)
|
config.OptionMap["SMTPPort"] = strconv.Itoa(config.SMTPPort)
|
||||||
config.OptionMap["SMTPAccount"] = ""
|
config.OptionMap["SMTPAccount"] = ""
|
||||||
config.OptionMap["SMTPToken"] = ""
|
config.OptionMap["SMTPToken"] = ""
|
||||||
|
config.OptionMap["SMTPAuthLoginEnabled"] = strconv.FormatBool(config.SMTPAuthLoginEnabled)
|
||||||
config.OptionMap["Notice"] = ""
|
config.OptionMap["Notice"] = ""
|
||||||
config.OptionMap["About"] = ""
|
config.OptionMap["About"] = ""
|
||||||
config.OptionMap["HomePageContent"] = ""
|
config.OptionMap["HomePageContent"] = ""
|
||||||
@@ -150,6 +151,8 @@ func updateOptionMap(key string, value string) (err error) {
|
|||||||
config.DisplayInCurrencyEnabled = boolValue
|
config.DisplayInCurrencyEnabled = boolValue
|
||||||
case "DisplayTokenStatEnabled":
|
case "DisplayTokenStatEnabled":
|
||||||
config.DisplayTokenStatEnabled = boolValue
|
config.DisplayTokenStatEnabled = boolValue
|
||||||
|
case "SMTPAuthLoginEnabled":
|
||||||
|
config.SMTPAuthLoginEnabled = boolValue
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
switch key {
|
switch key {
|
||||||
|
|||||||
@@ -18,6 +18,7 @@ const SystemSetting = () => {
|
|||||||
SMTPAccount: '',
|
SMTPAccount: '',
|
||||||
SMTPFrom: '',
|
SMTPFrom: '',
|
||||||
SMTPToken: '',
|
SMTPToken: '',
|
||||||
|
SMTPAuthLoginEnabled: '',
|
||||||
ServerAddress: '',
|
ServerAddress: '',
|
||||||
Footer: '',
|
Footer: '',
|
||||||
WeChatAuthEnabled: '',
|
WeChatAuthEnabled: '',
|
||||||
@@ -76,6 +77,7 @@ const SystemSetting = () => {
|
|||||||
case 'TurnstileCheckEnabled':
|
case 'TurnstileCheckEnabled':
|
||||||
case 'EmailDomainRestrictionEnabled':
|
case 'EmailDomainRestrictionEnabled':
|
||||||
case 'RegisterEnabled':
|
case 'RegisterEnabled':
|
||||||
|
case 'SMTPAuthLoginEnabled':
|
||||||
value = inputs[key] === 'true' ? 'false' : 'true';
|
value = inputs[key] === 'true' ? 'false' : 'true';
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
@@ -107,7 +109,7 @@ const SystemSetting = () => {
|
|||||||
}
|
}
|
||||||
if (
|
if (
|
||||||
name === 'Notice' ||
|
name === 'Notice' ||
|
||||||
name.startsWith('SMTP') ||
|
(name.startsWith('SMTP') && !name.endsWith('Enabled')) ||
|
||||||
name === 'ServerAddress' ||
|
name === 'ServerAddress' ||
|
||||||
name === 'GitHubClientId' ||
|
name === 'GitHubClientId' ||
|
||||||
name === 'GitHubClientSecret' ||
|
name === 'GitHubClientSecret' ||
|
||||||
@@ -444,6 +446,12 @@ const SystemSetting = () => {
|
|||||||
checked={inputs.RegisterEnabled === 'true'}
|
checked={inputs.RegisterEnabled === 'true'}
|
||||||
placeholder='敏感信息不会发送到前端显示'
|
placeholder='敏感信息不会发送到前端显示'
|
||||||
/>
|
/>
|
||||||
|
<Form.Checkbox
|
||||||
|
checked={inputs.SMTPAuthLoginEnabled === 'true'}
|
||||||
|
label='使用 SMTP LOGIN 认证方式'
|
||||||
|
name='SMTPAuthLoginEnabled'
|
||||||
|
onChange={handleInputChange}
|
||||||
|
/>
|
||||||
</Form.Group>
|
</Form.Group>
|
||||||
<Form.Button onClick={submitSMTP}>保存 SMTP 设置</Form.Button>
|
<Form.Button onClick={submitSMTP}>保存 SMTP 设置</Form.Button>
|
||||||
<Divider />
|
<Divider />
|
||||||
|
|||||||
Reference in New Issue
Block a user