Compare commits
	
		
			12 Commits
		
	
	
		
			v1.0.3
			...
			67b17c1df7
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 67b17c1df7 | |||
| 142c09e624 | |||
| 0b3797dc19 | |||
| d4dae21d8f | |||
| 9f2677485e | |||
| 8aa915e6dc | |||
| 37c390f911 | |||
| 2f9552e771 | |||
| dfe21445e5 | |||
| 5397ba0907 | |||
| 8c010665e1 | |||
| 916d3004de | 
| @@ -9,6 +9,7 @@ Create a `config.toml` file in the same directory as the drmdtool executable: | ||||
| ```toml | ||||
| BaseDir = "/path/to/save/downloads" | ||||
| Format = "mkv" | ||||
| TempBaseDir = "/tmp/nre" | ||||
|  | ||||
| [N_m3u8DL-RE] | ||||
| Path = "/path/to/N_m3u8DL-RE" | ||||
| @@ -36,10 +37,4 @@ To process a file directly from the command line: | ||||
| ./drmdtool -f /path/to/file.drmd | ||||
| ``` | ||||
|  | ||||
| This will download the file and save it in the base directory specified in the config. | ||||
|  | ||||
| ## TODO | ||||
| - ~~Filename Sanitation (Makes new directory on /... oops)~~ | ||||
| - ~~GoPlay Fix~~ | ||||
| - Windows? | ||||
| - Proper UI? | ||||
| This will download the file and save it in the base directory specified in the config. | ||||
| @@ -9,9 +9,10 @@ import ( | ||||
| ) | ||||
|  | ||||
| type Config struct { | ||||
| 	BaseDir    string | ||||
| 	Format     string | ||||
| 	N_m3u8DLRE struct { | ||||
| 	BaseDir     string | ||||
| 	Format      string | ||||
| 	TempBaseDir string | ||||
| 	N_m3u8DLRE  struct { | ||||
| 		Path string | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| BaseDir = "/mnt/media" | ||||
| Format = "mkv"  | ||||
| TempBaseDir = "/tmp/nre" | ||||
|  | ||||
| [N_m3u8DLRE] | ||||
| Path = "nre" | ||||
|   | ||||
| @@ -2,7 +2,6 @@ package main | ||||
|  | ||||
| import ( | ||||
| 	"encoding/base64" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"net/http" | ||||
| @@ -12,38 +11,6 @@ import ( | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| func processInputFile(inputFile string) error { | ||||
| 	jsonFile, err := os.Open(inputFile) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("error opening file %s: %v", inputFile, err) | ||||
| 	} | ||||
| 	defer jsonFile.Close() | ||||
|  | ||||
| 	byteValue, err := io.ReadAll(jsonFile) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("error reading file %s: %v", inputFile, err) | ||||
| 	} | ||||
|  | ||||
| 	byteValue = removeBOM(byteValue) | ||||
|  | ||||
| 	var items Items | ||||
| 	err = json.Unmarshal(byteValue, &items) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("error unmarshaling JSON: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	for i, item := range items.Items { | ||||
| 		updateProgress(filepath.Base(inputFile), float64(i)/float64(len(items.Items))*100, item.Filename) | ||||
| 		err := downloadFile(item) | ||||
| 		if err != nil { | ||||
| 			fmt.Printf("Error downloading file: %v\n", err) | ||||
| 		} | ||||
| 	} | ||||
| 	updateProgress(filepath.Base(inputFile), 100, "") | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func removeBOM(input []byte) []byte { | ||||
| 	if len(input) >= 3 && input[0] == 0xEF && input[1] == 0xBB && input[2] == 0xBF { | ||||
| 		return input[3:] | ||||
| @@ -51,9 +18,17 @@ func removeBOM(input []byte) []byte { | ||||
| 	return input | ||||
| } | ||||
|  | ||||
| func downloadFile(item Item) error { | ||||
| func downloadFile(item Item, jobInfo *JobInfo) error { | ||||
| 	fmt.Println("Downloading:", item.Filename) | ||||
|  | ||||
| 	tempDir := filepath.Join(config.TempBaseDir, sanitizeFilename(item.Filename)) | ||||
| 	err := os.MkdirAll(tempDir, 0755) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("error creating temporary directory: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	jobInfo.TempDir = tempDir | ||||
|  | ||||
| 	mpdPath := item.MPD | ||||
| 	if !isValidURL(item.MPD) { | ||||
| 		decodedMPD, err := base64.StdEncoding.DecodeString(item.MPD) | ||||
| @@ -108,7 +83,7 @@ func downloadFile(item Item) error { | ||||
| 		mpdPath = tempFile.Name() | ||||
| 	} | ||||
|  | ||||
| 	command := getDownloadCommand(item, mpdPath) | ||||
| 	command := getDownloadCommand(item, mpdPath, tempDir) | ||||
|  | ||||
| 	if item.Subtitles != "" { | ||||
| 		subtitlePaths, err := downloadAndConvertSubtitles(item.Subtitles) | ||||
| @@ -124,19 +99,45 @@ func downloadFile(item Item) error { | ||||
|  | ||||
| 	cmd := exec.Command("bash", "-c", command) | ||||
|  | ||||
| 	jobsMutex.Lock() | ||||
| 	jobInfo.Cmd = cmd | ||||
| 	jobsMutex.Unlock() | ||||
|  | ||||
| 	cmd.Stdout = os.Stdout | ||||
| 	cmd.Stderr = os.Stderr | ||||
|  | ||||
| 	err := cmd.Run() | ||||
| 	err = cmd.Start() | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("error executing download command: %v", err) | ||||
| 		return fmt.Errorf("error starting download command: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	done := make(chan error) | ||||
| 	go func() { | ||||
| 		done <- cmd.Wait() | ||||
| 	}() | ||||
|  | ||||
| 	select { | ||||
| 	case <-jobInfo.AbortChan: | ||||
| 		if cmd.Process != nil { | ||||
| 			cmd.Process.Kill() | ||||
| 		} | ||||
| 		os.RemoveAll(tempDir) | ||||
| 		return fmt.Errorf("download aborted") | ||||
| 	case err := <-done: | ||||
| 		if jobInfo.Paused { | ||||
| 			return fmt.Errorf("download paused") | ||||
| 		} | ||||
| 		if err != nil { | ||||
| 			return fmt.Errorf("error executing download command: %v", err) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	fmt.Println("Download completed successfully") | ||||
| 	os.RemoveAll(tempDir) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func getDownloadCommand(item Item, mpdPath string) string { | ||||
| func getDownloadCommand(item Item, mpdPath string, tempDir string) string { | ||||
| 	metadata := parseMetadata(item.Metadata) | ||||
| 	keys := getKeys(item.Keys) | ||||
|  | ||||
| @@ -165,6 +166,8 @@ func getDownloadCommand(item Item, mpdPath string) string { | ||||
| 	} | ||||
| 	command += fmt.Sprintf(" --save-dir \"%s\"", saveDir) | ||||
|  | ||||
| 	command += fmt.Sprintf(" --tmp-dir \"%s\"", tempDir) | ||||
|  | ||||
| 	fmt.Println(command) | ||||
|  | ||||
| 	return command | ||||
|   | ||||
							
								
								
									
										2
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								go.sum
									
									
									
									
									
								
							| @@ -10,6 +10,8 @@ github.com/beevik/etree v1.4.1 h1:PmQJDDYahBGNKDcpdX8uPy1xRCwoCGVUiW669MEirVI= | ||||
| github.com/beevik/etree v1.4.1/go.mod h1:gPNJNaBGVZ9AwsidazFZyygnd+0pAU38N4D+WemwKNs= | ||||
| github.com/davecgh/go-spew v1.1.0 h1:ZDRjVQ15GmhC3fiQ8ni8+OwkZQO4DARzQgrnXU1Liz8= | ||||
| github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/pkg/exec v0.0.0-20150614095509-0bd164ad2a5a h1:EN123kAtAAE2pg/+TvBsUBZfHCWNNFyL2ZBPPfNWAc0= | ||||
| github.com/pkg/exec v0.0.0-20150614095509-0bd164ad2a5a/go.mod h1:b95YoNrAnScjaWG+asr8lxqlrsPUcT2ZEBcjvVGshMo= | ||||
| github.com/pkg/profile v1.4.0/go.mod h1:NWz/XGvpEW1FyYQ7fCx4dqYBLlfTcE+A9FLAkNKqjFE= | ||||
| github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | ||||
| github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||||
|   | ||||
							
								
								
									
										229
									
								
								handlers.go
									
									
									
									
									
								
							
							
						
						
									
										229
									
								
								handlers.go
									
									
									
									
									
								
							| @@ -9,20 +9,41 @@ import ( | ||||
| 	"path/filepath" | ||||
| ) | ||||
|  | ||||
| type ProgressInfo struct { | ||||
| 	Percentage  float64 | ||||
| 	CurrentFile string | ||||
| 	Paused      bool | ||||
| } | ||||
|  | ||||
| func handleRoot(w http.ResponseWriter, r *http.Request) { | ||||
| 	if r.URL.Path != "/" { | ||||
| 		http.NotFound(w, r) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	progressMutex.Lock() | ||||
| 	jobs := make(map[string]*ProgressInfo) | ||||
| 	for k, v := range progress { | ||||
| 		jobs[k] = v | ||||
| 	} | ||||
| 	progressMutex.Unlock() | ||||
| 	defer progressMutex.Unlock() | ||||
|  | ||||
| 	err := templates.ExecuteTemplate(w, "index", struct{ Jobs map[string]*ProgressInfo }{jobs}) | ||||
| 	jobsInfo := make(map[string]struct { | ||||
| 		Percentage  float64 | ||||
| 		CurrentFile string | ||||
| 		Paused      bool | ||||
| 	}) | ||||
|  | ||||
| 	for filename, info := range progress { | ||||
| 		jobsInfo[filename] = struct { | ||||
| 			Percentage  float64 | ||||
| 			CurrentFile string | ||||
| 			Paused      bool | ||||
| 		}{ | ||||
| 			Percentage:  info.Percentage, | ||||
| 			CurrentFile: info.CurrentFile, | ||||
| 			Paused:      info.Paused, | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	err := templates.ExecuteTemplate(w, "index", struct { | ||||
| 		Jobs map[string]struct { | ||||
| 			Percentage  float64 | ||||
| 			CurrentFile string | ||||
| 			Paused      bool | ||||
| 		} | ||||
| 	}{jobsInfo}) | ||||
| 	if err != nil { | ||||
| 		http.Error(w, err.Error(), http.StatusInternalServerError) | ||||
| 	} | ||||
| @@ -51,16 +72,59 @@ func handleUpload(w http.ResponseWriter, r *http.Request) { | ||||
|  | ||||
| 	tempFilename := filepath.Base(tempFile.Name()) | ||||
|  | ||||
| 	_, err = parseInputFile(tempFile.Name()) | ||||
| 	if err != nil { | ||||
| 		http.Error(w, err.Error(), http.StatusInternalServerError) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	http.Redirect(w, r, "/select?filename="+tempFilename, http.StatusSeeOther) | ||||
| } | ||||
|  | ||||
| func handleSelect(w http.ResponseWriter, r *http.Request) { | ||||
| 	filename := r.URL.Query().Get("filename") | ||||
| 	items, err := parseInputFile(filepath.Join(uploadDir, filename)) | ||||
| 	if err != nil { | ||||
| 		http.Error(w, err.Error(), http.StatusInternalServerError) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	groupedItems := groupItemsBySeason(items) | ||||
|  | ||||
| 	err = templates.ExecuteTemplate(w, "select", struct { | ||||
| 		Filename string | ||||
| 		Items    map[string][]Item | ||||
| 	}{ | ||||
| 		Filename: filename, | ||||
| 		Items:    groupedItems, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		http.Error(w, err.Error(), http.StatusInternalServerError) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func handleProcess(w http.ResponseWriter, r *http.Request) { | ||||
| 	filename := r.FormValue("filename") | ||||
| 	selectedItems := r.Form["items"] | ||||
|  | ||||
| 	items, err := parseInputFile(filepath.Join(uploadDir, filename)) | ||||
| 	if err != nil { | ||||
| 		http.Error(w, err.Error(), http.StatusInternalServerError) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	filteredItems := filterSelectedItems(items, selectedItems) | ||||
|  | ||||
| 	go func() { | ||||
| 		err := processInputFile(tempFile.Name()) | ||||
| 		err := processItems(filename, filteredItems) | ||||
| 		if err != nil { | ||||
| 			fmt.Printf("Error processing file: %v\n", err) | ||||
| 		} | ||||
|  | ||||
| 		os.Remove(tempFile.Name()) | ||||
| 		os.Remove(filepath.Join(uploadDir, filename)) | ||||
| 	}() | ||||
|  | ||||
| 	http.Redirect(w, r, "/progress?filename="+tempFilename, http.StatusSeeOther) | ||||
| 	http.Redirect(w, r, "/progress?filename="+filename, http.StatusSeeOther) | ||||
| } | ||||
|  | ||||
| func handleProgress(w http.ResponseWriter, r *http.Request) { | ||||
| @@ -92,3 +156,140 @@ func handleProgress(w http.ResponseWriter, r *http.Request) { | ||||
| 		http.Error(w, err.Error(), http.StatusInternalServerError) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func handlePause(w http.ResponseWriter, r *http.Request) { | ||||
| 	filename := r.URL.Query().Get("filename") | ||||
| 	if filename == "" { | ||||
| 		http.Error(w, "Filename is required", http.StatusBadRequest) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	jobsMutex.Lock() | ||||
| 	jobInfo, exists := jobs[filename] | ||||
| 	jobsMutex.Unlock() | ||||
|  | ||||
| 	if !exists { | ||||
| 		http.Error(w, "Job not found", http.StatusNotFound) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	jobInfo.Paused = true | ||||
| 	if jobInfo.Cmd != nil && jobInfo.Cmd.Process != nil { | ||||
| 		jobInfo.Cmd.Process.Kill() | ||||
| 	} | ||||
|  | ||||
| 	progressMutex.Lock() | ||||
| 	if progressInfo, ok := progress[filename]; ok { | ||||
| 		progressInfo.Paused = true | ||||
| 	} | ||||
| 	progressMutex.Unlock() | ||||
|  | ||||
| 	fmt.Fprintf(w, "Pause signal sent for %s", filename) | ||||
| } | ||||
|  | ||||
| func handleResume(w http.ResponseWriter, r *http.Request) { | ||||
| 	filename := r.URL.Query().Get("filename") | ||||
| 	if filename == "" { | ||||
| 		http.Error(w, "Filename is required", http.StatusBadRequest) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	jobsMutex.Lock() | ||||
| 	jobInfo, exists := jobs[filename] | ||||
| 	jobsMutex.Unlock() | ||||
|  | ||||
| 	if !exists { | ||||
| 		http.Error(w, "Job not found", http.StatusNotFound) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	jobInfo.Paused = false | ||||
| 	jobInfo.ResumeChan <- struct{}{} | ||||
|  | ||||
| 	// Update the progress information | ||||
| 	progressMutex.Lock() | ||||
| 	if progressInfo, ok := progress[filename]; ok { | ||||
| 		progressInfo.Paused = false | ||||
| 	} | ||||
| 	progressMutex.Unlock() | ||||
|  | ||||
| 	fmt.Fprintf(w, "Resume signal sent for %s", filename) | ||||
| } | ||||
|  | ||||
| func handleAbort(w http.ResponseWriter, r *http.Request) { | ||||
| 	filename := r.URL.Query().Get("filename") | ||||
| 	if filename == "" { | ||||
| 		http.Error(w, "Filename is required", http.StatusBadRequest) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	jobsMutex.Lock() | ||||
| 	jobInfo, exists := jobs[filename] | ||||
| 	jobsMutex.Unlock() | ||||
|  | ||||
| 	if !exists { | ||||
| 		http.Error(w, "Job not found", http.StatusNotFound) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	close(jobInfo.AbortChan) | ||||
| 	if jobInfo.Cmd != nil && jobInfo.Cmd.Process != nil { | ||||
| 		jobInfo.Cmd.Process.Kill() | ||||
| 	} | ||||
|  | ||||
| 	if jobInfo.TempDir != "" { | ||||
| 		os.RemoveAll(jobInfo.TempDir) | ||||
| 	} | ||||
|  | ||||
| 	fmt.Fprintf(w, "Abort signal sent for %s", filename) | ||||
| } | ||||
|  | ||||
| func handleClearCompleted(w http.ResponseWriter, r *http.Request) { | ||||
| 	if r.Method != http.MethodPost { | ||||
| 		http.Error(w, "Method not allowed", http.StatusMethodNotAllowed) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	clearCompletedJobs() | ||||
|  | ||||
| 	w.Header().Set("Content-Type", "application/json") | ||||
| 	json.NewEncoder(w).Encode(map[string]bool{"success": true}) | ||||
| } | ||||
|  | ||||
| func clearCompletedJobs() { | ||||
| 	progressMutex.Lock() | ||||
| 	defer progressMutex.Unlock() | ||||
|  | ||||
| 	for filename, info := range progress { | ||||
| 		if info.Percentage >= 100 { | ||||
| 			delete(progress, filename) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func updateProgress(filename string, value float64, currentFile string) { | ||||
| 	progressMutex.Lock() | ||||
| 	defer progressMutex.Unlock() | ||||
|  | ||||
| 	jobsMutex.Lock() | ||||
| 	jobInfo, exists := jobs[filename] | ||||
| 	jobsMutex.Unlock() | ||||
|  | ||||
| 	paused := false | ||||
| 	if exists { | ||||
| 		paused = jobInfo.Paused | ||||
| 	} | ||||
|  | ||||
| 	if existingProgress, ok := progress[filename]; ok { | ||||
| 		existingProgress.Percentage = value | ||||
| 		existingProgress.CurrentFile = currentFile | ||||
| 		existingProgress.Paused = paused | ||||
| 	} else { | ||||
| 		progress[filename] = &ProgressInfo{ | ||||
| 			Percentage:  value, | ||||
| 			CurrentFile: currentFile, | ||||
| 			Paused:      paused, | ||||
| 		} | ||||
| 	} | ||||
| 	fmt.Printf("Progress updated for %s: %.2f%%, Current file: %s, Paused: %v\n", filename, value, currentFile, paused) | ||||
| } | ||||
|   | ||||
							
								
								
									
										28
									
								
								main.go
									
									
									
									
									
								
							
							
						
						
									
										28
									
								
								main.go
									
									
									
									
									
								
							| @@ -37,11 +37,6 @@ var progress = make(map[string]*ProgressInfo) | ||||
|  | ||||
| const uploadDir = "uploads" | ||||
|  | ||||
| type ProgressInfo struct { | ||||
| 	Percentage  float64 | ||||
| 	CurrentFile string | ||||
| } | ||||
|  | ||||
| var templates *template.Template | ||||
|  | ||||
| //go:embed templates | ||||
| @@ -63,29 +58,30 @@ func main() { | ||||
| 	if *inputFile == "" { | ||||
| 		startWebServer() | ||||
| 	} else { | ||||
| 		processInputFile(*inputFile) | ||||
| 		items, err := parseInputFile(*inputFile) | ||||
| 		if err != nil { | ||||
| 			fmt.Printf("Error parsing input file: %v\n", err) | ||||
| 			return | ||||
| 		} | ||||
| 		processItems(*inputFile, items) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func startWebServer() { | ||||
| 	http.HandleFunc("/", handleRoot) | ||||
| 	http.HandleFunc("/upload", handleUpload) | ||||
| 	http.HandleFunc("/select", handleSelect) | ||||
| 	http.HandleFunc("/process", handleProcess) | ||||
| 	http.HandleFunc("/progress", handleProgress) | ||||
| 	http.HandleFunc("/abort", handleAbort) | ||||
| 	http.HandleFunc("/pause", handlePause) | ||||
| 	http.HandleFunc("/resume", handleResume) | ||||
| 	http.HandleFunc("/clear-completed", handleClearCompleted) | ||||
|  | ||||
| 	fmt.Println("Starting web server on http://0.0.0.0:8080") | ||||
| 	http.ListenAndServe(":8080", nil) | ||||
| } | ||||
|  | ||||
| func updateProgress(filename string, value float64, currentFile string) { | ||||
| 	progressMutex.Lock() | ||||
| 	defer progressMutex.Unlock() | ||||
| 	progress[filename] = &ProgressInfo{ | ||||
| 		Percentage:  value, | ||||
| 		CurrentFile: currentFile, | ||||
| 	} | ||||
| 	fmt.Printf("Progress updated for %s: %.2f%%, Current file: %s\n", filename, value, currentFile) | ||||
| } | ||||
|  | ||||
| func getProgress(filename string) *ProgressInfo { | ||||
| 	progressMutex.Lock() | ||||
| 	defer progressMutex.Unlock() | ||||
|   | ||||
| @@ -45,6 +45,7 @@ | ||||
|         ul { | ||||
|             list-style-type: none; | ||||
|             padding: 0; | ||||
|             margin-bottom: 10px;  | ||||
|         } | ||||
|         li { | ||||
|             background-color: #2d2d2d; | ||||
| @@ -73,6 +74,9 @@ | ||||
|             display: inline-block; | ||||
|             width: 5em; | ||||
|         } | ||||
|         .paused { | ||||
|             color: #ffa500; | ||||
|         } | ||||
|         @media (max-width: 600px) { | ||||
|             body { | ||||
|                 padding: 10px; | ||||
| @@ -83,6 +87,25 @@ | ||||
|             input[type="file"], input[type="submit"] { | ||||
|                 font-size: 16px; | ||||
|             } | ||||
|             input[type="submit"], #clear-completed { | ||||
|                 font-size: 16px; | ||||
|             } | ||||
|         } | ||||
|         input[type="submit"], #clear-completed { | ||||
|             cursor: pointer; | ||||
|             color: white; | ||||
|             border: 1px solid #444; | ||||
|             padding: 8px 12px; | ||||
|             border-radius: 4px; | ||||
|             margin-bottom: 10px; | ||||
|             max-width: 100%; | ||||
|             width: 100%; | ||||
|         } | ||||
|         #clear-completed { | ||||
|             background-color: #f44336; | ||||
|         } | ||||
|         #clear-completed:hover { | ||||
|             background-color: #d32f2f; | ||||
|         } | ||||
|     </style> | ||||
| </head> | ||||
| @@ -100,12 +123,30 @@ | ||||
|                     <a href="/progress?filename={{$filename}}">{{$filename}}</a> | ||||
|                 </div> | ||||
|                 <div class="job-info"> | ||||
|                     Progress: <span class="progress-text">{{printf "%5.1f%%" $info.Percentage}}</span> Current file: {{$info.CurrentFile}} | ||||
|                     Progress: <span class="progress-text">{{printf "%5.1f%%" $info.Percentage}}</span> | ||||
|                     Current file: {{$info.CurrentFile}} | ||||
|                     {{if $info.Paused}} | ||||
|                         <span class="paused">(Paused)</span> | ||||
|                     {{end}} | ||||
|                 </div> | ||||
|             </li> | ||||
|         {{else}} | ||||
|             <li>No active jobs</li> | ||||
|         {{end}} | ||||
|     </ul> | ||||
|     <button id="clear-completed" onclick="clearCompleted()">Clear Completed Jobs</button> | ||||
|     <script> | ||||
|         function clearCompleted() { | ||||
|             fetch('/clear-completed', { method: 'POST' }) | ||||
|                 .then(response => response.json()) | ||||
|                 .then(data => { | ||||
|                     if (data.success) { | ||||
|                         location.reload(); | ||||
|                     } else { | ||||
|                         alert('Failed to clear completed jobs'); | ||||
|                     } | ||||
|                 }); | ||||
|         } | ||||
|     </script> | ||||
| </body> | ||||
| </html> | ||||
|   | ||||
| @@ -56,6 +56,46 @@ | ||||
|             margin-top: 10px; | ||||
|             word-wrap: break-word; | ||||
|         } | ||||
|         #abort-button { | ||||
|             background-color: #f44336; | ||||
|             color: white; | ||||
|             border: none; | ||||
|             padding: 10px 15px; | ||||
|             margin-top: 10px; | ||||
|             border-radius: 4px; | ||||
|             cursor: pointer; | ||||
|         } | ||||
|         #abort-button:hover { | ||||
|             background-color: #d32f2f; | ||||
|         } | ||||
|         #pause-button, #resume-button { | ||||
|             background-color: #4CAF50; | ||||
|             color: white; | ||||
|             border: none; | ||||
|             padding: 10px 15px; | ||||
|             margin-top: 10px; | ||||
|             border-radius: 4px; | ||||
|             cursor: pointer; | ||||
|         } | ||||
|         #pause-button:hover, #resume-button:hover { | ||||
|             background-color: #45a049; | ||||
|         } | ||||
|         #resume-button { | ||||
|             display: none; | ||||
|         } | ||||
|         #back-button { | ||||
|             background-color: #2196F3; | ||||
|             color: white; | ||||
|             border: none; | ||||
|             padding: 10px 15px; | ||||
|             margin-top: 10px; | ||||
|             border-radius: 4px; | ||||
|             cursor: pointer; | ||||
|             float: right; | ||||
|         } | ||||
|         #back-button:hover { | ||||
|             background-color: #1976D2; | ||||
|         } | ||||
|         @media (max-width: 600px) { | ||||
|             body { | ||||
|                 padding: 10px; | ||||
| @@ -84,7 +124,15 @@ | ||||
|         </div> | ||||
|         <div id="currentFile"></div> | ||||
|     </div> | ||||
|     <div> | ||||
|         <button id="abort-button" onclick="abortDownload()">Abort Download</button> | ||||
|         <button id="pause-button" onclick="pauseDownload()">Pause Download</button> | ||||
|         <button id="resume-button" onclick="resumeDownload()" style="display: none;">Resume Download</button> | ||||
|         <button id="back-button" onclick="window.location.href='/'">Back to Index</button> | ||||
|     </div> | ||||
|     <script> | ||||
|         let isPaused = false; | ||||
|  | ||||
|         function updateProgress() { | ||||
|             fetch('/progress?filename={{.Filename}}', { | ||||
|                 headers: { | ||||
| @@ -97,11 +145,64 @@ | ||||
|                     document.getElementById('progress-bar').style.width = progress + '%'; | ||||
|                     document.getElementById('progress-text').innerText = progress + '%'; | ||||
|                     document.getElementById('currentFile').innerText = 'Current file: ' + (data.CurrentFile || 'None'); | ||||
|                     if (progress < 100) { | ||||
|                      | ||||
|                     isPaused = data.Paused; | ||||
|                     updatePauseResumeButtons(); | ||||
|                      | ||||
|                     if (progress < 100 && !isPaused) { | ||||
|                         setTimeout(updateProgress, 1000); | ||||
|                     } | ||||
|                 }); | ||||
|         } | ||||
|  | ||||
|         function updatePauseResumeButtons() { | ||||
|             if (isPaused) { | ||||
|                 document.getElementById('pause-button').style.display = 'none'; | ||||
|                 document.getElementById('resume-button').style.display = 'inline-block'; | ||||
|             } else { | ||||
|                 document.getElementById('pause-button').style.display = 'inline-block'; | ||||
|                 document.getElementById('resume-button').style.display = 'none'; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         function abortDownload() { | ||||
|             fetch('/abort?filename={{.Filename}}', { method: 'POST' }) | ||||
|                 .then(response => { | ||||
|                     if (response.ok) { | ||||
|                         console.log('Abort signal sent. The download will stop soon.'); | ||||
|                     } else { | ||||
|                         alert('Failed to abort the download.'); | ||||
|                     } | ||||
|                 }); | ||||
|         } | ||||
|  | ||||
|         function pauseDownload() { | ||||
|             fetch('/pause?filename={{.Filename}}', { method: 'POST' }) | ||||
|                 .then(response => { | ||||
|                     if (response.ok) { | ||||
|                         console.log('Pause signal sent. The download will pause soon.'); | ||||
|                         isPaused = true; | ||||
|                         updatePauseResumeButtons(); | ||||
|                     } else { | ||||
|                         alert('Failed to pause the download.'); | ||||
|                     } | ||||
|                 }); | ||||
|         } | ||||
|  | ||||
|         function resumeDownload() { | ||||
|             fetch('/resume?filename={{.Filename}}', { method: 'POST' }) | ||||
|                 .then(response => { | ||||
|                     if (response.ok) { | ||||
|                         console.log('Resume signal sent. The download will resume soon.'); | ||||
|                         isPaused = false; | ||||
|                         updatePauseResumeButtons(); | ||||
|                         updateProgress(); | ||||
|                     } else { | ||||
|                         alert('Failed to resume the download.'); | ||||
|                     } | ||||
|                 }); | ||||
|         } | ||||
|  | ||||
|         updateProgress(); | ||||
|     </script> | ||||
| </body> | ||||
|   | ||||
							
								
								
									
										102
									
								
								templates/select
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										102
									
								
								templates/select
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,102 @@ | ||||
| <!DOCTYPE html> | ||||
| <html lang="en"> | ||||
| <head> | ||||
|     <meta charset="UTF-8"> | ||||
|     <meta name="viewport" content="width=device-width, initial-scale=1.0"> | ||||
|     <title>Select Items to Download</title> | ||||
|     <style> | ||||
|         body { | ||||
|             font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Helvetica, Arial, sans-serif; | ||||
|             background-color: #1e1e1e; | ||||
|             color: #d4d4d4; | ||||
|             line-height: 1.6; | ||||
|             padding: 20px; | ||||
|             max-width: 800px; | ||||
|             margin: 0 auto; | ||||
|             box-sizing: border-box; | ||||
|         } | ||||
|         h1 { | ||||
|             border-bottom: 1px solid #333; | ||||
|             padding-bottom: 10px; | ||||
|         } | ||||
|         .season { | ||||
|             margin-bottom: 20px; | ||||
|             background-color: #2d2d2d; | ||||
|             padding: 10px; | ||||
|             border-radius: 4px; | ||||
|         } | ||||
|         .season-title { | ||||
|             font-size: 1.2em; | ||||
|             font-weight: bold; | ||||
|             margin-bottom: 10px; | ||||
|             display: flex; | ||||
|             align-items: center; | ||||
|         } | ||||
|         .season-checkbox { | ||||
|             margin-right: 10px; | ||||
|         } | ||||
|         .item { | ||||
|             margin-left: 20px; | ||||
|         } | ||||
|         button, input[type="submit"] { | ||||
|             background-color: #4CAF50; | ||||
|             color: white; | ||||
|             border: none; | ||||
|             padding: 10px 15px; | ||||
|             margin: 5px; | ||||
|             border-radius: 4px; | ||||
|             cursor: pointer; | ||||
|         } | ||||
|         button:hover, input[type="submit"]:hover { | ||||
|             background-color: #45a049; | ||||
|         } | ||||
|     </style> | ||||
| </head> | ||||
| <body> | ||||
|     <h1>Select Items to Download</h1> | ||||
|     <form action="/process" method="post"> | ||||
|         <input type="hidden" name="filename" value="{{.Filename}}"> | ||||
|         {{range $season, $items := .Items}} | ||||
|             <div class="season"> | ||||
|                 <div class="season-title"> | ||||
|                     <input type="checkbox" class="season-checkbox" id="season-{{$season}}" checked onchange="toggleSeason('{{$season}}')"> | ||||
|                     <label for="season-{{$season}}">{{$season}}</label> | ||||
|                 </div> | ||||
|                 {{range $item := $items}} | ||||
|                     <div class="item"> | ||||
|                         <label> | ||||
|                             <input type="checkbox" name="items" value="{{$item.Filename}}" checked class="episode-{{$season}}"> | ||||
|                             {{$item.Filename}} | ||||
|                         </label> | ||||
|                     </div> | ||||
|                 {{end}} | ||||
|             </div> | ||||
|         {{end}} | ||||
|         <div> | ||||
|             <button type="button" onclick="selectAll(true)">Select All</button> | ||||
|             <button type="button" onclick="selectAll(false)">Select None</button> | ||||
|             <input type="submit" value="Start Download"> | ||||
|         </div> | ||||
|     </form> | ||||
|     <script> | ||||
|         function selectAll(checked) { | ||||
|             var checkboxes = document.getElementsByName('items'); | ||||
|             for (var i = 0; i < checkboxes.length; i++) { | ||||
|                 checkboxes[i].checked = checked; | ||||
|             } | ||||
|             var seasonCheckboxes = document.getElementsByClassName('season-checkbox'); | ||||
|             for (var i = 0; i < seasonCheckboxes.length; i++) { | ||||
|                 seasonCheckboxes[i].checked = checked; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         function toggleSeason(season) { | ||||
|             var seasonCheckbox = document.getElementById('season-' + season); | ||||
|             var episodeCheckboxes = document.getElementsByClassName('episode-' + season); | ||||
|             for (var i = 0; i < episodeCheckboxes.length; i++) { | ||||
|                 episodeCheckboxes[i].checked = seasonCheckbox.checked; | ||||
|             } | ||||
|         } | ||||
|     </script> | ||||
| </body> | ||||
| </html> | ||||
							
								
								
									
										164
									
								
								utils.go
									
									
									
									
									
								
							
							
						
						
									
										164
									
								
								utils.go
									
									
									
									
									
								
							| @@ -1,15 +1,34 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"net/url" | ||||
| 	"os" | ||||
| 	"os/exec" | ||||
| 	"path/filepath" | ||||
| 	"regexp" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"sync" | ||||
|  | ||||
| 	"github.com/beevik/etree" | ||||
| ) | ||||
|  | ||||
| type JobInfo struct { | ||||
| 	AbortChan  chan struct{} | ||||
| 	ResumeChan chan struct{} | ||||
| 	Cmd        *exec.Cmd | ||||
| 	Paused     bool | ||||
| 	TempDir    string | ||||
| } | ||||
|  | ||||
| var ( | ||||
| 	jobsMutex sync.Mutex | ||||
| 	jobs      = make(map[string]*JobInfo) | ||||
| ) | ||||
|  | ||||
| func sanitizeFilename(filename string) string { | ||||
| 	filename = regexp.MustCompile(`[<>:"/\\|?*]`).ReplaceAllString(filename, "_") | ||||
|  | ||||
| @@ -118,3 +137,148 @@ func findOrCreateSegmentTimeline(adaptationSet *etree.Element) *etree.Element { | ||||
| 	segmentTemplate := representation.CreateElement("SegmentTemplate") | ||||
| 	return segmentTemplate.CreateElement("SegmentTimeline") | ||||
| } | ||||
|  | ||||
| func parseInputFile(inputFile string) ([]Item, error) { | ||||
| 	jsonFile, err := os.Open(inputFile) | ||||
| 	if err != nil { | ||||
| 		return nil, fmt.Errorf("error opening file %s: %v", inputFile, err) | ||||
| 	} | ||||
| 	defer jsonFile.Close() | ||||
|  | ||||
| 	byteValue, err := io.ReadAll(jsonFile) | ||||
| 	if err != nil { | ||||
| 		return nil, fmt.Errorf("error reading file %s: %v", inputFile, err) | ||||
| 	} | ||||
|  | ||||
| 	byteValue = removeBOM(byteValue) | ||||
|  | ||||
| 	var items Items | ||||
| 	err = json.Unmarshal(byteValue, &items) | ||||
| 	if err != nil { | ||||
| 		return nil, fmt.Errorf("error unmarshaling JSON: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	return items.Items, nil | ||||
| } | ||||
|  | ||||
| func groupItemsBySeason(items []Item) map[string][]Item { | ||||
| 	grouped := make(map[string][]Item) | ||||
| 	for _, item := range items { | ||||
| 		metadata := parseMetadata(item.Metadata) | ||||
| 		if metadata.Type == "serie" { | ||||
| 			key := fmt.Sprintf("%s - %s", metadata.Title, metadata.Season) | ||||
| 			grouped[key] = append(grouped[key], item) | ||||
| 		} else { | ||||
| 			grouped["Movies"] = append(grouped["Movies"], item) | ||||
| 		} | ||||
| 	} | ||||
| 	return grouped | ||||
| } | ||||
|  | ||||
| func filterSelectedItems(items []Item, selectedItems []string) []Item { | ||||
| 	var filtered []Item | ||||
| 	for _, item := range items { | ||||
| 		for _, selected := range selectedItems { | ||||
| 			if item.Filename == selected { | ||||
| 				filtered = append(filtered, item) | ||||
| 				break | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return filtered | ||||
| } | ||||
|  | ||||
| func processItems(filename string, items []Item) error { | ||||
| 	jobsMutex.Lock() | ||||
| 	jobInfo := &JobInfo{ | ||||
| 		AbortChan:  make(chan struct{}), | ||||
| 		ResumeChan: make(chan struct{}), | ||||
| 	} | ||||
| 	jobs[filename] = jobInfo | ||||
| 	jobsMutex.Unlock() | ||||
|  | ||||
| 	defer func() { | ||||
| 		jobsMutex.Lock() | ||||
| 		delete(jobs, filename) | ||||
| 		jobsMutex.Unlock() | ||||
|  | ||||
| 		if jobInfo.TempDir != "" { | ||||
| 			os.RemoveAll(jobInfo.TempDir) | ||||
| 		} | ||||
| 	}() | ||||
|  | ||||
| 	for i := 0; i < len(items); i++ { | ||||
| 		select { | ||||
| 		case <-jobInfo.AbortChan: | ||||
| 			updateProgress(filename, 100, "Aborted") | ||||
| 			return fmt.Errorf("download aborted") | ||||
| 		default: | ||||
| 			if jobInfo.Paused { | ||||
| 				select { | ||||
| 				case <-jobInfo.ResumeChan: | ||||
| 					jobInfo.Paused = false | ||||
| 					fmt.Printf("Resuming download for %s\n", filename) | ||||
| 				case <-jobInfo.AbortChan: | ||||
| 					updateProgress(filename, 100, "Aborted") | ||||
| 					return fmt.Errorf("download aborted") | ||||
| 				} | ||||
| 			} | ||||
| 			updateProgress(filename, float64(i)/float64(len(items))*100, items[i].Filename) | ||||
| 			err := downloadFile(items[i], jobInfo) | ||||
| 			if err != nil { | ||||
| 				if err.Error() == "download paused" { | ||||
| 					removeCompletedEpisodes(filename, items[:i]) | ||||
| 					i-- | ||||
| 					continue | ||||
| 				} | ||||
| 				fmt.Printf("Error downloading file: %v\n", err) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	updateProgress(filename, 100, "") | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func removeCompletedEpisodes(filename string, completedItems []Item) error { | ||||
| 	inputFile := filepath.Join(uploadDir, filename) | ||||
| 	items, err := parseInputFile(inputFile) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("error parsing input file: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	remainingItems := make([]Item, 0) | ||||
| 	for _, item := range items { | ||||
| 		if !isItemCompleted(item, completedItems) || isLastCompletedItem(item, completedItems) { | ||||
| 			remainingItems = append(remainingItems, item) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	updatedItems := Items{Items: remainingItems} | ||||
| 	jsonData, err := json.MarshalIndent(updatedItems, "", "  ") | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("error marshaling updated items: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	err = os.WriteFile(inputFile, jsonData, 0644) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("error writing updated DRMD file: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func isItemCompleted(item Item, completedItems []Item) bool { | ||||
| 	for _, completedItem := range completedItems { | ||||
| 		if item.Filename == completedItem.Filename { | ||||
| 			return true | ||||
| 		} | ||||
| 	} | ||||
| 	return false | ||||
| } | ||||
|  | ||||
| func isLastCompletedItem(item Item, completedItems []Item) bool { | ||||
| 	if len(completedItems) == 0 { | ||||
| 		return false | ||||
| 	} | ||||
| 	return item.Filename == completedItems[len(completedItems)-1].Filename | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user