domain.go 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271
  1. package api
  2. import (
  3. "github.com/0xJacky/Nginx-UI/server/model"
  4. "github.com/0xJacky/Nginx-UI/server/pkg/config_list"
  5. nginx2 "github.com/0xJacky/Nginx-UI/server/pkg/nginx"
  6. "github.com/gin-gonic/gin"
  7. "net/http"
  8. "os"
  9. "path/filepath"
  10. "strings"
  11. )
  12. func GetDomains(c *gin.Context) {
  13. orderBy := c.Query("order_by")
  14. sort := c.DefaultQuery("sort", "desc")
  15. mySort := map[string]string{
  16. "enabled": "bool",
  17. "name": "string",
  18. "modify": "time",
  19. }
  20. configFiles, err := os.ReadDir(nginx2.GetNginxConfPath("sites-available"))
  21. if err != nil {
  22. ErrHandler(c, err)
  23. return
  24. }
  25. enabledConfig, err := os.ReadDir(filepath.Join(nginx2.GetNginxConfPath("sites-enabled")))
  26. if err != nil {
  27. ErrHandler(c, err)
  28. return
  29. }
  30. enabledConfigMap := make(map[string]bool)
  31. for i := range enabledConfig {
  32. enabledConfigMap[enabledConfig[i].Name()] = true
  33. }
  34. var configs []gin.H
  35. for i := range configFiles {
  36. file := configFiles[i]
  37. fileInfo, _ := file.Info()
  38. if !file.IsDir() {
  39. configs = append(configs, gin.H{
  40. "name": file.Name(),
  41. "size": fileInfo.Size(),
  42. "modify": fileInfo.ModTime(),
  43. "enabled": enabledConfigMap[file.Name()],
  44. })
  45. }
  46. }
  47. configs = config_list.Sort(orderBy, sort, mySort[orderBy], configs)
  48. c.JSON(http.StatusOK, gin.H{
  49. "data": configs,
  50. })
  51. }
  52. func GetDomain(c *gin.Context) {
  53. name := c.Param("name")
  54. path := filepath.Join(nginx2.GetNginxConfPath("sites-available"), name)
  55. enabled := true
  56. if _, err := os.Stat(filepath.Join(nginx2.GetNginxConfPath("sites-enabled"), name)); os.IsNotExist(err) {
  57. enabled = false
  58. }
  59. config, err := nginx2.ParseNgxConfig(path)
  60. if err != nil {
  61. ErrHandler(c, err)
  62. return
  63. }
  64. _, err = model.FirstCert(name)
  65. c.JSON(http.StatusOK, gin.H{
  66. "enabled": enabled,
  67. "name": name,
  68. "config": config.BuildConfig(),
  69. "tokenized": config,
  70. "auto_cert": err == nil,
  71. })
  72. }
  73. func EditDomain(c *gin.Context) {
  74. var err error
  75. name := c.Param("name")
  76. request := make(gin.H)
  77. err = c.BindJSON(&request)
  78. path := filepath.Join(nginx2.GetNginxConfPath("sites-available"), name)
  79. err = os.WriteFile(path, []byte(request["content"].(string)), 0644)
  80. if err != nil {
  81. ErrHandler(c, err)
  82. return
  83. }
  84. enabledConfigFilePath := filepath.Join(nginx2.GetNginxConfPath("sites-enabled"), name)
  85. if _, err = os.Stat(enabledConfigFilePath); err == nil {
  86. // Test nginx configuration
  87. err = nginx2.TestNginxConf()
  88. if err != nil {
  89. c.JSON(http.StatusInternalServerError, gin.H{
  90. "message": err.Error(),
  91. })
  92. return
  93. }
  94. output := nginx2.ReloadNginx()
  95. if output != "" && strings.Contains(output, "error") {
  96. c.JSON(http.StatusInternalServerError, gin.H{
  97. "message": output,
  98. })
  99. return
  100. }
  101. }
  102. GetDomain(c)
  103. }
  104. func EnableDomain(c *gin.Context) {
  105. configFilePath := filepath.Join(nginx2.GetNginxConfPath("sites-available"), c.Param("name"))
  106. enabledConfigFilePath := filepath.Join(nginx2.GetNginxConfPath("sites-enabled"), c.Param("name"))
  107. _, err := os.Stat(configFilePath)
  108. if err != nil {
  109. ErrHandler(c, err)
  110. return
  111. }
  112. if _, err = os.Stat(enabledConfigFilePath); os.IsNotExist(err) {
  113. err = os.Symlink(configFilePath, enabledConfigFilePath)
  114. if err != nil {
  115. ErrHandler(c, err)
  116. return
  117. }
  118. }
  119. // Test nginx config, if not pass then rollback.
  120. err = nginx2.TestNginxConf()
  121. if err != nil {
  122. _ = os.Remove(enabledConfigFilePath)
  123. c.JSON(http.StatusInternalServerError, gin.H{
  124. "message": err.Error(),
  125. })
  126. return
  127. }
  128. output := nginx2.ReloadNginx()
  129. if output != "" && strings.Contains(output, "error") {
  130. c.JSON(http.StatusInternalServerError, gin.H{
  131. "message": output,
  132. })
  133. return
  134. }
  135. c.JSON(http.StatusOK, gin.H{
  136. "message": "ok",
  137. })
  138. }
  139. func DisableDomain(c *gin.Context) {
  140. enabledConfigFilePath := filepath.Join(nginx2.GetNginxConfPath("sites-enabled"), c.Param("name"))
  141. _, err := os.Stat(enabledConfigFilePath)
  142. if err != nil {
  143. ErrHandler(c, err)
  144. return
  145. }
  146. err = os.Remove(enabledConfigFilePath)
  147. if err != nil {
  148. ErrHandler(c, err)
  149. return
  150. }
  151. // delete auto cert record
  152. cert := model.Cert{Domain: c.Param("name")}
  153. err = cert.Remove()
  154. if err != nil {
  155. ErrHandler(c, err)
  156. return
  157. }
  158. output := nginx2.ReloadNginx()
  159. if output != "" {
  160. c.JSON(http.StatusInternalServerError, gin.H{
  161. "message": output,
  162. })
  163. return
  164. }
  165. c.JSON(http.StatusOK, gin.H{
  166. "message": "ok",
  167. })
  168. }
  169. func DeleteDomain(c *gin.Context) {
  170. var err error
  171. name := c.Param("name")
  172. availablePath := filepath.Join(nginx2.GetNginxConfPath("sites-available"), name)
  173. enabledPath := filepath.Join(nginx2.GetNginxConfPath("sites-enabled"), name)
  174. if _, err = os.Stat(availablePath); os.IsNotExist(err) {
  175. c.JSON(http.StatusNotFound, gin.H{
  176. "message": "site not found",
  177. })
  178. return
  179. }
  180. if _, err = os.Stat(enabledPath); err == nil {
  181. c.JSON(http.StatusNotAcceptable, gin.H{
  182. "message": "site is enabled",
  183. })
  184. return
  185. }
  186. cert := model.Cert{Domain: name}
  187. _ = cert.Remove()
  188. err = os.Remove(availablePath)
  189. if err != nil {
  190. ErrHandler(c, err)
  191. return
  192. }
  193. c.JSON(http.StatusOK, gin.H{
  194. "message": "ok",
  195. })
  196. }
  197. func AddDomainToAutoCert(c *gin.Context) {
  198. domain := c.Param("domain")
  199. cert, err := model.FirstOrCreateCert(domain)
  200. if err != nil {
  201. ErrHandler(c, err)
  202. return
  203. }
  204. c.JSON(http.StatusOK, cert)
  205. }
  206. func RemoveDomainFromAutoCert(c *gin.Context) {
  207. cert := model.Cert{
  208. Domain: c.Param("domain"),
  209. }
  210. err := cert.Remove()
  211. if err != nil {
  212. ErrHandler(c, err)
  213. return
  214. }
  215. c.JSON(http.StatusOK, nil)
  216. }