mirror of
https://github.com/acepanel/panel.git
synced 2026-02-04 05:31:44 +08:00
187 lines
4.7 KiB
Go
187 lines
4.7 KiB
Go
package pureftpd
|
|
|
|
import (
|
|
"net/http"
|
|
"regexp"
|
|
"strings"
|
|
|
|
"github.com/go-chi/chi/v5"
|
|
"github.com/go-rat/chix"
|
|
"github.com/spf13/cast"
|
|
|
|
"github.com/tnb-labs/panel/internal/app"
|
|
"github.com/tnb-labs/panel/internal/service"
|
|
"github.com/tnb-labs/panel/pkg/firewall"
|
|
"github.com/tnb-labs/panel/pkg/io"
|
|
"github.com/tnb-labs/panel/pkg/shell"
|
|
"github.com/tnb-labs/panel/pkg/systemctl"
|
|
)
|
|
|
|
type App struct{}
|
|
|
|
func NewApp() *App {
|
|
return &App{}
|
|
}
|
|
|
|
func (s *App) Route(r chi.Router) {
|
|
r.Get("/users", s.List)
|
|
r.Post("/users", s.Create)
|
|
r.Delete("/users/{username}", s.Delete)
|
|
r.Post("/users/{username}/password", s.ChangePassword)
|
|
r.Get("/port", s.GetPort)
|
|
r.Post("/port", s.UpdatePort)
|
|
}
|
|
|
|
// List 获取用户列表
|
|
func (s *App) List(w http.ResponseWriter, r *http.Request) {
|
|
listRaw, err := shell.Execf("pure-pw list")
|
|
if err != nil {
|
|
service.Success(w, chix.M{
|
|
"total": 0,
|
|
"items": []User{},
|
|
})
|
|
}
|
|
|
|
listArr := strings.Split(listRaw, "\n")
|
|
var users []User
|
|
for _, v := range listArr {
|
|
if len(v) == 0 {
|
|
continue
|
|
}
|
|
|
|
match := regexp.MustCompile(`(\S+)\s+(\S+)`).FindStringSubmatch(v)
|
|
users = append(users, User{
|
|
Username: match[1],
|
|
Path: strings.Replace(match[2], "/./", "/", 1),
|
|
})
|
|
}
|
|
|
|
paged, total := service.Paginate(r, users)
|
|
|
|
service.Success(w, chix.M{
|
|
"total": total,
|
|
"items": paged,
|
|
})
|
|
}
|
|
|
|
// Create 创建用户
|
|
func (s *App) Create(w http.ResponseWriter, r *http.Request) {
|
|
req, err := service.Bind[Create](r)
|
|
if err != nil {
|
|
service.Error(w, http.StatusUnprocessableEntity, "%v", err)
|
|
return
|
|
}
|
|
|
|
if !strings.HasPrefix(req.Path, "/") {
|
|
req.Path = "/" + req.Path
|
|
}
|
|
if !io.Exists(req.Path) {
|
|
service.Error(w, http.StatusUnprocessableEntity, "目录不存在")
|
|
return
|
|
}
|
|
|
|
if err = io.Chmod(req.Path, 0755); err != nil {
|
|
service.Error(w, http.StatusUnprocessableEntity, "修改目录权限失败")
|
|
return
|
|
}
|
|
if err = io.Chown(req.Path, "www", "www"); err != nil {
|
|
service.Error(w, http.StatusUnprocessableEntity, "修改目录权限失败")
|
|
return
|
|
}
|
|
if _, err = shell.Execf(`yes '%s' | pure-pw useradd '%s' -u www -g www -d '%s'`, req.Password, req.Username, req.Path); err != nil {
|
|
service.Error(w, http.StatusInternalServerError, "%v", err)
|
|
return
|
|
}
|
|
if _, err = shell.Execf("pure-pw mkdb"); err != nil {
|
|
service.Error(w, http.StatusInternalServerError, "%v", err)
|
|
return
|
|
}
|
|
|
|
service.Success(w, nil)
|
|
}
|
|
|
|
// Delete 删除用户
|
|
func (s *App) Delete(w http.ResponseWriter, r *http.Request) {
|
|
req, err := service.Bind[Delete](r)
|
|
if err != nil {
|
|
service.Error(w, http.StatusUnprocessableEntity, "%v", err)
|
|
return
|
|
}
|
|
|
|
if _, err = shell.Execf("pure-pw userdel '%s' -m", req.Username); err != nil {
|
|
service.Error(w, http.StatusInternalServerError, "%v", err)
|
|
return
|
|
}
|
|
if _, err = shell.Execf("pure-pw mkdb"); err != nil {
|
|
service.Error(w, http.StatusInternalServerError, "%v", err)
|
|
return
|
|
}
|
|
|
|
service.Success(w, nil)
|
|
}
|
|
|
|
// ChangePassword 修改密码
|
|
func (s *App) ChangePassword(w http.ResponseWriter, r *http.Request) {
|
|
req, err := service.Bind[ChangePassword](r)
|
|
if err != nil {
|
|
service.Error(w, http.StatusUnprocessableEntity, "%v", err)
|
|
return
|
|
}
|
|
|
|
if _, err = shell.Execf(`yes '%s' | pure-pw passwd '%s' -m`, req.Password, req.Username); err != nil {
|
|
service.Error(w, http.StatusInternalServerError, "%v", err)
|
|
return
|
|
}
|
|
if _, err = shell.Execf("pure-pw mkdb"); err != nil {
|
|
service.Error(w, http.StatusInternalServerError, "%v", err)
|
|
return
|
|
}
|
|
|
|
service.Success(w, nil)
|
|
}
|
|
|
|
// GetPort 获取端口
|
|
func (s *App) GetPort(w http.ResponseWriter, r *http.Request) {
|
|
port, err := shell.Execf(`cat %s/server/pure-ftpd/etc/pure-ftpd.conf | grep "Bind" | awk '{print $2}' | awk -F "," '{print $2}'`, app.Root)
|
|
if err != nil {
|
|
service.Error(w, http.StatusInternalServerError, "获取PureFtpd端口失败")
|
|
return
|
|
}
|
|
|
|
service.Success(w, cast.ToInt(port))
|
|
}
|
|
|
|
// UpdatePort 设置端口
|
|
func (s *App) UpdatePort(w http.ResponseWriter, r *http.Request) {
|
|
req, err := service.Bind[UpdatePort](r)
|
|
if err != nil {
|
|
service.Error(w, http.StatusUnprocessableEntity, "%v", err)
|
|
return
|
|
}
|
|
|
|
if _, err = shell.Execf(`sed -i "s/Bind.*/Bind 0.0.0.0,%d/g" %s/server/pure-ftpd/etc/pure-ftpd.conf`, req.Port, app.Root); err != nil {
|
|
service.Error(w, http.StatusInternalServerError, "%v", err)
|
|
return
|
|
}
|
|
|
|
fw := firewall.NewFirewall()
|
|
err = fw.Port(firewall.FireInfo{
|
|
Type: firewall.TypeNormal,
|
|
PortStart: req.Port,
|
|
PortEnd: req.Port,
|
|
Direction: firewall.DirectionIn,
|
|
Strategy: firewall.StrategyAccept,
|
|
}, firewall.OperationAdd)
|
|
if err != nil {
|
|
service.Error(w, http.StatusInternalServerError, "%v", err)
|
|
return
|
|
}
|
|
|
|
if err = systemctl.Restart("pure-ftpd"); err != nil {
|
|
service.Error(w, http.StatusInternalServerError, "%v", err)
|
|
return
|
|
}
|
|
|
|
service.Success(w, nil)
|
|
}
|