2018-10-25 21:51:47 +08:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
2018-11-27 14:51:59 +08:00
|
|
|
"strings"
|
2018-10-25 21:51:47 +08:00
|
|
|
|
2020-03-08 02:47:54 +08:00
|
|
|
"github.com/knadh/listmonk/internal/subimporter"
|
2018-10-25 21:51:47 +08:00
|
|
|
"github.com/labstack/echo"
|
|
|
|
)
|
|
|
|
|
|
|
|
// reqImport represents file upload import params.
|
|
|
|
type reqImport struct {
|
2020-07-06 00:05:17 +08:00
|
|
|
Mode string `json:"mode"`
|
|
|
|
Overwrite bool `json:"overwrite"`
|
|
|
|
Delim string `json:"delim"`
|
|
|
|
ListIDs []int `json:"lists"`
|
2018-10-25 21:51:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// handleImportSubscribers handles the uploading and bulk importing of
|
|
|
|
// a ZIP file of one or more CSV files.
|
|
|
|
func handleImportSubscribers(c echo.Context) error {
|
|
|
|
app := c.Get("app").(*App)
|
|
|
|
|
|
|
|
// Is an import already running?
|
2020-03-08 02:33:22 +08:00
|
|
|
if app.importer.GetStats().Status == subimporter.StatusImporting {
|
2020-12-19 18:55:52 +08:00
|
|
|
return echo.NewHTTPError(http.StatusBadRequest, app.i18n.T("import.alreadyRunning"))
|
2018-10-25 21:51:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Unmarsal the JSON params.
|
|
|
|
var r reqImport
|
|
|
|
if err := json.Unmarshal([]byte(c.FormValue("params")), &r); err != nil {
|
|
|
|
return echo.NewHTTPError(http.StatusBadRequest,
|
2021-01-23 21:24:33 +08:00
|
|
|
app.i18n.Ts("import.invalidParams", "error", err.Error()))
|
2018-10-25 21:51:47 +08:00
|
|
|
}
|
|
|
|
|
2020-08-01 19:15:29 +08:00
|
|
|
if r.Mode != subimporter.ModeSubscribe && r.Mode != subimporter.ModeBlocklist {
|
2020-12-19 18:55:52 +08:00
|
|
|
return echo.NewHTTPError(http.StatusBadRequest, app.i18n.T("import.invalidMode"))
|
2018-11-05 19:29:09 +08:00
|
|
|
}
|
|
|
|
|
2018-10-25 21:51:47 +08:00
|
|
|
if len(r.Delim) != 1 {
|
2020-12-19 18:55:52 +08:00
|
|
|
return echo.NewHTTPError(http.StatusBadRequest, app.i18n.T("import.invalidDelim"))
|
2018-10-25 21:51:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
file, err := c.FormFile("file")
|
|
|
|
if err != nil {
|
|
|
|
return echo.NewHTTPError(http.StatusBadRequest,
|
2021-01-23 21:24:33 +08:00
|
|
|
app.i18n.Ts("import.invalidFile", "error", err.Error()))
|
2018-10-25 21:51:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
src, err := file.Open()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer src.Close()
|
|
|
|
|
|
|
|
out, err := ioutil.TempFile("", "listmonk")
|
|
|
|
if err != nil {
|
|
|
|
return echo.NewHTTPError(http.StatusInternalServerError,
|
2021-01-23 21:24:33 +08:00
|
|
|
app.i18n.Ts("import.errorCopyingFile", "error", err.Error()))
|
2018-10-25 21:51:47 +08:00
|
|
|
}
|
|
|
|
defer out.Close()
|
|
|
|
|
|
|
|
if _, err = io.Copy(out, src); err != nil {
|
|
|
|
return echo.NewHTTPError(http.StatusInternalServerError,
|
2021-01-23 21:24:33 +08:00
|
|
|
app.i18n.Ts("import.errorCopyingFile", "error", err.Error()))
|
2018-10-25 21:51:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Start the importer session.
|
2020-07-06 00:05:17 +08:00
|
|
|
impSess, err := app.importer.NewSession(file.Filename, r.Mode, r.Overwrite, r.ListIDs)
|
2018-10-25 21:51:47 +08:00
|
|
|
if err != nil {
|
2020-12-19 18:55:52 +08:00
|
|
|
return echo.NewHTTPError(http.StatusInternalServerError,
|
2021-01-23 21:24:33 +08:00
|
|
|
app.i18n.Ts("import.errorStarting", "error", err.Error()))
|
2018-10-25 21:51:47 +08:00
|
|
|
}
|
|
|
|
go impSess.Start()
|
|
|
|
|
2018-11-27 14:51:59 +08:00
|
|
|
if strings.HasSuffix(strings.ToLower(file.Filename), ".csv") {
|
|
|
|
go impSess.LoadCSV(out.Name(), rune(r.Delim[0]))
|
|
|
|
} else {
|
|
|
|
// Only 1 CSV from the ZIP is considered. If multiple files have
|
|
|
|
// to be processed, counting the net number of lines (to track progress),
|
|
|
|
// keeping the global import state (failed / successful) etc. across
|
|
|
|
// multiple files becomes complex. Instead, it's just easier for the
|
|
|
|
// end user to concat multiple CSVs (if there are multiple in the first)
|
|
|
|
// place and uploada as one in the first place.
|
|
|
|
dir, files, err := impSess.ExtractZIP(out.Name(), 1)
|
|
|
|
if err != nil {
|
|
|
|
return echo.NewHTTPError(http.StatusInternalServerError,
|
2021-01-23 21:24:33 +08:00
|
|
|
app.i18n.Ts("import.errorProcessingZIP", "error", err.Error()))
|
2018-11-27 14:51:59 +08:00
|
|
|
}
|
|
|
|
go impSess.LoadCSV(dir+"/"+files[0], rune(r.Delim[0]))
|
2018-10-25 21:51:47 +08:00
|
|
|
}
|
|
|
|
|
2020-03-08 02:33:22 +08:00
|
|
|
return c.JSON(http.StatusOK, okResp{app.importer.GetStats()})
|
2018-10-25 21:51:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// handleGetImportSubscribers returns import statistics.
|
|
|
|
func handleGetImportSubscribers(c echo.Context) error {
|
|
|
|
var (
|
|
|
|
app = c.Get("app").(*App)
|
2020-03-08 02:33:22 +08:00
|
|
|
s = app.importer.GetStats()
|
2018-10-25 21:51:47 +08:00
|
|
|
)
|
|
|
|
return c.JSON(http.StatusOK, okResp{s})
|
|
|
|
}
|
|
|
|
|
2018-11-28 15:59:57 +08:00
|
|
|
// handleGetImportSubscriberStats returns import statistics.
|
|
|
|
func handleGetImportSubscriberStats(c echo.Context) error {
|
2018-10-25 21:51:47 +08:00
|
|
|
app := c.Get("app").(*App)
|
2020-03-08 02:33:22 +08:00
|
|
|
return c.JSON(http.StatusOK, okResp{string(app.importer.GetLogs())})
|
2018-10-25 21:51:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// handleStopImportSubscribers sends a stop signal to the importer.
|
|
|
|
// If there's an ongoing import, it'll be stopped, and if an import
|
|
|
|
// is finished, it's state is cleared.
|
|
|
|
func handleStopImportSubscribers(c echo.Context) error {
|
|
|
|
app := c.Get("app").(*App)
|
2020-03-08 02:33:22 +08:00
|
|
|
app.importer.Stop()
|
|
|
|
return c.JSON(http.StatusOK, okResp{app.importer.GetStats()})
|
2018-10-25 21:51:47 +08:00
|
|
|
}
|