mirror of
				https://github.com/bestnite/bilinovel-downloader.git
				synced 2025-10-26 17:14:24 +00:00 
			
		
		
		
	Compare commits
	
		
			17 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| b0f8f31dcc | |||
| 6084386989 | |||
| f1320cb978 | |||
|  | 434d5f54bd | ||
| b8cd053b00 | |||
| 560cdfdec9 | |||
| ed5440f5fb | |||
| 26f82dd9ea | |||
| e9fbe5c5db | |||
| 75745b9431 | |||
| ca3fdf8980 | |||
| 042b383988 | |||
| c9a7853cef | |||
| b2130f60d5 | |||
| d80c6053ab | |||
| 0c746c984b | |||
| 9d1d3f0f17 | 
							
								
								
									
										25
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										25
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							| @@ -2,12 +2,31 @@ | |||||||
|   "version": "0.2.0", |   "version": "0.2.0", | ||||||
|   "configurations": [ |   "configurations": [ | ||||||
|     { |     { | ||||||
|       "name": "Debug download volume", |       "name": "novel", | ||||||
|       "type": "go", |       "type": "go", | ||||||
|       "request": "launch", |       "request": "launch", | ||||||
|       "mode": "auto", |       "mode": "auto", | ||||||
|       "program": "${workspaceFolder}", |       "program": "${workspaceFolder}", | ||||||
|       "args": ["download", "volume", "-n", "2013", "-v", "165880"] |       "args": [ | ||||||
|  |         "download", | ||||||
|  |         "-n", | ||||||
|  |         "2727", | ||||||
|  |         "--concurrency", | ||||||
|  |         "5" | ||||||
|  |       ] | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       "name": "volume", | ||||||
|  |       "type": "go", | ||||||
|  |       "request": "launch", | ||||||
|  |       "mode": "auto", | ||||||
|  |       "program": "${workspaceFolder}", | ||||||
|  |       "args": [ | ||||||
|  |         "download", | ||||||
|  |         "-n=2727", | ||||||
|  |         "-v=150098", | ||||||
|  |         "--headless=false" | ||||||
|  |       ] | ||||||
|     } |     } | ||||||
|   ] |   ] | ||||||
| } | } | ||||||
							
								
								
									
										34
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										34
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,24 +1,24 @@ | |||||||
| # BiliNovel Downloader | # Bilinovel Downloader | ||||||
|  |  | ||||||
| 这是一个用于下载和生成轻小说 EPUB 电子书的工具。 | 这是一个用于从 Bilinovel 下载和生成轻小说 EPUB 电子书的工具。 | ||||||
|  | 生成的 EPUB 文件完全符合 EPUB 标准,可以在 Calibre 检查中无错误通过。 | ||||||
|  |  | ||||||
| ## 功能特点 | ## 使用示例 | ||||||
|  |  | ||||||
| - 支持下载轻小说并转换为标准 EPUB 格式 | 1. 下载整本 `https://www.bilinovel.com/novel/2388.html` | ||||||
| - 自动处理图片和文本内容 |  | ||||||
| - 生成符合 EPUB 3.0 规范的电子书文件 |  | ||||||
| - 支持多章节内容的组织和管理 |  | ||||||
| - 保留原有插图和排版格式 |  | ||||||
|  |  | ||||||
| ## 使用说明 |    ```bash | ||||||
|  |    bilinovel-downloader download -n 2388 | ||||||
|  |    ``` | ||||||
|  |  | ||||||
| 1. 确保系统环境满足要求 | 2. 下载单卷 `https://www.bilinovel.com/novel/2388/vol_84522.html` | ||||||
| 2. 运行下载器获取小说内容 |  | ||||||
| 3. 程序会自动处理并生成标准格式的 EPUB 文件 |  | ||||||
| 4. 生成的电子书文件可以在任何支持 EPUB 3.0 的阅读器中打开 |  | ||||||
|  |  | ||||||
| ## 注意事项 |    ```bash | ||||||
|  |    bilinovel-downloader download -n 2388 -v 84522 | ||||||
|  |    ``` | ||||||
|  |  | ||||||
| - 生成的 EPUB 文件严格遵循 EPUB 3.0 规范 | 3. 对自动生成的 epub 格式不满意可以自行修改后使用命令打包 | ||||||
| - 建议使用支持 EPUB 3.0 的阅读器以获得最佳阅读体验 |  | ||||||
| - 请遵守相关法律法规,合理使用下载的内容 |    ```bash | ||||||
|  |    bilinovel-downloader pack -d <目录路径> | ||||||
|  |    ``` | ||||||
|   | |||||||
							
								
								
									
										205
									
								
								cmd/download.go
									
									
									
									
									
								
							
							
						
						
									
										205
									
								
								cmd/download.go
									
									
									
									
									
								
							| @@ -2,7 +2,15 @@ package cmd | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"bilinovel-downloader/downloader" | 	"bilinovel-downloader/downloader" | ||||||
|  | 	"bilinovel-downloader/downloader/bilinovel" | ||||||
|  | 	"bilinovel-downloader/epub" | ||||||
|  | 	"bilinovel-downloader/model" | ||||||
|  | 	"bilinovel-downloader/text" | ||||||
|  | 	"encoding/json" | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  | 	"log" | ||||||
|  | 	"os" | ||||||
|  | 	"path/filepath" | ||||||
|  |  | ||||||
| 	"github.com/spf13/cobra" | 	"github.com/spf13/cobra" | ||||||
| ) | ) | ||||||
| @@ -11,74 +19,173 @@ var downloadCmd = &cobra.Command{ | |||||||
| 	Use:   "download", | 	Use:   "download", | ||||||
| 	Short: "Download a novel or volume", | 	Short: "Download a novel or volume", | ||||||
| 	Long:  "Download a novel or volume", | 	Long:  "Download a novel or volume", | ||||||
|  | 	Run: func(cmd *cobra.Command, args []string) { | ||||||
|  | 		err := runDownloadNovel() | ||||||
|  | 		if err != nil { | ||||||
|  | 			log.Printf("failed to download novel: %v", err) | ||||||
|  | 		} | ||||||
|  | 	}, | ||||||
| } | } | ||||||
|  |  | ||||||
| var downloadNovelCmd = &cobra.Command{ | type downloadCmdArgs struct { | ||||||
| 	Use:   "novel", | 	NovelId     int `validate:"required"` | ||||||
| 	Short: "Download a novel, default download all volumes", | 	VolumeId    int `validate:"required"` | ||||||
| 	Long:  "Download a novel, default download all volumes", | 	outputPath  string | ||||||
| 	RunE:  runDownloadNovel, | 	outputType  string | ||||||
| } | 	headless    bool | ||||||
|  | 	concurrency int | ||||||
| var downloadVolumeCmd = &cobra.Command{ |  | ||||||
| 	Use:   "volume", |  | ||||||
| 	Short: "Download a volume", |  | ||||||
| 	Long:  "Download a volume", |  | ||||||
| 	RunE:  runDownloadVolume, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type downloadNovelArgs struct { |  | ||||||
| 	NovelId    int `validate:"required"` |  | ||||||
| 	outputPath string |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type downloadVolumeArgs struct { |  | ||||||
| 	NovelId    int `validate:"required"` |  | ||||||
| 	VolumeId   int `validate:"required"` |  | ||||||
| 	outputPath string |  | ||||||
| } | } | ||||||
|  |  | ||||||
| var ( | var ( | ||||||
| 	novelArgs  downloadNovelArgs | 	downloadArgs downloadCmdArgs | ||||||
| 	volumeArgs downloadVolumeArgs |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func init() { | func init() { | ||||||
| 	downloadNovelCmd.Flags().IntVarP(&novelArgs.NovelId, "novel-id", "n", 0, "novel id") | 	downloadCmd.Flags().IntVarP(&downloadArgs.NovelId, "novel-id", "n", 0, "novel id") | ||||||
| 	downloadNovelCmd.Flags().StringVarP(&novelArgs.outputPath, "output-path", "o", "./novels", "output path") | 	downloadCmd.Flags().IntVarP(&downloadArgs.VolumeId, "volume-id", "v", 0, "volume id") | ||||||
|  | 	downloadCmd.Flags().StringVarP(&downloadArgs.outputPath, "output-path", "o", "novels", "output path") | ||||||
| 	downloadVolumeCmd.Flags().IntVarP(&volumeArgs.NovelId, "novel-id", "n", 0, "novel id") | 	downloadCmd.Flags().StringVarP(&downloadArgs.outputType, "output-type", "t", "epub", "output type, epub or text") | ||||||
| 	downloadVolumeCmd.Flags().IntVarP(&volumeArgs.VolumeId, "volume-id", "v", 0, "volume id") | 	downloadCmd.Flags().BoolVar(&downloadArgs.headless, "headless", true, "headless mode") | ||||||
| 	downloadVolumeCmd.Flags().StringVarP(&volumeArgs.outputPath, "output-path", "o", "./novels", "output path") | 	downloadCmd.Flags().IntVar(&downloadArgs.concurrency, "concurrency", 3, "concurrency of downloading volumes") | ||||||
|  |  | ||||||
| 	downloadCmd.AddCommand(downloadNovelCmd) |  | ||||||
| 	downloadCmd.AddCommand(downloadVolumeCmd) |  | ||||||
| 	RootCmd.AddCommand(downloadCmd) | 	RootCmd.AddCommand(downloadCmd) | ||||||
| } | } | ||||||
|  |  | ||||||
| func runDownloadNovel(cmd *cobra.Command, args []string) error { | func runDownloadNovel() error { | ||||||
| 	if novelArgs.NovelId == 0 { | 	downloader, err := bilinovel.New(bilinovel.BilinovelNewOption{ | ||||||
|  | 		Headless:    downloadArgs.headless, | ||||||
|  | 		Concurrency: downloadArgs.concurrency, | ||||||
|  | 	}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("failed to create downloader: %v", err) | ||||||
|  | 	} | ||||||
|  | 	// 确保在函数结束时关闭资源 | ||||||
|  | 	defer func() { | ||||||
|  | 		if closeErr := downloader.Close(); closeErr != nil { | ||||||
|  | 			log.Printf("Failed to close downloader: %v", closeErr) | ||||||
|  | 		} | ||||||
|  | 	}() | ||||||
|  |  | ||||||
|  | 	if downloadArgs.NovelId == 0 { | ||||||
| 		return fmt.Errorf("novel id is required") | 		return fmt.Errorf("novel id is required") | ||||||
| 	} | 	} | ||||||
| 	err := downloader.DownloadNovel(novelArgs.NovelId, novelArgs.outputPath) |  | ||||||
| 	if err != nil { | 	if downloadArgs.VolumeId == 0 { | ||||||
| 		return fmt.Errorf("failed to download novel: %v", err) | 		// 下载整本小说 | ||||||
|  | 		err := downloadNovel(downloader, downloadArgs.NovelId) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("failed to get novel: %v", err) | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		// 下载单卷 | ||||||
|  | 		err = downloadVolume(downloader, downloadArgs.VolumeId) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("failed to download volume: %v", err) | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func runDownloadVolume(cmd *cobra.Command, args []string) error { | func downloadNovel(downloader downloader.Downloader, novelId int) error { | ||||||
| 	if volumeArgs.NovelId == 0 { | 	novelInfo, err := downloader.GetNovel(novelId, true, nil) | ||||||
| 		return fmt.Errorf("novel id is required") |  | ||||||
| 	} |  | ||||||
| 	if volumeArgs.VolumeId == 0 { |  | ||||||
| 		return fmt.Errorf("volume id is required") |  | ||||||
| 	} |  | ||||||
| 	err := downloader.DownloadVolume(volumeArgs.NovelId, volumeArgs.VolumeId, volumeArgs.outputPath) |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return fmt.Errorf("failed to download volume: %v", err) | 		return fmt.Errorf("failed to get novel info: %w", err) | ||||||
|  | 	} | ||||||
|  | 	skipVolumes := make([]int, 0) | ||||||
|  | 	for _, volume := range novelInfo.Volumes { | ||||||
|  | 		jsonPath := filepath.Join(downloadArgs.outputPath, fmt.Sprintf("volume-%d-%d.json", downloadArgs.NovelId, volume.Id)) | ||||||
|  | 		err = os.MkdirAll(filepath.Dir(jsonPath), 0755) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("failed to create directory: %v", err) | ||||||
|  | 		} | ||||||
|  | 		_, err = os.Stat(jsonPath) | ||||||
|  | 		if err == nil { | ||||||
|  | 			// 已经下载 | ||||||
|  | 			skipVolumes = append(skipVolumes, volume.Id) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	novel, err := downloader.GetNovel(novelId, false, skipVolumes) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("failed to download novel: %w", err) | ||||||
|  | 	} | ||||||
|  | 	for _, volume := range novel.Volumes { | ||||||
|  | 		jsonPath := filepath.Join(downloadArgs.outputPath, fmt.Sprintf("volume-%d-%d.json", downloadArgs.NovelId, volume.Id)) | ||||||
|  | 		err = os.MkdirAll(filepath.Dir(jsonPath), 0755) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("failed to create directory: %v", err) | ||||||
|  | 		} | ||||||
|  | 		jsonFile, err := os.Create(jsonPath) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("failed to create json file: %v", err) | ||||||
|  | 		} | ||||||
|  | 		err = json.NewEncoder(jsonFile).Encode(volume) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("failed to encode json file: %v", err) | ||||||
|  | 		} | ||||||
|  | 		switch downloadArgs.outputType { | ||||||
|  | 		case "epub": | ||||||
|  | 			err = epub.PackVolumeToEpub(volume, downloadArgs.outputPath, downloader.GetStyleCSS(), downloader.GetExtraFiles()) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return fmt.Errorf("failed to pack volume: %v", err) | ||||||
|  | 			} | ||||||
|  | 		case "text": | ||||||
|  | 			err = text.PackVolumeToText(volume, downloadArgs.outputPath) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return fmt.Errorf("failed to pack volume: %v", err) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func downloadVolume(downloader downloader.Downloader, volumeId int) error { | ||||||
|  | 	jsonPath := filepath.Join(downloadArgs.outputPath, fmt.Sprintf("volume-%d-%d.json", downloadArgs.NovelId, volumeId)) | ||||||
|  | 	err := os.MkdirAll(filepath.Dir(jsonPath), 0755) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("failed to create directory: %v", err) | ||||||
|  | 	} | ||||||
|  | 	_, err = os.Stat(jsonPath) | ||||||
|  | 	volume := &model.Volume{} | ||||||
|  | 	if err != nil { | ||||||
|  | 		if os.IsNotExist(err) { | ||||||
|  | 			volume, err = downloader.GetVolume(downloadArgs.NovelId, volumeId, false) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return fmt.Errorf("failed to get volume: %v", err) | ||||||
|  | 			} | ||||||
|  | 			jsonFile, err := os.Create(jsonPath) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return fmt.Errorf("failed to create json file: %v", err) | ||||||
|  | 			} | ||||||
|  | 			err = json.NewEncoder(jsonFile).Encode(volume) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return fmt.Errorf("failed to encode json file: %v", err) | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  | 			return fmt.Errorf("failed to get volume: %v", err) | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		jsonFile, err := os.Open(jsonPath) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("failed to open json file: %v", err) | ||||||
|  | 		} | ||||||
|  | 		defer jsonFile.Close() | ||||||
|  | 		err = json.NewDecoder(jsonFile).Decode(volume) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("failed to decode json file: %v", err) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	switch downloadArgs.outputType { | ||||||
|  | 	case "epub": | ||||||
|  | 		err = epub.PackVolumeToEpub(volume, downloadArgs.outputPath, downloader.GetStyleCSS(), downloader.GetExtraFiles()) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("failed to pack volume: %v", err) | ||||||
|  | 		} | ||||||
|  | 	case "text": | ||||||
|  | 		err = text.PackVolumeToText(volume, downloadArgs.outputPath) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("failed to pack volume: %v", err) | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| package cmd | package cmd | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"bilinovel-downloader/utils" | 	"bilinovel-downloader/epub" | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  |  | ||||||
| 	"github.com/spf13/cobra" | 	"github.com/spf13/cobra" | ||||||
| @@ -28,7 +28,7 @@ func init() { | |||||||
| } | } | ||||||
|  |  | ||||||
| func runPackage(cmd *cobra.Command, args []string) error { | func runPackage(cmd *cobra.Command, args []string) error { | ||||||
| 	err := utils.CreateEpub(pArgs.DirPath) | 	err := epub.PackEpub(pArgs.DirPath) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return fmt.Errorf("failed to create epub: %v", err) | 		return fmt.Errorf("failed to create epub: %v", err) | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -1,657 +0,0 @@ | |||||||
| package downloader |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"bilinovel-downloader/model" |  | ||||||
| 	"bilinovel-downloader/template" |  | ||||||
| 	"bilinovel-downloader/utils" |  | ||||||
| 	"bytes" |  | ||||||
| 	"context" |  | ||||||
| 	"encoding/json" |  | ||||||
| 	"fmt" |  | ||||||
| 	"log" |  | ||||||
| 	"net/http" |  | ||||||
| 	"os" |  | ||||||
| 	"path" |  | ||||||
| 	"path/filepath" |  | ||||||
| 	"regexp" |  | ||||||
| 	"strconv" |  | ||||||
| 	"strings" |  | ||||||
| 	"time" |  | ||||||
|  |  | ||||||
| 	"github.com/PuerkitoBio/goquery" |  | ||||||
| 	"github.com/google/uuid" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func GetNovel(novelId int) (*model.Novel, error) { |  | ||||||
| 	novelUrl := fmt.Sprintf("https://www.bilinovel.com/novel/%v.html", novelId) |  | ||||||
| 	resp, err := utils.Request().Get(novelUrl) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, fmt.Errorf("failed to get novel info: %v", err) |  | ||||||
| 	} |  | ||||||
| 	if resp.StatusCode() != http.StatusOK { |  | ||||||
| 		return nil, fmt.Errorf("failed to get novel info: %v", resp.Status()) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	doc, err := goquery.NewDocumentFromReader(bytes.NewReader(resp.Body())) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, fmt.Errorf("failed to parse html: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	novel := &model.Novel{} |  | ||||||
|  |  | ||||||
| 	novel.Title = strings.TrimSpace(doc.Find(".book-title").First().Text()) |  | ||||||
| 	novel.Description = strings.TrimSpace(doc.Find(".book-summary>content").First().Text()) |  | ||||||
| 	novel.Id = novelId |  | ||||||
|  |  | ||||||
| 	doc.Find(".authorname>a").Each(func(i int, s *goquery.Selection) { |  | ||||||
| 		novel.Authors = append(novel.Authors, strings.TrimSpace(s.Text())) |  | ||||||
| 	}) |  | ||||||
| 	doc.Find(".illname>a").Each(func(i int, s *goquery.Selection) { |  | ||||||
| 		novel.Authors = append(novel.Authors, strings.TrimSpace(s.Text())) |  | ||||||
| 	}) |  | ||||||
|  |  | ||||||
| 	volumes, err := getNovelVolumes(novelId) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, fmt.Errorf("failed to get novel volumes: %v", err) |  | ||||||
| 	} |  | ||||||
| 	novel.Volumes = volumes |  | ||||||
|  |  | ||||||
| 	return novel, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func GetVolume(novelId int, volumeId int) (*model.Volume, error) { |  | ||||||
| 	novelUrl := fmt.Sprintf("https://www.bilinovel.com/novel/%v/catalog", novelId) |  | ||||||
| 	resp, err := utils.Request().Get(novelUrl) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, fmt.Errorf("failed to get novel info: %v", err) |  | ||||||
| 	} |  | ||||||
| 	if resp.StatusCode() != http.StatusOK { |  | ||||||
| 		return nil, fmt.Errorf("failed to get novel info: %v", resp.Status()) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	doc, err := goquery.NewDocumentFromReader(bytes.NewReader(resp.Body())) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, fmt.Errorf("failed to parse html: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	seriesIdx := 0 |  | ||||||
| 	doc.Find("a.volume-cover-img").Each(func(i int, s *goquery.Selection) { |  | ||||||
| 		if s.AttrOr("href", "") == fmt.Sprintf("/novel/%v/vol_%v.html", novelId, volumeId) { |  | ||||||
| 			seriesIdx = i + 1 |  | ||||||
| 		} |  | ||||||
| 	}) |  | ||||||
|  |  | ||||||
| 	novelTitle := strings.TrimSpace(doc.Find(".book-title").First().Text()) |  | ||||||
|  |  | ||||||
| 	if seriesIdx == 0 { |  | ||||||
| 		return nil, fmt.Errorf("volume not found: %v", volumeId) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	volumeUrl := fmt.Sprintf("https://www.bilinovel.com/novel/%v/vol_%v.html", novelId, volumeId) |  | ||||||
| 	resp, err = utils.Request().Get(volumeUrl) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, fmt.Errorf("failed to get novel info: %v", err) |  | ||||||
| 	} |  | ||||||
| 	if resp.StatusCode() != http.StatusOK { |  | ||||||
| 		return nil, fmt.Errorf("failed to get novel info: %v", resp.Status()) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	doc, err = goquery.NewDocumentFromReader(bytes.NewReader(resp.Body())) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, fmt.Errorf("failed to parse html: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	volume := &model.Volume{} |  | ||||||
| 	volume.NovelId = novelId |  | ||||||
| 	volume.NovelTitle = novelTitle |  | ||||||
| 	volume.Id = volumeId |  | ||||||
| 	volume.SeriesIdx = seriesIdx |  | ||||||
| 	volume.Title = strings.TrimSpace(doc.Find(".book-title").First().Text()) |  | ||||||
| 	volume.Description = strings.TrimSpace(doc.Find(".book-summary>content").First().Text()) |  | ||||||
| 	volume.Cover = doc.Find(".book-cover").First().AttrOr("src", "") |  | ||||||
| 	volume.Url = volumeUrl |  | ||||||
| 	volume.Chapters = make([]*model.Chapter, 0) |  | ||||||
|  |  | ||||||
| 	doc.Find(".authorname>a").Each(func(i int, s *goquery.Selection) { |  | ||||||
| 		volume.Authors = append(volume.Authors, strings.TrimSpace(s.Text())) |  | ||||||
| 	}) |  | ||||||
| 	doc.Find(".illname>a").Each(func(i int, s *goquery.Selection) { |  | ||||||
| 		volume.Authors = append(volume.Authors, strings.TrimSpace(s.Text())) |  | ||||||
| 	}) |  | ||||||
|  |  | ||||||
| 	doc.Find(".chapter-li.jsChapter").Each(func(i int, s *goquery.Selection) { |  | ||||||
| 		volume.Chapters = append(volume.Chapters, &model.Chapter{ |  | ||||||
| 			Title: s.Find("a").Text(), |  | ||||||
| 			Url:   fmt.Sprintf("https://www.bilinovel.com%v", s.Find("a").AttrOr("href", "")), |  | ||||||
| 		}) |  | ||||||
| 	}) |  | ||||||
|  |  | ||||||
| 	return volume, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func getNovelVolumes(novelId int) ([]*model.Volume, error) { |  | ||||||
| 	catelogUrl := fmt.Sprintf("https://www.bilinovel.com/novel/%v/catalog", novelId) |  | ||||||
| 	resp, err := utils.Request().Get(catelogUrl) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, fmt.Errorf("failed to get catelog: %v", err) |  | ||||||
| 	} |  | ||||||
| 	if resp.StatusCode() != http.StatusOK { |  | ||||||
| 		return nil, fmt.Errorf("failed to get catelog: %v", resp.Status()) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	doc, err := goquery.NewDocumentFromReader(bytes.NewReader(resp.Body())) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, fmt.Errorf("failed to parse html: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	volumeRegexp := regexp.MustCompile(fmt.Sprintf(`/novel/%v/vol_(\d+).html`, novelId)) |  | ||||||
|  |  | ||||||
| 	volumeIds := make([]string, 0) |  | ||||||
| 	doc.Find("a.volume-cover-img").Each(func(i int, s *goquery.Selection) { |  | ||||||
| 		link := s.AttrOr("href", "") |  | ||||||
| 		matches := volumeRegexp.FindStringSubmatch(link) |  | ||||||
| 		if len(matches) > 0 { |  | ||||||
| 			volumeIds = append(volumeIds, matches[1]) |  | ||||||
| 		} |  | ||||||
| 	}) |  | ||||||
|  |  | ||||||
| 	volumes := make([]*model.Volume, 0) |  | ||||||
| 	for i, volumeIdStr := range volumeIds { |  | ||||||
| 		volumeId, err := strconv.Atoi(volumeIdStr) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return nil, fmt.Errorf("failed to convert volume id: %v", err) |  | ||||||
| 		} |  | ||||||
| 		volume, err := GetVolume(novelId, volumeId) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return nil, fmt.Errorf("failed to get volume info: %v", err) |  | ||||||
| 		} |  | ||||||
| 		volume.SeriesIdx = i |  | ||||||
| 		volumes = append(volumes, volume) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return volumes, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DownloadNovel(novelId int, outputPath string) error { |  | ||||||
| 	log.Printf("Downloading Novel: %v", novelId) |  | ||||||
|  |  | ||||||
| 	novel, err := GetNovel(novelId) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to get novel info: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	outputPath = filepath.Join(outputPath, utils.CleanDirName(novel.Title)) |  | ||||||
| 	err = os.MkdirAll(outputPath, 0755) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to create output directory: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	for _, volume := range novel.Volumes { |  | ||||||
| 		err := downloadVolume(volume, outputPath) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return fmt.Errorf("failed to download volume: %v", err) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DownloadVolume(novelId, volumeId int, outputPath string) error { |  | ||||||
| 	volume, err := GetVolume(novelId, volumeId) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to get volume info: %v", err) |  | ||||||
| 	} |  | ||||||
| 	err = downloadVolume(volume, outputPath) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to download volume: %v", err) |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func downloadVolume(volume *model.Volume, outputPath string) error { |  | ||||||
| 	log.Printf("Downloading Volume: %s", volume.Title) |  | ||||||
| 	outputPath = filepath.Join(outputPath, utils.CleanDirName(volume.Title)) |  | ||||||
| 	err := os.MkdirAll(outputPath, 0755) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to create output directory: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	_, err = os.Stat(filepath.Join(outputPath, "volume.json")) |  | ||||||
| 	if os.IsNotExist(err) { |  | ||||||
| 		for idx, chapter := range volume.Chapters { |  | ||||||
| 			err := DownloadChapter(idx, chapter, outputPath) |  | ||||||
| 			if err != nil { |  | ||||||
| 				return fmt.Errorf("failed to download chapter: %v", err) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} else { |  | ||||||
| 		jsonBytes, err := os.ReadFile(filepath.Join(outputPath, "volume.json")) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return fmt.Errorf("failed to read volume: %v", err) |  | ||||||
| 		} |  | ||||||
| 		err = json.Unmarshal(jsonBytes, volume) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return fmt.Errorf("failed to unmarshal volume: %v", err) |  | ||||||
| 		} |  | ||||||
| 		for idx, chapter := range volume.Chapters { |  | ||||||
| 			file, err := os.Create(filepath.Join(outputPath, fmt.Sprintf("OEBPS/Text/chapter-%03v.xhtml", idx+1))) |  | ||||||
| 			if err != nil { |  | ||||||
| 				return fmt.Errorf("failed to create chapter file: %v", err) |  | ||||||
| 			} |  | ||||||
| 			err = template.ContentXHTML(chapter).Render(context.Background(), file) |  | ||||||
| 			if err != nil { |  | ||||||
| 				return fmt.Errorf("failed to render text file: %v", err) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	for i := range volume.Chapters { |  | ||||||
| 		volume.Chapters[i].ImageFullPaths = utils.Unique(volume.Chapters[i].ImageFullPaths) |  | ||||||
| 		volume.Chapters[i].ImageOEBPSPaths = utils.Unique(volume.Chapters[i].ImageOEBPSPaths) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	jsonBytes, err := json.Marshal(volume) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to marshal volume: %v", err) |  | ||||||
| 	} |  | ||||||
| 	err = os.WriteFile(filepath.Join(outputPath, "volume.json"), jsonBytes, 0644) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to write volume: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	coverPath := filepath.Join(outputPath, "OEBPS/Images/cover.jpg") |  | ||||||
| 	err = os.MkdirAll(path.Dir(coverPath), 0755) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to create cover directory: %v", err) |  | ||||||
| 	} |  | ||||||
| 	err = DownloadImg(volume.Cover, coverPath) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to download cover: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	coverXHTMLPath := filepath.Join(outputPath, "OEBPS/Text/cover.xhtml") |  | ||||||
| 	err = os.MkdirAll(path.Dir(coverXHTMLPath), 0755) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to create cover directory: %v", err) |  | ||||||
| 	} |  | ||||||
| 	file, err := os.Create(coverXHTMLPath) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to create cover file: %v", err) |  | ||||||
| 	} |  | ||||||
| 	err = template.ContentXHTML(&model.Chapter{ |  | ||||||
| 		Title:   "封面", |  | ||||||
| 		Content: fmt.Sprintf(`<img src="../Images/cover%s" />`, path.Ext(volume.Cover)), |  | ||||||
| 	}).Render(context.Background(), file) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to render cover: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	contentsXHTMLPath := filepath.Join(outputPath, "OEBPS/Text/contents.xhtml") |  | ||||||
| 	err = os.MkdirAll(path.Dir(contentsXHTMLPath), 0755) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to create contents directory: %v", err) |  | ||||||
| 	} |  | ||||||
| 	file, err = os.Create(contentsXHTMLPath) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to create contents file: %v", err) |  | ||||||
| 	} |  | ||||||
| 	contents := strings.Builder{} |  | ||||||
| 	contents.WriteString(`<nav epub:type="toc" id="toc">`) |  | ||||||
| 	contents.WriteString(`<ol>`) |  | ||||||
| 	for _, chapter := range volume.Chapters { |  | ||||||
| 		contents.WriteString(fmt.Sprintf(`<li><a href="%s">%s</a></li>`, strings.TrimPrefix(chapter.TextOEBPSPath, "Text/"), chapter.Title)) |  | ||||||
| 	} |  | ||||||
| 	contents.WriteString(`</ol>`) |  | ||||||
| 	contents.WriteString(`</nav>`) |  | ||||||
| 	err = template.ContentXHTML(&model.Chapter{ |  | ||||||
| 		Title:   "目录", |  | ||||||
| 		Content: contents.String(), |  | ||||||
| 	}).Render(context.Background(), file) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to render contents: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	err = CreateContainerXML(outputPath) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to create container xml: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	u, err := uuid.NewV7() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to generate uuid: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	err = CreateContentOPF(outputPath, u.String(), volume) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to create content opf: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	err = CreateTocNCX(outputPath, u.String(), volume) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to create toc ncx: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	err = utils.CreateEpub(outputPath) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to create epub: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DownloadChapter(chapterIdx int, chapter *model.Chapter, outputPath string) error { |  | ||||||
| 	chapter.TextFullPath = filepath.Join(outputPath, fmt.Sprintf("OEBPS/Text/chapter-%03v.xhtml", chapterIdx+1)) |  | ||||||
| 	chapter.TextOEBPSPath = fmt.Sprintf("Text/chapter-%03v.xhtml", chapterIdx+1) |  | ||||||
| 	err := os.MkdirAll(path.Dir(chapter.TextFullPath), 0755) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to create text directory: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	page := 1 |  | ||||||
| 	for { |  | ||||||
| 		hasNext, err := downloadChapterByPage(page, chapterIdx, chapter, outputPath) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return fmt.Errorf("failed to download chapter: %v", err) |  | ||||||
| 		} |  | ||||||
| 		if !hasNext { |  | ||||||
| 			break |  | ||||||
| 		} |  | ||||||
| 		page++ |  | ||||||
| 		time.Sleep(time.Second) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	file, err := os.Create(chapter.TextFullPath) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to create text file: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	err = template.ContentXHTML(chapter).Render(context.Background(), file) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to render text file: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func downloadChapterByPage(page, chapterIdx int, chapter *model.Chapter, outputPath string) (bool, error) { |  | ||||||
| 	Url := strings.TrimSuffix(chapter.Url, ".html") + fmt.Sprintf("_%v.html", page) |  | ||||||
| 	log.Printf("Downloading Chapter: %s", Url) |  | ||||||
|  |  | ||||||
| 	hasNext := false |  | ||||||
| 	headers := map[string]string{ |  | ||||||
| 		"Accept":          "text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.7", |  | ||||||
| 		"Accept-Language": "zh-CN,zh;q=0.9,en-GB;q=0.8,en;q=0.7,zh-TW;q=0.6", |  | ||||||
| 		"Cookie":          "night=1;", |  | ||||||
| 	} |  | ||||||
| 	resp, err := utils.Request().SetHeaders(headers).Get(Url) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return hasNext, err |  | ||||||
| 	} |  | ||||||
| 	if resp.StatusCode() != http.StatusOK { |  | ||||||
| 		return hasNext, fmt.Errorf("failed to get chapter: %v", resp.Status()) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if strings.Contains(resp.String(), `<a onclick="window.location.href = ReadParams.url_next;">下一頁</a>`) { |  | ||||||
| 		hasNext = true |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	doc, err := goquery.NewDocumentFromReader(bytes.NewReader(resp.Body())) |  | ||||||
| 	if err != nil { |  | ||||||
| 		fmt.Println(err) |  | ||||||
| 		return hasNext, err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	imgSavePath := fmt.Sprintf("OEBPS/Images/chapter-%03v", chapterIdx+1) |  | ||||||
|  |  | ||||||
| 	content := doc.Find("#acontent").First() |  | ||||||
| 	content.Find(".cgo").Remove() |  | ||||||
| 	content.Find("center").Remove() |  | ||||||
| 	content.Find(".google-auto-placed").Remove() |  | ||||||
|  |  | ||||||
| 	content.Find("img").Each(func(i int, s *goquery.Selection) { |  | ||||||
| 		if err != nil { |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 		imgUrl := s.AttrOr("data-src", "") |  | ||||||
| 		if imgUrl == "" { |  | ||||||
| 			imgUrl = s.AttrOr("src", "") |  | ||||||
| 			if imgUrl == "" { |  | ||||||
| 				return |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		fileName := filepath.Join(imgSavePath, fmt.Sprintf("%03v%s", i+1, path.Ext(imgUrl))) |  | ||||||
| 		err = DownloadImg(imgUrl, filepath.Join(outputPath, fileName)) |  | ||||||
| 		if err == nil { |  | ||||||
| 			s.SetAttr("src", "../"+strings.TrimPrefix(fileName, "OEBPS/")) |  | ||||||
| 			s.RemoveAttr("class") |  | ||||||
| 			s.RemoveAttr("data-src") |  | ||||||
| 			if s.AttrOr("alt", "") == "" { |  | ||||||
| 				s.SetAttr("alt", fmt.Sprintf("image-%03d", i+1)) |  | ||||||
| 			} |  | ||||||
| 			chapter.ImageFullPaths = append(chapter.ImageFullPaths, filepath.Join(outputPath, fileName)) |  | ||||||
| 			chapter.ImageOEBPSPaths = append(chapter.ImageOEBPSPaths, strings.TrimPrefix(fileName, "OEBPS/")) |  | ||||||
| 		} |  | ||||||
| 	}) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return false, fmt.Errorf("failed to download img: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	html, err := content.Html() |  | ||||||
| 	if err != nil { |  | ||||||
| 		return false, fmt.Errorf("failed to get html: %v", err) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	chapter.Content += strings.TrimSpace(html) |  | ||||||
|  |  | ||||||
| 	return hasNext, nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func DownloadImg(url string, fileName string) error { |  | ||||||
| 	_, err := os.Stat(fileName) |  | ||||||
| 	if !os.IsNotExist(err) { |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	log.Printf("Downloading Image: %s", url) |  | ||||||
| 	dir := filepath.Dir(fileName) |  | ||||||
| 	err = os.MkdirAll(dir, 0755) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	resp, err := utils.Request().SetHeader("Referer", "https://www.bilinovel.com").Get(url) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	err = os.WriteFile(fileName, resp.Body(), 0644) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func CreateContainerXML(dirPath string) error { |  | ||||||
| 	containerPath := filepath.Join(dirPath, "META-INF/container.xml") |  | ||||||
| 	err := os.MkdirAll(path.Dir(containerPath), 0755) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to create container directory: %v", err) |  | ||||||
| 	} |  | ||||||
| 	file, err := os.Create(containerPath) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to create container file: %v", err) |  | ||||||
| 	} |  | ||||||
| 	err = template.ContainerXML().Render(context.Background(), file) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to render container: %v", err) |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func CreateContentOPF(dirPath string, uuid string, volume *model.Volume) error { |  | ||||||
| 	creators := make([]model.DCCreator, 0) |  | ||||||
| 	for _, author := range volume.Authors { |  | ||||||
| 		creators = append(creators, model.DCCreator{ |  | ||||||
| 			Value: author, |  | ||||||
| 		}) |  | ||||||
| 	} |  | ||||||
| 	dc := &model.DublinCoreMetadata{ |  | ||||||
| 		Titles: []model.DCTitle{ |  | ||||||
| 			{ |  | ||||||
| 				Value: volume.Title, |  | ||||||
| 			}, |  | ||||||
| 		}, |  | ||||||
| 		Identifiers: []model.DCIdentifier{ |  | ||||||
| 			{ |  | ||||||
| 				Value: fmt.Sprintf("urn:uuid:%s", uuid), |  | ||||||
| 				ID:    "book-id", |  | ||||||
| 				// Scheme: "UUID", |  | ||||||
| 			}, |  | ||||||
| 		}, |  | ||||||
| 		Languages: []model.DCLanguage{ |  | ||||||
| 			{ |  | ||||||
| 				Value: "zh-TW", |  | ||||||
| 			}, |  | ||||||
| 		}, |  | ||||||
| 		Descriptions: []model.DCDescription{ |  | ||||||
| 			{ |  | ||||||
| 				Value: volume.Description, |  | ||||||
| 			}, |  | ||||||
| 		}, |  | ||||||
| 		Creators: creators, |  | ||||||
| 		Metas: []model.DublinCoreMeta{ |  | ||||||
| 			{ |  | ||||||
| 				Name:    "cover", |  | ||||||
| 				Content: fmt.Sprintf("Images/cover%s", path.Ext(volume.Cover)), |  | ||||||
| 			}, |  | ||||||
| 			{ |  | ||||||
| 				Property: "dcterms:modified", |  | ||||||
| 				Value:    time.Now().UTC().Format("2006-01-02T15:04:05Z"), |  | ||||||
| 			}, |  | ||||||
| 			{ |  | ||||||
| 				Name:    "calibre:series", |  | ||||||
| 				Content: volume.NovelTitle, |  | ||||||
| 			}, |  | ||||||
| 			{ |  | ||||||
| 				Name:    "calibre:series_index", |  | ||||||
| 				Content: strconv.Itoa(volume.SeriesIdx), |  | ||||||
| 			}, |  | ||||||
| 		}, |  | ||||||
| 	} |  | ||||||
| 	manifest := &model.Manifest{ |  | ||||||
| 		Items: make([]model.ManifestItem, 0), |  | ||||||
| 	} |  | ||||||
| 	manifest.Items = append(manifest.Items, model.ManifestItem{ |  | ||||||
| 		ID:    "ncx", |  | ||||||
| 		Link:  "toc.ncx", |  | ||||||
| 		Media: "application/x-dtbncx+xml", |  | ||||||
| 	}) |  | ||||||
| 	manifest.Items = append(manifest.Items, model.ManifestItem{ |  | ||||||
| 		ID:    "cover", |  | ||||||
| 		Link:  "Text/cover.xhtml", |  | ||||||
| 		Media: "application/xhtml+xml", |  | ||||||
| 	}) |  | ||||||
| 	manifest.Items = append(manifest.Items, model.ManifestItem{ |  | ||||||
| 		ID:         "contents", |  | ||||||
| 		Link:       "Text/contents.xhtml", |  | ||||||
| 		Media:      "application/xhtml+xml", |  | ||||||
| 		Properties: "nav", |  | ||||||
| 	}) |  | ||||||
| 	manifest.Items = append(manifest.Items, model.ManifestItem{ |  | ||||||
| 		ID:    "images-cover", |  | ||||||
| 		Link:  fmt.Sprintf("Images/cover%s", path.Ext(volume.Cover)), |  | ||||||
| 		Media: fmt.Sprintf("image/%s", strings.ReplaceAll(strings.TrimPrefix(path.Ext(volume.Cover), "."), "jpg", "jpeg")), |  | ||||||
| 	}) |  | ||||||
| 	for _, chapter := range volume.Chapters { |  | ||||||
| 		manifest.Items = append(manifest.Items, model.ManifestItem{ |  | ||||||
| 			ID:    path.Base(chapter.TextOEBPSPath), |  | ||||||
| 			Link:  chapter.TextOEBPSPath, |  | ||||||
| 			Media: "application/xhtml+xml", |  | ||||||
| 		}) |  | ||||||
| 		for _, image := range chapter.ImageOEBPSPaths { |  | ||||||
| 			item := model.ManifestItem{ |  | ||||||
| 				ID:   strings.Join(strings.Split(strings.ToLower(image), string(filepath.Separator)), "-"), |  | ||||||
| 				Link: image, |  | ||||||
| 			} |  | ||||||
| 			item.Media = fmt.Sprintf("image/%s", strings.ReplaceAll(strings.TrimPrefix(path.Ext(volume.Cover), "."), "jpg", "jpeg")) |  | ||||||
| 			manifest.Items = append(manifest.Items, item) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	manifest.Items = append(manifest.Items, model.ManifestItem{ |  | ||||||
| 		ID:    "style", |  | ||||||
| 		Link:  "Styles/style.css", |  | ||||||
| 		Media: "text/css", |  | ||||||
| 	}) |  | ||||||
|  |  | ||||||
| 	spine := &model.Spine{ |  | ||||||
| 		Items: make([]model.SpineItem, 0), |  | ||||||
| 	} |  | ||||||
| 	for _, item := range manifest.Items { |  | ||||||
| 		if filepath.Ext(item.Link) == ".xhtml" { |  | ||||||
| 			spine.Items = append(spine.Items, model.SpineItem{ |  | ||||||
| 				IDref: item.ID, |  | ||||||
| 			}) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	contentOPFPath := filepath.Join(dirPath, "OEBPS/content.opf") |  | ||||||
| 	err := os.MkdirAll(path.Dir(contentOPFPath), 0755) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to create content directory: %v", err) |  | ||||||
| 	} |  | ||||||
| 	file, err := os.Create(contentOPFPath) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to create content file: %v", err) |  | ||||||
| 	} |  | ||||||
| 	err = template.ContentOPF("book-id", dc, manifest, spine, nil).Render(context.Background(), file) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to render content: %v", err) |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func CreateTocNCX(dirPath string, uuid string, volume *model.Volume) error { |  | ||||||
| 	navMap := &model.NavMap{Points: make([]*model.NavPoint, 0)} |  | ||||||
| 	navMap.Points = append(navMap.Points, &model.NavPoint{ |  | ||||||
| 		Id:        "cover", |  | ||||||
| 		PlayOrder: 1, |  | ||||||
| 		Label:     "封面", |  | ||||||
| 		Content:   model.NavPointContent{Src: "Text/cover.xhtml"}, |  | ||||||
| 	}) |  | ||||||
| 	navMap.Points = append(navMap.Points, &model.NavPoint{ |  | ||||||
| 		Id:        "contents", |  | ||||||
| 		PlayOrder: 2, |  | ||||||
| 		Label:     "目录", |  | ||||||
| 		Content:   model.NavPointContent{Src: "Text/contents.xhtml"}, |  | ||||||
| 	}) |  | ||||||
| 	for idx, chapter := range volume.Chapters { |  | ||||||
| 		navMap.Points = append(navMap.Points, &model.NavPoint{ |  | ||||||
| 			Id:        fmt.Sprintf("chapter-%03v", idx+1), |  | ||||||
| 			PlayOrder: len(navMap.Points) + 1, |  | ||||||
| 			Label:     chapter.Title, |  | ||||||
| 			Content:   model.NavPointContent{Src: chapter.TextOEBPSPath}, |  | ||||||
| 		}) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	head := &model.TocNCXHead{ |  | ||||||
| 		Meta: []model.TocNCXHeadMeta{ |  | ||||||
| 			{Name: "dtb:uid", Content: fmt.Sprintf("urn:uuid:%s", uuid)}, |  | ||||||
| 		}, |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	ncxPath := filepath.Join(dirPath, "OEBPS/toc.ncx") |  | ||||||
| 	err := os.MkdirAll(path.Dir(ncxPath), 0755) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to create toc directory: %v", err) |  | ||||||
| 	} |  | ||||||
| 	file, err := os.Create(ncxPath) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to create toc file: %v", err) |  | ||||||
| 	} |  | ||||||
| 	err = template.TocNCX(volume.Title, head, navMap).Render(context.Background(), file) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("failed to render toc: %v", err) |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
							
								
								
									
										
											BIN
										
									
								
								downloader/bilinovel/MI LANTING.ttf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								downloader/bilinovel/MI LANTING.ttf
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										573
									
								
								downloader/bilinovel/bilinovel.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										573
									
								
								downloader/bilinovel/bilinovel.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,573 @@ | |||||||
|  | package bilinovel | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"bilinovel-downloader/model" | ||||||
|  | 	"bilinovel-downloader/utils" | ||||||
|  | 	"bytes" | ||||||
|  | 	"crypto/sha256" | ||||||
|  | 	_ "embed" | ||||||
|  | 	"fmt" | ||||||
|  | 	"log" | ||||||
|  | 	"net/http" | ||||||
|  | 	"os" | ||||||
|  | 	"path" | ||||||
|  | 	"path/filepath" | ||||||
|  | 	"regexp" | ||||||
|  | 	"slices" | ||||||
|  | 	"strconv" | ||||||
|  | 	"strings" | ||||||
|  | 	"sync" | ||||||
|  |  | ||||||
|  | 	"github.com/PuerkitoBio/goquery" | ||||||
|  | 	mapper "github.com/bestnite/font-mapper" | ||||||
|  | 	"github.com/playwright-community/playwright-go" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | //go:embed read.ttf | ||||||
|  | var readTTF []byte | ||||||
|  |  | ||||||
|  | //go:embed "MI LANTING.ttf" | ||||||
|  | var miLantingTTF []byte | ||||||
|  |  | ||||||
|  | type Bilinovel struct { | ||||||
|  | 	fontMapper  *mapper.GlyphOutlineMapper | ||||||
|  | 	textOnly    bool | ||||||
|  | 	restyClient *utils.RestyClient | ||||||
|  |  | ||||||
|  | 	// 浏览器实例复用 | ||||||
|  | 	browser        playwright.Browser | ||||||
|  | 	browserContext playwright.BrowserContext | ||||||
|  | 	pages          map[string]playwright.Page | ||||||
|  | 	concurrency    int | ||||||
|  | 	concurrentChan chan any | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type BilinovelNewOption struct { | ||||||
|  | 	Headless    bool | ||||||
|  | 	Concurrency int | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func New(option BilinovelNewOption) (*Bilinovel, error) { | ||||||
|  | 	fontMapper, err := mapper.NewGlyphOutlineMapper(readTTF, miLantingTTF) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("failed to create font mapper: %v", err) | ||||||
|  | 	} | ||||||
|  | 	restyClient := utils.NewRestyClient(50) | ||||||
|  |  | ||||||
|  | 	b := &Bilinovel{ | ||||||
|  | 		fontMapper:     fontMapper, | ||||||
|  | 		textOnly:       false, | ||||||
|  | 		restyClient:    restyClient, | ||||||
|  | 		pages:          make(map[string]playwright.Page), | ||||||
|  | 		concurrency:    option.Concurrency, | ||||||
|  | 		concurrentChan: make(chan any, option.Concurrency), | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// 初始化浏览器实例 | ||||||
|  | 	err = b.initBrowser(option.Headless) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("failed to init browser: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return b, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bilinovel) SetTextOnly(textOnly bool) { | ||||||
|  | 	b.textOnly = textOnly | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bilinovel) GetExtraFiles() []model.ExtraFile { | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // initBrowser 初始化浏览器实例 | ||||||
|  | func (b *Bilinovel) initBrowser(headless bool) error { | ||||||
|  | 	pw, err := playwright.Run() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("could not start playwright: %w", err) | ||||||
|  | 	} | ||||||
|  | 	b.browser, err = pw.Chromium.Launch(playwright.BrowserTypeLaunchOptions{ | ||||||
|  | 		Headless: playwright.Bool(headless), | ||||||
|  | 	}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("could not launch browser: %w", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	b.browserContext, err = b.browser.NewContext() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("could not create browser context: %w", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	log.Println("Browser initialized successfully") | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Close 清理资源 | ||||||
|  | func (b *Bilinovel) Close() error { | ||||||
|  | 	if b.browser != nil { | ||||||
|  | 		if err := b.browser.Close(); err != nil { | ||||||
|  | 			log.Printf("could not close browser: %v", err) | ||||||
|  | 		} | ||||||
|  | 		b.browser = nil | ||||||
|  | 		b.browserContext = nil | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | //go:embed style.css | ||||||
|  | var styleCSS []byte | ||||||
|  |  | ||||||
|  | func (b *Bilinovel) GetStyleCSS() string { | ||||||
|  | 	return string(styleCSS) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bilinovel) GetNovel(novelId int, skipChapterContent bool, skipVolumes []int) (*model.Novel, error) { | ||||||
|  | 	log.Printf("Getting novel %v\n", novelId) | ||||||
|  |  | ||||||
|  | 	novelUrl := fmt.Sprintf("https://www.bilinovel.com/novel/%v.html", novelId) | ||||||
|  | 	resp, err := b.restyClient.R().Get(novelUrl) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("failed to get novel info: %w", err) | ||||||
|  | 	} | ||||||
|  | 	if resp.StatusCode() != http.StatusOK { | ||||||
|  | 		return nil, fmt.Errorf("failed to get novel info: %v", resp.Status()) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	doc, err := goquery.NewDocumentFromReader(bytes.NewReader(resp.Body())) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("failed to parse html: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	novel := &model.Novel{} | ||||||
|  |  | ||||||
|  | 	novel.Title = strings.TrimSpace(doc.Find(".book-title").First().Text()) | ||||||
|  | 	novel.Description = strings.TrimSpace(doc.Find(".book-summary>content").First().Text()) | ||||||
|  | 	novel.Id = novelId | ||||||
|  |  | ||||||
|  | 	doc.Find(".authorname>a").Each(func(i int, s *goquery.Selection) { | ||||||
|  | 		novel.Authors = append(novel.Authors, strings.TrimSpace(s.Text())) | ||||||
|  | 	}) | ||||||
|  | 	doc.Find(".illname>a").Each(func(i int, s *goquery.Selection) { | ||||||
|  | 		novel.Authors = append(novel.Authors, strings.TrimSpace(s.Text())) | ||||||
|  | 	}) | ||||||
|  |  | ||||||
|  | 	volumes, err := b.getAllVolumes(novelId, skipChapterContent, skipVolumes) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("failed to get novel volumes: %v", err) | ||||||
|  | 	} | ||||||
|  | 	novel.Volumes = volumes | ||||||
|  |  | ||||||
|  | 	return novel, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bilinovel) GetVolume(novelId int, volumeId int, skipChapterContent bool) (*model.Volume, error) { | ||||||
|  | 	log.Printf("Getting volume %v of novel %v\n", volumeId, novelId) | ||||||
|  |  | ||||||
|  | 	novelUrl := fmt.Sprintf("https://www.bilinovel.com/novel/%v/catalog", novelId) | ||||||
|  | 	resp, err := b.restyClient.R().Get(novelUrl) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("failed to get novel info: %w", err) | ||||||
|  | 	} | ||||||
|  | 	if resp.StatusCode() != http.StatusOK { | ||||||
|  | 		return nil, fmt.Errorf("failed to get novel info: %v", resp.Status()) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	doc, err := goquery.NewDocumentFromReader(bytes.NewReader(resp.Body())) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("failed to parse html: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	seriesIdx := 0 | ||||||
|  | 	doc.Find("a.volume-cover-img").Each(func(i int, s *goquery.Selection) { | ||||||
|  | 		if s.AttrOr("href", "") == fmt.Sprintf("/novel/%v/vol_%v.html", novelId, volumeId) { | ||||||
|  | 			seriesIdx = i + 1 | ||||||
|  | 		} | ||||||
|  | 	}) | ||||||
|  |  | ||||||
|  | 	novelTitle := strings.TrimSpace(doc.Find(".book-title").First().Text()) | ||||||
|  |  | ||||||
|  | 	if seriesIdx == 0 { | ||||||
|  | 		return nil, fmt.Errorf("volume not found: %v", volumeId) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	volumeUrl := fmt.Sprintf("https://www.bilinovel.com/novel/%v/vol_%v.html", novelId, volumeId) | ||||||
|  | 	resp, err = b.restyClient.R().Get(volumeUrl) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("failed to get novel info: %v", err) | ||||||
|  | 	} | ||||||
|  | 	if resp.StatusCode() != http.StatusOK { | ||||||
|  | 		return nil, fmt.Errorf("failed to get novel info: %v", resp.Status()) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	doc, err = goquery.NewDocumentFromReader(bytes.NewReader(resp.Body())) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("failed to parse html: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	volume := &model.Volume{} | ||||||
|  | 	volume.NovelId = novelId | ||||||
|  | 	volume.NovelTitle = novelTitle | ||||||
|  | 	volume.Id = volumeId | ||||||
|  | 	volume.SeriesIdx = seriesIdx | ||||||
|  | 	volume.Title = strings.TrimSpace(doc.Find(".book-title").First().Text()) | ||||||
|  | 	volume.Description = strings.TrimSpace(doc.Find(".book-summary>content").First().Text()) | ||||||
|  | 	volume.Url = volumeUrl | ||||||
|  | 	volume.Chapters = make([]*model.Chapter, 0) | ||||||
|  | 	volume.CoverUrl = doc.Find(".book-cover").First().AttrOr("src", "") | ||||||
|  | 	cover, err := b.getImg(volume.CoverUrl) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("failed to get cover: %v", err) | ||||||
|  | 	} | ||||||
|  | 	volume.Cover = cover | ||||||
|  |  | ||||||
|  | 	doc.Find(".authorname>a").Each(func(i int, s *goquery.Selection) { | ||||||
|  | 		volume.Authors = append(volume.Authors, strings.TrimSpace(s.Text())) | ||||||
|  | 	}) | ||||||
|  | 	doc.Find(".illname>a").Each(func(i int, s *goquery.Selection) { | ||||||
|  | 		volume.Authors = append(volume.Authors, strings.TrimSpace(s.Text())) | ||||||
|  | 	}) | ||||||
|  | 	doc.Find(".chapter-li.jsChapter").Each(func(i int, s *goquery.Selection) { | ||||||
|  | 		volume.Chapters = append(volume.Chapters, &model.Chapter{ | ||||||
|  | 			Title: s.Find("a").Text(), | ||||||
|  | 			Url:   fmt.Sprintf("https://www.bilinovel.com%v", s.Find("a").AttrOr("href", "")), | ||||||
|  | 		}) | ||||||
|  | 	}) | ||||||
|  |  | ||||||
|  | 	idRegexp := regexp.MustCompile(`/novel/(\d+)/(\d+).html`) | ||||||
|  |  | ||||||
|  | 	if !skipChapterContent { | ||||||
|  | 		for i := range volume.Chapters { | ||||||
|  | 			matches := idRegexp.FindStringSubmatch(volume.Chapters[i].Url) | ||||||
|  | 			if len(matches) > 0 { | ||||||
|  | 				chapterId, err := strconv.Atoi(matches[2]) | ||||||
|  | 				if err != nil { | ||||||
|  | 					return nil, fmt.Errorf("failed to convert chapter id: %v", err) | ||||||
|  | 				} | ||||||
|  | 				chapter, err := b.GetChapter(novelId, volumeId, chapterId) | ||||||
|  | 				if err != nil { | ||||||
|  | 					return nil, fmt.Errorf("failed to get chapter: %v", err) | ||||||
|  | 				} | ||||||
|  | 				chapter.Id = chapterId | ||||||
|  | 				volume.Chapters[i] = chapter | ||||||
|  | 			} else { | ||||||
|  | 				return nil, fmt.Errorf("failed to get chapter id: %v", volume.Chapters[i].Url) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return volume, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bilinovel) getAllVolumes(novelId int, skipChapterContent bool, skipVolumes []int) ([]*model.Volume, error) { | ||||||
|  | 	log.Printf("Getting all volumes of novel %v\n", novelId) | ||||||
|  |  | ||||||
|  | 	catelogUrl := fmt.Sprintf("https://www.bilinovel.com/novel/%v/catalog", novelId) | ||||||
|  | 	resp, err := b.restyClient.R().Get(catelogUrl) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("failed to get catelog: %v", err) | ||||||
|  | 	} | ||||||
|  | 	if resp.StatusCode() != http.StatusOK { | ||||||
|  | 		return nil, fmt.Errorf("failed to get catelog: %v", resp.Status()) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	doc, err := goquery.NewDocumentFromReader(bytes.NewReader(resp.Body())) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("failed to parse html: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	volumeRegexp := regexp.MustCompile(fmt.Sprintf(`/novel/%v/vol_(\d+).html`, novelId)) | ||||||
|  |  | ||||||
|  | 	volumeIds := make([]string, 0) | ||||||
|  | 	doc.Find("a.volume-cover-img").Each(func(i int, s *goquery.Selection) { | ||||||
|  | 		link := s.AttrOr("href", "") | ||||||
|  | 		matches := volumeRegexp.FindStringSubmatch(link) | ||||||
|  | 		if len(matches) > 0 { | ||||||
|  | 			volumeIds = append(volumeIds, matches[1]) | ||||||
|  | 		} | ||||||
|  | 	}) | ||||||
|  |  | ||||||
|  | 	volumes := make([]*model.Volume, len(volumeIds)) | ||||||
|  | 	var wg sync.WaitGroup | ||||||
|  | 	var mu sync.Mutex // 保护 volumes 写入的互斥锁 | ||||||
|  |  | ||||||
|  | 	for i, volumeIdStr := range volumeIds { | ||||||
|  | 		wg.Add(1) | ||||||
|  | 		b.concurrentChan <- struct{}{} // 获取一个并发槽 | ||||||
|  |  | ||||||
|  | 		go func(i int, volumeIdStr string) { | ||||||
|  | 			defer wg.Done() | ||||||
|  | 			defer func() { <-b.concurrentChan }() // 释放并发槽 | ||||||
|  |  | ||||||
|  | 			volumeId, err := strconv.Atoi(volumeIdStr) | ||||||
|  | 			if err != nil { | ||||||
|  | 				log.Printf("failed to convert volume id %s: %v", volumeIdStr, err) | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 			if slices.Contains(skipVolumes, volumeId) { | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 			volume, err := b.GetVolume(novelId, volumeId, skipChapterContent) | ||||||
|  | 			if err != nil { | ||||||
|  | 				log.Printf("failed to get volume info for novel %d, volume %d: %v", novelId, volumeId, err) | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 			volume.SeriesIdx = i | ||||||
|  |  | ||||||
|  | 			// 关闭浏览器标签页 | ||||||
|  | 			pwPageKey := fmt.Sprintf("%v-%v", novelId, volumeId) | ||||||
|  | 			if pwPage, ok := b.pages[pwPageKey]; ok { | ||||||
|  | 				_ = pwPage.Close() | ||||||
|  | 				delete(b.pages, pwPageKey) | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			mu.Lock() | ||||||
|  | 			volumes[i] = volume | ||||||
|  | 			mu.Unlock() | ||||||
|  | 		}(i, volumeIdStr) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	wg.Wait() | ||||||
|  |  | ||||||
|  | 	// 过滤掉获取失败的 nil volume | ||||||
|  | 	filteredVolumes := make([]*model.Volume, 0, len(volumes)) | ||||||
|  | 	for _, vol := range volumes { | ||||||
|  | 		if vol != nil { | ||||||
|  | 			filteredVolumes = append(filteredVolumes, vol) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return filteredVolumes, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bilinovel) GetChapter(novelId int, volumeId int, chapterId int) (*model.Chapter, error) { | ||||||
|  | 	log.Printf("Getting chapter %v of novel %v\n", chapterId, novelId) | ||||||
|  |  | ||||||
|  | 	pageNum := 1 | ||||||
|  | 	chapter := &model.Chapter{ | ||||||
|  | 		Id:       chapterId, | ||||||
|  | 		NovelId:  novelId, | ||||||
|  | 		VolumeId: volumeId, | ||||||
|  | 		Url:      fmt.Sprintf("https://www.bilinovel.com/novel/%v/%v.html", novelId, chapterId), | ||||||
|  | 	} | ||||||
|  | 	for { | ||||||
|  | 		pwPageKey := fmt.Sprintf("%v-%v", novelId, volumeId) | ||||||
|  | 		if _, ok := b.pages[pwPageKey]; !ok { | ||||||
|  | 			pwPage, err := b.browserContext.NewPage() | ||||||
|  | 			if err != nil { | ||||||
|  | 				return nil, fmt.Errorf("failed to create browser page: %w", err) | ||||||
|  | 			} | ||||||
|  | 			b.pages[pwPageKey] = pwPage | ||||||
|  | 		} | ||||||
|  | 		hasNext, err := b.getChapterByPage(b.pages[pwPageKey], chapter, pageNum) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return nil, fmt.Errorf("failed to download chapter: %w", err) | ||||||
|  | 		} | ||||||
|  | 		if !hasNext { | ||||||
|  | 			break | ||||||
|  | 		} | ||||||
|  | 		pageNum++ | ||||||
|  | 	} | ||||||
|  | 	return chapter, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bilinovel) getChapterByPage(pwPage playwright.Page, chapter *model.Chapter, pageNum int) (bool, error) { | ||||||
|  | 	log.Printf("Getting chapter %v by page %v\n", chapter.Id, pageNum) | ||||||
|  |  | ||||||
|  | 	Url := strings.TrimSuffix(chapter.Url, ".html") + fmt.Sprintf("_%v.html", pageNum) | ||||||
|  |  | ||||||
|  | 	hasNext := false | ||||||
|  | 	headers := map[string]string{ | ||||||
|  | 		"Accept":          "text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.7", | ||||||
|  | 		"Accept-Language": "zh-CN,zh;q=0.9,en-GB;q=0.8,en;q=0.7,zh-TW;q=0.6", | ||||||
|  | 		"Cookie":          "night=1;", | ||||||
|  | 	} | ||||||
|  | 	resp, err := b.restyClient.R().SetHeaders(headers).Get(Url) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return false, fmt.Errorf("failed to get chapter: %w", err) | ||||||
|  | 	} | ||||||
|  | 	if resp.StatusCode() != http.StatusOK { | ||||||
|  | 		return false, fmt.Errorf("failed to get chapter: %v", resp.Status()) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if strings.Contains(resp.String(), `<a onclick="window.location.href = ReadParams.url_next;">下一頁</a>`) { | ||||||
|  | 		hasNext = true | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	html := resp.Body() | ||||||
|  |  | ||||||
|  | 	// 解决乱序问题 | ||||||
|  | 	resortedHtml, err := b.processContentWithPlaywright(pwPage, string(html)) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return false, fmt.Errorf("failed to process html: %w", err) | ||||||
|  | 	} | ||||||
|  | 	doc, err := goquery.NewDocumentFromReader(strings.NewReader(resortedHtml)) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return false, fmt.Errorf("failed to parse html: %w", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if pageNum == 1 { | ||||||
|  | 		chapter.Title = doc.Find("#atitle").Text() | ||||||
|  | 	} | ||||||
|  | 	content := doc.Find("#acontent").First() | ||||||
|  | 	content.Find(".cgo").Remove() | ||||||
|  | 	content.Find("center").Remove() | ||||||
|  | 	content.Find(".google-auto-placed").Remove() | ||||||
|  |  | ||||||
|  | 	if strings.Contains(resp.String(), `font-family: "read"`) { | ||||||
|  | 		html, err := content.Find("p").Last().Html() | ||||||
|  | 		if err != nil { | ||||||
|  | 			return false, fmt.Errorf("failed to get html: %v", err) | ||||||
|  | 		} | ||||||
|  | 		builder := strings.Builder{} | ||||||
|  | 		for _, r := range html { | ||||||
|  | 			_, newRune, ok := b.fontMapper.MappingRune(r) | ||||||
|  | 			if ok { | ||||||
|  | 				builder.WriteRune(newRune) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		content.Find("p").Last().SetHtml(builder.String()) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if b.textOnly { | ||||||
|  | 		content.Find("img").Remove() | ||||||
|  | 	} else { | ||||||
|  | 		content.Find("img").Each(func(i int, s *goquery.Selection) { | ||||||
|  | 			imgUrl := s.AttrOr("data-src", "") | ||||||
|  | 			if imgUrl == "" { | ||||||
|  | 				imgUrl = s.AttrOr("src", "") | ||||||
|  | 				if imgUrl == "" { | ||||||
|  | 					return | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			imageHash := sha256.Sum256([]byte(imgUrl)) | ||||||
|  | 			imageFilename := fmt.Sprintf("%x%s", string(imageHash[:]), path.Ext(imgUrl)) | ||||||
|  | 			s.SetAttr("src", imageFilename) | ||||||
|  | 			s.SetAttr("alt", imgUrl) | ||||||
|  | 			s.RemoveAttr("class") | ||||||
|  | 			img, err := b.getImg(imgUrl) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 			if chapter.Content == nil { | ||||||
|  | 				chapter.Content = &model.ChaperContent{} | ||||||
|  | 			} | ||||||
|  | 			if chapter.Content.Images == nil { | ||||||
|  | 				chapter.Content.Images = make(map[string][]byte) | ||||||
|  | 			} | ||||||
|  | 			chapter.Content.Images[imageFilename] = img | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	doc.Find("*").Each(func(i int, s *goquery.Selection) { | ||||||
|  | 		if len(s.Nodes) > 0 && len(s.Nodes[0].Attr) > 0 { | ||||||
|  | 			// 遍历元素的所有属性 | ||||||
|  | 			for _, attr := range s.Nodes[0].Attr { | ||||||
|  | 				// 3. 检查属性名是否以 "data-k" 开头,且属性值是否为空 | ||||||
|  | 				if strings.HasPrefix(attr.Key, "data-k") { | ||||||
|  | 					// 4. 如果满足条件,就移除这个属性 | ||||||
|  | 					s.RemoveAttr(attr.Key) | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	}) | ||||||
|  |  | ||||||
|  | 	htmlStr, err := content.Html() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return false, fmt.Errorf("failed to get html: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if chapter.Content == nil { | ||||||
|  | 		chapter.Content = &model.ChaperContent{} | ||||||
|  | 	} | ||||||
|  | 	chapter.Content.Html += strings.TrimSpace(htmlStr) | ||||||
|  |  | ||||||
|  | 	return hasNext, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (b *Bilinovel) getImg(url string) ([]byte, error) { | ||||||
|  | 	log.Printf("Getting img %v\n", url) | ||||||
|  | 	resp, err := b.restyClient.R().SetHeader("Referer", "https://www.bilinovel.com").Get(url) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return resp.Body(), nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // processContentWithPlaywright 使用复用的浏览器实例处理内容 | ||||||
|  | func (b *Bilinovel) processContentWithPlaywright(page playwright.Page, htmlContent string) (string, error) { | ||||||
|  | 	tempFile, err := os.CreateTemp("", "bilinovel-temp-*.html") | ||||||
|  | 	if err != nil { | ||||||
|  | 		return "", fmt.Errorf("failed to create temp file: %w", err) | ||||||
|  | 	} | ||||||
|  | 	defer os.Remove(tempFile.Name()) | ||||||
|  |  | ||||||
|  | 	_, err = tempFile.WriteString(htmlContent) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return "", fmt.Errorf("failed to write temp file: %w", err) | ||||||
|  | 	} | ||||||
|  | 	tempFile.Close() | ||||||
|  | 	tempFilePath := tempFile.Name() | ||||||
|  |  | ||||||
|  | 	_, err = page.ExpectResponse(func(url string) bool { | ||||||
|  | 		return strings.Contains(url, "chapterlog.js") | ||||||
|  | 	}, func() error { | ||||||
|  | 		_, err = page.Goto("file://" + filepath.ToSlash(tempFilePath)) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("could not navigate to file: %w", err) | ||||||
|  | 		} | ||||||
|  | 		return nil | ||||||
|  | 	}, playwright.PageExpectResponseOptions{ | ||||||
|  | 		Timeout: playwright.Float(5000), | ||||||
|  | 	}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return "", fmt.Errorf("failed to wait for network request finish") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	err = page.Locator("#acontent").WaitFor(playwright.LocatorWaitForOptions{ | ||||||
|  | 		State: playwright.WaitForSelectorStateVisible, | ||||||
|  | 	}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return "", fmt.Errorf("could not wait for #acontent: %w", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// 遍历所有 #acontent 的子元素, 通过 window.getComputedStyle().display 检测是否是 none, 如果是 none 则从页面删除这个元素 | ||||||
|  | 	result, err := page.Evaluate(` | ||||||
|  | 		(function() { | ||||||
|  | 			const acontent = document.getElementById('acontent'); | ||||||
|  | 			if (!acontent) { | ||||||
|  | 				return 'acontent element not found'; | ||||||
|  | 			} | ||||||
|  | 			 | ||||||
|  | 			let removedCount = 0; | ||||||
|  | 			const elements = acontent.querySelectorAll('*'); | ||||||
|  | 			 | ||||||
|  | 			// 从后往前遍历,避免删除元素时影响索引 | ||||||
|  | 			for (let i = elements.length - 1; i >= 0; i--) { | ||||||
|  | 				const element = elements[i]; | ||||||
|  | 				const computedStyle = window.getComputedStyle(element); | ||||||
|  | 				 | ||||||
|  | 				if (computedStyle.display === 'none' || computedStyle.transform == 'matrix(0, 0, 0, 0, 0, 0)') { | ||||||
|  | 					element.remove(); | ||||||
|  | 					removedCount++; | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			 | ||||||
|  | 			return 'Removed ' + removedCount + ' hidden elements'; | ||||||
|  | 		})() | ||||||
|  | 	`) | ||||||
|  |  | ||||||
|  | 	if err != nil { | ||||||
|  | 		return "", fmt.Errorf("failed to remove hidden elements: %w", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	log.Printf("Hidden elements removal result: %s", result) | ||||||
|  |  | ||||||
|  | 	processedHTML, err := page.Content() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return "", fmt.Errorf("could not get page content: %w", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return processedHTML, nil | ||||||
|  | } | ||||||
							
								
								
									
										
											BIN
										
									
								
								downloader/bilinovel/read.ttf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								downloader/bilinovel/read.ttf
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| @@ -1,6 +1,3 @@ | |||||||
| package template |  | ||||||
| 
 |  | ||||||
| const StyleCSS = ` |  | ||||||
| body > div { | body > div { | ||||||
|   margin: 0 auto; |   margin: 0 auto; | ||||||
|   padding: 20px; |   padding: 20px; | ||||||
| @@ -40,4 +37,3 @@ img { | |||||||
|   margin-top: 1em; |   margin-top: 1em; | ||||||
|   margin-bottom: 1em; |   margin-bottom: 1em; | ||||||
| } | } | ||||||
| ` |  | ||||||
							
								
								
									
										12
									
								
								downloader/downloader.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								downloader/downloader.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | package downloader | ||||||
|  |  | ||||||
|  | import "bilinovel-downloader/model" | ||||||
|  |  | ||||||
|  | type Downloader interface { | ||||||
|  | 	GetNovel(novelId int, skipChapterContent bool, skipVolumes []int) (*model.Novel, error) | ||||||
|  | 	GetVolume(novelId int, volumeId int, skipChapterContent bool) (*model.Volume, error) | ||||||
|  | 	GetChapter(novelId int, volumeId int, chapterId int) (*model.Chapter, error) | ||||||
|  | 	GetStyleCSS() string | ||||||
|  | 	GetExtraFiles() []model.ExtraFile | ||||||
|  | 	Close() error | ||||||
|  | } | ||||||
							
								
								
									
										365
									
								
								epub/wrapper.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										365
									
								
								epub/wrapper.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,365 @@ | |||||||
|  | package epub | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"archive/zip" | ||||||
|  | 	"bilinovel-downloader/model" | ||||||
|  | 	"bilinovel-downloader/template" | ||||||
|  | 	"bilinovel-downloader/utils" | ||||||
|  | 	"context" | ||||||
|  | 	"fmt" | ||||||
|  | 	"io" | ||||||
|  | 	"os" | ||||||
|  | 	"path" | ||||||
|  | 	"path/filepath" | ||||||
|  | 	"strconv" | ||||||
|  | 	"strings" | ||||||
|  | 	"time" | ||||||
|  |  | ||||||
|  | 	"github.com/google/uuid" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func PackVolumeToEpub(volume *model.Volume, outputPath string, styleCSS string, extraFiles []model.ExtraFile) error { | ||||||
|  | 	outputPath = filepath.Join(outputPath, utils.CleanDirName(volume.Title)) | ||||||
|  | 	_, err := os.Stat(outputPath) | ||||||
|  | 	if err != nil { | ||||||
|  | 		if os.IsNotExist(err) { | ||||||
|  | 			err = os.MkdirAll(outputPath, 0755) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return fmt.Errorf("failed to create output directory: %v", err) | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  | 			return fmt.Errorf("failed to get output directory: %v", err) | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		err = os.RemoveAll(outputPath) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("failed to remove output directory: %v", err) | ||||||
|  | 		} | ||||||
|  | 		err = os.MkdirAll(outputPath, 0755) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("failed to create output directory: %v", err) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// 将文字写入 OEBPS/Text/chapter-%03v.xhtml | ||||||
|  | 	// 将图片写入 OEBPS/Images/chapter-%03v/ | ||||||
|  | 	for i, chapter := range volume.Chapters { | ||||||
|  | 		imageNames := make([]string, 0) | ||||||
|  | 		for imgName, imgData := range chapter.Content.Images { | ||||||
|  | 			imageNames = append(imageNames, imgName) | ||||||
|  | 			imgPath := filepath.Join(outputPath, fmt.Sprintf("OEBPS/Images/chapter-%03v/%s", i, imgName)) | ||||||
|  | 			err := os.MkdirAll(filepath.Dir(imgPath), 0755) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return fmt.Errorf("failed to create image directory: %v", err) | ||||||
|  | 			} | ||||||
|  | 			err = os.WriteFile(imgPath, imgData, 0644) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return fmt.Errorf("failed to write image: %v", err) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		chapterPath := filepath.Join(outputPath, fmt.Sprintf("OEBPS/Text/chapter-%03v.xhtml", i)) | ||||||
|  | 		err = os.MkdirAll(filepath.Dir(chapterPath), 0755) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("failed to create chapter directory: %v", err) | ||||||
|  | 		} | ||||||
|  | 		file, err := os.Create(chapterPath) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("failed to create chapter file: %v", err) | ||||||
|  | 		} | ||||||
|  | 		defer file.Close() | ||||||
|  | 		text := chapter.Content.Html | ||||||
|  | 		for _, imgName := range imageNames { | ||||||
|  | 			text = strings.ReplaceAll(text, imgName, fmt.Sprintf("../Images/chapter-%03v/%s", i, imgName)) | ||||||
|  | 		} | ||||||
|  | 		err = template.ContentXHTML(chapter.Title, text).Render(context.Background(), file) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("failed to write chapter: %v", err) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// 将 Cover 写入 | ||||||
|  | 	coverPath := filepath.Join(outputPath, fmt.Sprintf("cover%s", filepath.Ext(volume.CoverUrl))) | ||||||
|  | 	err = os.WriteFile(coverPath, volume.Cover, 0644) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("failed to write cover: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// 将 CoverXHTML 写入 OEBPS/Text/cover.xhtml | ||||||
|  | 	coverXHTMLPath := filepath.Join(outputPath, "OEBPS/Text/cover.xhtml") | ||||||
|  | 	file, err := os.Create(coverXHTMLPath) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("failed to create cover XHTML file: %v", err) | ||||||
|  | 	} | ||||||
|  | 	defer file.Close() | ||||||
|  | 	err = template.CoverXHTML(fmt.Sprintf("../../%s", filepath.Base(coverPath))).Render(context.Background(), file) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("failed to render cover XHTML: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// OEBPS/Text/contents.xhtml 目录 | ||||||
|  | 	contentsXHTMLPath := filepath.Join(outputPath, "OEBPS/Text/contents.xhtml") | ||||||
|  | 	file, err = os.Create(contentsXHTMLPath) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("failed to create contents XHTML file: %v", err) | ||||||
|  | 	} | ||||||
|  | 	defer file.Close() | ||||||
|  | 	contents := strings.Builder{} | ||||||
|  | 	contents.WriteString(`<nav epub:type="toc" id="toc">`) | ||||||
|  | 	contents.WriteString(`<ol>`) | ||||||
|  | 	for i, chapter := range volume.Chapters { | ||||||
|  | 		contents.WriteString(fmt.Sprintf(`<li><a href="chapter-%03v.xhtml">%s</a></li>`, i, chapter.Title)) | ||||||
|  | 	} | ||||||
|  | 	contents.WriteString(`</ol>`) | ||||||
|  | 	contents.WriteString(`</nav>`) | ||||||
|  | 	err = template.ContentXHTML("目录", contents.String()).Render(context.Background(), file) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("failed to render contents XHTML: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// ContainerXML | ||||||
|  | 	containerPath := filepath.Join(outputPath, "META-INF/container.xml") | ||||||
|  | 	err = os.MkdirAll(filepath.Dir(containerPath), 0755) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("failed to create container directory: %v", err) | ||||||
|  | 	} | ||||||
|  | 	file, err = os.Create(containerPath) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("failed to create container file: %v", err) | ||||||
|  | 	} | ||||||
|  | 	defer file.Close() | ||||||
|  | 	err = template.ContainerXML().Render(context.Background(), file) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("failed to render container: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// ContentOPF | ||||||
|  | 	u := uuid.New() | ||||||
|  | 	err = CreateContentOPF(outputPath, u.String(), volume, extraFiles) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("failed to create content OPF: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// 写入 CSS | ||||||
|  | 	cssPath := filepath.Join(outputPath, "style.css") | ||||||
|  | 	err = os.WriteFile(cssPath, []byte(styleCSS), 0644) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("failed to write CSS: %v", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// 写入 extraFiles | ||||||
|  | 	for _, file := range extraFiles { | ||||||
|  | 		extraFilePath := filepath.Join(outputPath, file.Path) | ||||||
|  | 		err = os.WriteFile(extraFilePath, file.Data, 0644) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("failed to write extra file: %v", err) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// 打包成 epub 文件 | ||||||
|  | 	err = PackEpub(outputPath) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("failed to pack epub: %v", err) | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func CreateContentOPF(outputPath string, uuid string, volume *model.Volume, extraFiles []model.ExtraFile) error { | ||||||
|  | 	creators := make([]model.DCCreator, 0) | ||||||
|  | 	for _, author := range volume.Authors { | ||||||
|  | 		creators = append(creators, model.DCCreator{ | ||||||
|  | 			Value: author, | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  | 	dc := &model.DublinCoreMetadata{ | ||||||
|  | 		Titles: []model.DCTitle{ | ||||||
|  | 			{ | ||||||
|  | 				Value: volume.Title, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		Identifiers: []model.DCIdentifier{ | ||||||
|  | 			{ | ||||||
|  | 				Value: fmt.Sprintf("urn:uuid:%s", uuid), | ||||||
|  | 				ID:    "book-id", | ||||||
|  | 				// Scheme: "UUID", | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		Languages: []model.DCLanguage{ | ||||||
|  | 			{ | ||||||
|  | 				Value: "zh-CN", | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		Descriptions: []model.DCDescription{ | ||||||
|  | 			{ | ||||||
|  | 				Value: volume.Description, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		Creators: creators, | ||||||
|  | 		Metas: []model.DublinCoreMeta{ | ||||||
|  | 			{ | ||||||
|  | 				Name:    "cover", | ||||||
|  | 				Content: "cover", | ||||||
|  | 			}, | ||||||
|  | 			{ | ||||||
|  | 				Property: "dcterms:modified", | ||||||
|  | 				Value:    time.Now().UTC().Format("2006-01-02T15:04:05Z"), | ||||||
|  | 			}, | ||||||
|  | 			{ | ||||||
|  | 				Name:    "calibre:series", | ||||||
|  | 				Content: volume.NovelTitle, | ||||||
|  | 			}, | ||||||
|  | 			{ | ||||||
|  | 				Name:    "calibre:series_index", | ||||||
|  | 				Content: strconv.Itoa(volume.SeriesIdx), | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  | 	manifest := &model.Manifest{ | ||||||
|  | 		Items: make([]model.ManifestItem, 0), | ||||||
|  | 	} | ||||||
|  | 	manifest.Items = append(manifest.Items, model.ManifestItem{ | ||||||
|  | 		ID:    "cover.xhtml", | ||||||
|  | 		Link:  "OEBPS/Text/cover.xhtml", | ||||||
|  | 		Media: "application/xhtml+xml", | ||||||
|  | 	}) | ||||||
|  | 	manifest.Items = append(manifest.Items, model.ManifestItem{ | ||||||
|  | 		ID:         "contents.xhtml", | ||||||
|  | 		Link:       "OEBPS/Text/contents.xhtml", | ||||||
|  | 		Media:      "application/xhtml+xml", | ||||||
|  | 		Properties: "nav", | ||||||
|  | 	}) | ||||||
|  | 	manifest.Items = append(manifest.Items, model.ManifestItem{ | ||||||
|  | 		ID:         "cover", | ||||||
|  | 		Link:       fmt.Sprintf("cover%s", filepath.Ext(volume.CoverUrl)), | ||||||
|  | 		Media:      fmt.Sprintf("image/%s", strings.ReplaceAll(strings.TrimPrefix(filepath.Ext(volume.CoverUrl), "."), "jpg", "jpeg")), | ||||||
|  | 		Properties: "cover-image", | ||||||
|  | 	}) | ||||||
|  | 	for i, chapter := range volume.Chapters { | ||||||
|  | 		manifest.Items = append(manifest.Items, model.ManifestItem{ | ||||||
|  | 			ID:    fmt.Sprintf("chapter-%03v.xhtml", i), | ||||||
|  | 			Link:  fmt.Sprintf("OEBPS/Text/chapter-%03v.xhtml", i), | ||||||
|  | 			Media: "application/xhtml+xml", | ||||||
|  | 		}) | ||||||
|  | 		for filename := range chapter.Content.Images { | ||||||
|  | 			item := model.ManifestItem{ | ||||||
|  | 				ID:    fmt.Sprintf("chapter-%03v-%s", i, filepath.Base(filename)), | ||||||
|  | 				Link:  fmt.Sprintf("OEBPS/Images/chapter-%03v/%s", i, filepath.Base(filename)), | ||||||
|  | 				Media: fmt.Sprintf("image/%s", strings.ReplaceAll(strings.TrimPrefix(filepath.Ext(filename), "."), "jpg", "jpeg")), | ||||||
|  | 			} | ||||||
|  | 			manifest.Items = append(manifest.Items, item) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	manifest.Items = append(manifest.Items, model.ManifestItem{ | ||||||
|  | 		ID:    "style", | ||||||
|  | 		Link:  "style.css", | ||||||
|  | 		Media: "text/css", | ||||||
|  | 	}) | ||||||
|  | 	// ExtraFiles | ||||||
|  | 	for _, file := range extraFiles { | ||||||
|  | 		manifest.Items = append(manifest.Items, file.ManifestItem) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	spine := &model.Spine{ | ||||||
|  | 		Items: make([]model.SpineItem, 0), | ||||||
|  | 	} | ||||||
|  | 	for _, item := range manifest.Items { | ||||||
|  | 		if filepath.Ext(item.Link) == ".xhtml" { | ||||||
|  | 			spine.Items = append(spine.Items, model.SpineItem{ | ||||||
|  | 				IDref: item.ID, | ||||||
|  | 			}) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	contentOPFPath := filepath.Join(outputPath, "content.opf") | ||||||
|  | 	err := os.MkdirAll(path.Dir(contentOPFPath), 0755) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("failed to create content directory: %v", err) | ||||||
|  | 	} | ||||||
|  | 	file, err := os.Create(contentOPFPath) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("failed to create content file: %v", err) | ||||||
|  | 	} | ||||||
|  | 	err = template.ContentOPF("book-id", dc, manifest, spine, nil).Render(context.Background(), file) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("failed to render content: %v", err) | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func PackEpub(dirPath string) error { | ||||||
|  | 	savePath := strings.TrimSuffix(dirPath, string(filepath.Separator)) + ".epub" | ||||||
|  | 	zipFile, err := os.Create(savePath) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	defer zipFile.Close() | ||||||
|  |  | ||||||
|  | 	zipWriter := zip.NewWriter(zipFile) | ||||||
|  | 	defer zipWriter.Close() | ||||||
|  |  | ||||||
|  | 	err = addStringToZip(zipWriter, "mimetype", "application/epub+zip", zip.Store) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	err = addDirContentToZip(zipWriter, dirPath, zip.Deflate) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func addStringToZip(zipWriter *zip.Writer, relPath, content string, method uint16) error { | ||||||
|  | 	header := &zip.FileHeader{ | ||||||
|  | 		Name:   relPath, | ||||||
|  | 		Method: method, | ||||||
|  | 	} | ||||||
|  | 	writer, err := zipWriter.CreateHeader(header) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	_, err = writer.Write([]byte(content)) | ||||||
|  | 	return err | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func addDirContentToZip(zipWriter *zip.Writer, dirPath string, method uint16) error { | ||||||
|  | 	return filepath.Walk(dirPath, func(filePath string, info os.FileInfo, err error) error { | ||||||
|  | 		if filepath.Base(filePath) == "volume.json" { | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 		if info.IsDir() { | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		relPath, err := filepath.Rel(dirPath, filePath) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		relPath = filepath.ToSlash(relPath) | ||||||
|  |  | ||||||
|  | 		file, err := os.Open(filePath) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 		defer file.Close() | ||||||
|  |  | ||||||
|  | 		header, err := zip.FileInfoHeader(info) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 		header.Name = relPath | ||||||
|  | 		header.Method = method | ||||||
|  |  | ||||||
|  | 		writer, err := zipWriter.CreateHeader(header) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		_, err = io.Copy(writer, file) | ||||||
|  | 		return err | ||||||
|  | 	}) | ||||||
|  | } | ||||||
							
								
								
									
										13
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										13
									
								
								go.mod
									
									
									
									
									
								
							| @@ -4,15 +4,22 @@ go 1.24.2 | |||||||
|  |  | ||||||
| require ( | require ( | ||||||
| 	github.com/PuerkitoBio/goquery v1.10.3 | 	github.com/PuerkitoBio/goquery v1.10.3 | ||||||
| 	github.com/a-h/templ v0.3.857 | 	github.com/a-h/templ v0.3.943 | ||||||
|  | 	github.com/bestnite/font-mapper v0.0.0-20250823155658-56c76d820267 | ||||||
| 	github.com/go-resty/resty/v2 v2.16.5 | 	github.com/go-resty/resty/v2 v2.16.5 | ||||||
| 	github.com/google/uuid v1.6.0 | 	github.com/google/uuid v1.6.0 | ||||||
|  | 	github.com/playwright-community/playwright-go v0.5200.1 | ||||||
| 	github.com/spf13/cobra v1.9.1 | 	github.com/spf13/cobra v1.9.1 | ||||||
| ) | ) | ||||||
|  |  | ||||||
| require ( | require ( | ||||||
| 	github.com/andybalholm/cascadia v1.3.3 // indirect | 	github.com/andybalholm/cascadia v1.3.3 // indirect | ||||||
|  | 	github.com/deckarep/golang-set/v2 v2.8.0 // indirect | ||||||
|  | 	github.com/go-jose/go-jose/v3 v3.0.4 // indirect | ||||||
|  | 	github.com/go-stack/stack v1.8.1 // indirect | ||||||
|  | 	github.com/golang/freetype v0.0.0-20170609003504-e2365dfdc4a0 // indirect | ||||||
| 	github.com/inconshreveable/mousetrap v1.1.0 // indirect | 	github.com/inconshreveable/mousetrap v1.1.0 // indirect | ||||||
| 	github.com/spf13/pflag v1.0.6 // indirect | 	github.com/spf13/pflag v1.0.7 // indirect | ||||||
| 	golang.org/x/net v0.39.0 // indirect | 	golang.org/x/image v0.30.0 // indirect | ||||||
|  | 	golang.org/x/net v0.43.0 // indirect | ||||||
| ) | ) | ||||||
|   | |||||||
							
								
								
									
										39
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										39
									
								
								go.sum
									
									
									
									
									
								
							| @@ -1,23 +1,48 @@ | |||||||
| github.com/PuerkitoBio/goquery v1.10.3 h1:pFYcNSqHxBD06Fpj/KsbStFRsgRATgnf3LeXiUkhzPo= | github.com/PuerkitoBio/goquery v1.10.3 h1:pFYcNSqHxBD06Fpj/KsbStFRsgRATgnf3LeXiUkhzPo= | ||||||
| github.com/PuerkitoBio/goquery v1.10.3/go.mod h1:tMUX0zDMHXYlAQk6p35XxQMqMweEKB7iK7iLNd4RH4Y= | github.com/PuerkitoBio/goquery v1.10.3/go.mod h1:tMUX0zDMHXYlAQk6p35XxQMqMweEKB7iK7iLNd4RH4Y= | ||||||
| github.com/a-h/templ v0.3.857 h1:6EqcJuGZW4OL+2iZ3MD+NnIcG7nGkaQeF2Zq5kf9ZGg= | github.com/a-h/templ v0.3.943 h1:o+mT/4yqhZ33F3ootBiHwaY4HM5EVaOJfIshvd5UNTY= | ||||||
| github.com/a-h/templ v0.3.857/go.mod h1:qhrhAkRFubE7khxLZHsBFHfX+gWwVNKbzKeF9GlPV4M= | github.com/a-h/templ v0.3.943/go.mod h1:oCZcnKRf5jjsGpf2yELzQfodLphd2mwecwG4Crk5HBo= | ||||||
| github.com/andybalholm/cascadia v1.3.3 h1:AG2YHrzJIm4BZ19iwJ/DAua6Btl3IwJX+VI4kktS1LM= | github.com/andybalholm/cascadia v1.3.3 h1:AG2YHrzJIm4BZ19iwJ/DAua6Btl3IwJX+VI4kktS1LM= | ||||||
| github.com/andybalholm/cascadia v1.3.3/go.mod h1:xNd9bqTn98Ln4DwST8/nG+H0yuB8Hmgu1YHNnWw0GeA= | github.com/andybalholm/cascadia v1.3.3/go.mod h1:xNd9bqTn98Ln4DwST8/nG+H0yuB8Hmgu1YHNnWw0GeA= | ||||||
|  | github.com/bestnite/font-mapper v0.0.0-20250823155658-56c76d820267 h1:nmUTJV2u/0XmVjQ++VIy/Hu+MtxdpQvOevvcSZtUATA= | ||||||
|  | github.com/bestnite/font-mapper v0.0.0-20250823155658-56c76d820267/go.mod h1:cfB1e9YhoI/QWrXPp3h6QVAKU6iCI2ifbjRPHP3xf/0= | ||||||
| github.com/cpuguy83/go-md2man/v2 v2.0.6/go.mod h1:oOW0eioCTA6cOiMLiUPZOpcVxMig6NIQQ7OS05n1F4g= | github.com/cpuguy83/go-md2man/v2 v2.0.6/go.mod h1:oOW0eioCTA6cOiMLiUPZOpcVxMig6NIQQ7OS05n1F4g= | ||||||
|  | github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||||
|  | github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= | ||||||
|  | github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||||
|  | github.com/deckarep/golang-set/v2 v2.8.0 h1:swm0rlPCmdWn9mESxKOjWk8hXSqoxOp+ZlfuyaAdFlQ= | ||||||
|  | github.com/deckarep/golang-set/v2 v2.8.0/go.mod h1:VAky9rY/yGXJOLEDv3OMci+7wtDpOF4IN+y82NBOac4= | ||||||
|  | github.com/go-jose/go-jose/v3 v3.0.4 h1:Wp5HA7bLQcKnf6YYao/4kpRpVMp/yf6+pJKV8WFSaNY= | ||||||
|  | github.com/go-jose/go-jose/v3 v3.0.4/go.mod h1:5b+7YgP7ZICgJDBdfjZaIt+H/9L9T/YQrVfLAMboGkQ= | ||||||
| github.com/go-resty/resty/v2 v2.16.5 h1:hBKqmWrr7uRc3euHVqmh1HTHcKn99Smr7o5spptdhTM= | github.com/go-resty/resty/v2 v2.16.5 h1:hBKqmWrr7uRc3euHVqmh1HTHcKn99Smr7o5spptdhTM= | ||||||
| github.com/go-resty/resty/v2 v2.16.5/go.mod h1:hkJtXbA2iKHzJheXYvQ8snQES5ZLGKMwQ07xAwp/fiA= | github.com/go-resty/resty/v2 v2.16.5/go.mod h1:hkJtXbA2iKHzJheXYvQ8snQES5ZLGKMwQ07xAwp/fiA= | ||||||
|  | github.com/go-stack/stack v1.8.1 h1:ntEHSVwIt7PNXNpgPmVfMrNhLtgjlmnZha2kOpuRiDw= | ||||||
|  | github.com/go-stack/stack v1.8.1/go.mod h1:dcoOX6HbPZSZptuspn9bctJ+N/CnF5gGygcUP3XYfe4= | ||||||
|  | github.com/golang/freetype v0.0.0-20170609003504-e2365dfdc4a0 h1:DACJavvAHhabrF08vX0COfcOBJRhZ8lUbR+ZWIs0Y5g= | ||||||
|  | github.com/golang/freetype v0.0.0-20170609003504-e2365dfdc4a0/go.mod h1:E/TSTwGwJL78qG/PmXZO1EjYhfJinVAhrmmHX6Z8B9k= | ||||||
|  | github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= | ||||||
| github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= | github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= | ||||||
| github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= | github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= | ||||||
| github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= | github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= | ||||||
| github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||||
| github.com/inconshreveable/mousetrap v1.1.0 h1:wN+x4NVGpMsO7ErUn/mUI3vEoE6Jt13X2s0bqwp9tc8= | github.com/inconshreveable/mousetrap v1.1.0 h1:wN+x4NVGpMsO7ErUn/mUI3vEoE6Jt13X2s0bqwp9tc8= | ||||||
| github.com/inconshreveable/mousetrap v1.1.0/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLfsEA9PFc4w1p2J65bw= | github.com/inconshreveable/mousetrap v1.1.0/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLfsEA9PFc4w1p2J65bw= | ||||||
|  | github.com/mitchellh/go-ps v1.0.0 h1:i6ampVEEF4wQFF+bkYfwYgY+F/uYJDktmvLPf7qIgjc= | ||||||
|  | github.com/mitchellh/go-ps v1.0.0/go.mod h1:J4lOc8z8yJs6vUwklHw2XEIiT4z4C40KtWVN3nvg8Pg= | ||||||
|  | github.com/playwright-community/playwright-go v0.5200.1 h1:Sm2oOuhqt0M5Y4kUi/Qh9w4cyyi3ZIWTBeGKImc2UVo= | ||||||
|  | github.com/playwright-community/playwright-go v0.5200.1/go.mod h1:UnnyQZaqUOO5ywAZu60+N4EiWReUqX1MQBBA3Oofvf8= | ||||||
|  | github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | ||||||
|  | github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||||||
| github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= | github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= | ||||||
| github.com/spf13/cobra v1.9.1 h1:CXSaggrXdbHK9CF+8ywj8Amf7PBRmPCOJugH954Nnlo= | github.com/spf13/cobra v1.9.1 h1:CXSaggrXdbHK9CF+8ywj8Amf7PBRmPCOJugH954Nnlo= | ||||||
| github.com/spf13/cobra v1.9.1/go.mod h1:nDyEzZ8ogv936Cinf6g1RU9MRY64Ir93oCnqb9wxYW0= | github.com/spf13/cobra v1.9.1/go.mod h1:nDyEzZ8ogv936Cinf6g1RU9MRY64Ir93oCnqb9wxYW0= | ||||||
| github.com/spf13/pflag v1.0.6 h1:jFzHGLGAlb3ruxLB8MhbI6A8+AQX/2eW4qeyNZXNp2o= |  | ||||||
| github.com/spf13/pflag v1.0.6/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= | github.com/spf13/pflag v1.0.6/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= | ||||||
|  | github.com/spf13/pflag v1.0.7 h1:vN6T9TfwStFPFM5XzjsvmzZkLuaLX+HS+0SeFLRgU6M= | ||||||
|  | github.com/spf13/pflag v1.0.7/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= | ||||||
|  | github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||||
|  | github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||||
|  | github.com/stretchr/testify v1.10.0 h1:Xv5erBjTwe/5IxqUQTdXv5kgmIvbHo3QQyRwhJsOfJA= | ||||||
|  | github.com/stretchr/testify v1.10.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY= | ||||||
| github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY= | github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY= | ||||||
| golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= | golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= | ||||||
| golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= | golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= | ||||||
| @@ -25,6 +50,8 @@ golang.org/x/crypto v0.13.0/go.mod h1:y6Z2r+Rw4iayiXXAIxJIDAJ1zMW4yaTpebo8fPOliY | |||||||
| golang.org/x/crypto v0.19.0/go.mod h1:Iy9bg/ha4yyC70EfRS8jz+B6ybOBKMaSxLj6P6oBDfU= | golang.org/x/crypto v0.19.0/go.mod h1:Iy9bg/ha4yyC70EfRS8jz+B6ybOBKMaSxLj6P6oBDfU= | ||||||
| golang.org/x/crypto v0.23.0/go.mod h1:CKFgDieR+mRhux2Lsu27y0fO304Db0wZe70UKqHu0v8= | golang.org/x/crypto v0.23.0/go.mod h1:CKFgDieR+mRhux2Lsu27y0fO304Db0wZe70UKqHu0v8= | ||||||
| golang.org/x/crypto v0.31.0/go.mod h1:kDsLvtWBEx7MV9tJOj9bnXsPbxwJQ6csT/x4KIN4Ssk= | golang.org/x/crypto v0.31.0/go.mod h1:kDsLvtWBEx7MV9tJOj9bnXsPbxwJQ6csT/x4KIN4Ssk= | ||||||
|  | golang.org/x/image v0.30.0 h1:jD5RhkmVAnjqaCUXfbGBrn3lpxbknfN9w2UhHHU+5B4= | ||||||
|  | golang.org/x/image v0.30.0/go.mod h1:SAEUTxCCMWSrJcCy/4HwavEsfZZJlYxeHLc6tTiAe/c= | ||||||
| golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= | golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= | ||||||
| golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= | golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= | ||||||
| golang.org/x/mod v0.12.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= | golang.org/x/mod v0.12.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= | ||||||
| @@ -39,8 +66,8 @@ golang.org/x/net v0.15.0/go.mod h1:idbUs1IY1+zTqbi8yxTbhexhEEk5ur9LInksu6HrEpk= | |||||||
| golang.org/x/net v0.21.0/go.mod h1:bIjVDfnllIU7BJ2DNgfnXvpSvtn8VRwhlsaeUTyUS44= | golang.org/x/net v0.21.0/go.mod h1:bIjVDfnllIU7BJ2DNgfnXvpSvtn8VRwhlsaeUTyUS44= | ||||||
| golang.org/x/net v0.25.0/go.mod h1:JkAGAh7GEvH74S6FOH42FLoXpXbE/aqXSrIQjXgsiwM= | golang.org/x/net v0.25.0/go.mod h1:JkAGAh7GEvH74S6FOH42FLoXpXbE/aqXSrIQjXgsiwM= | ||||||
| golang.org/x/net v0.33.0/go.mod h1:HXLR5J+9DxmrqMwG9qjGCxZ+zKXxBru04zlTvWlWuN4= | golang.org/x/net v0.33.0/go.mod h1:HXLR5J+9DxmrqMwG9qjGCxZ+zKXxBru04zlTvWlWuN4= | ||||||
| golang.org/x/net v0.39.0 h1:ZCu7HMWDxpXpaiKdhzIfaltL9Lp31x/3fCP11bc6/fY= | golang.org/x/net v0.43.0 h1:lat02VYK2j4aLzMzecihNvTlJNQUq316m2Mr9rnM6YE= | ||||||
| golang.org/x/net v0.39.0/go.mod h1:X7NRbYVEA+ewNkCNyJ513WmMdQ3BineSwVtN2zD/d+E= | golang.org/x/net v0.43.0/go.mod h1:vhO1fvI4dGsIjh73sWfUVjj3N7CA9WkKJNQm2svM6Jg= | ||||||
| golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||||
| golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||||
| golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||||
| @@ -87,4 +114,6 @@ golang.org/x/tools v0.13.0/go.mod h1:HvlwmtVNQAhOuCjW7xxvovg8wbNq7LwfXh/k7wXUl58 | |||||||
| golang.org/x/tools v0.21.1-0.20240508182429-e35e4ccd0d2d/go.mod h1:aiJjzUbINMkxbQROHiO6hDPo2LHcIPhhQsa9DLh0yGk= | golang.org/x/tools v0.21.1-0.20240508182429-e35e4ccd0d2d/go.mod h1:aiJjzUbINMkxbQROHiO6hDPo2LHcIPhhQsa9DLh0yGk= | ||||||
| golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||||
|  | gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||||
|  | gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= | ||||||
| gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||||
|   | |||||||
							
								
								
									
										13
									
								
								main.go
									
									
									
									
									
								
							
							
						
						
									
										13
									
								
								main.go
									
									
									
									
									
								
							| @@ -2,11 +2,20 @@ package main | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"bilinovel-downloader/cmd" | 	"bilinovel-downloader/cmd" | ||||||
|  | 	"io" | ||||||
| 	"log" | 	"log" | ||||||
|  |  | ||||||
|  | 	"github.com/playwright-community/playwright-go" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func main() { | func main() { | ||||||
| 	if err := cmd.RootCmd.Execute(); err != nil { | 	log.Println("Installing playwright") | ||||||
| 		log.Fatalf("Error executing command: %v", err) | 	err := playwright.Install(&playwright.RunOptions{ | ||||||
|  | 		Browsers: []string{"chromium"}, | ||||||
|  | 		Stdout:   io.Discard, | ||||||
|  | 	}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Panicf("failed to install playwright") | ||||||
| 	} | 	} | ||||||
|  | 	_ = cmd.RootCmd.Execute() | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,6 +1,14 @@ | |||||||
| package model | package model | ||||||
| 
 | 
 | ||||||
| import "encoding/xml" | import ( | ||||||
|  | 	"encoding/xml" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | type ExtraFile struct { | ||||||
|  | 	Data         []byte | ||||||
|  | 	Path         string | ||||||
|  | 	ManifestItem ManifestItem | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| type DublinCoreMetadata struct { | type DublinCoreMetadata struct { | ||||||
| 	XMLName xml.Name `xml:"metadata"` | 	XMLName xml.Name `xml:"metadata"` | ||||||
| @@ -158,7 +166,6 @@ type Spine struct { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (s *Spine) Marshal() (string, error) { | func (s *Spine) Marshal() (string, error) { | ||||||
| 	s.Toc = "ncx" |  | ||||||
| 	xmlBytes, err := xml.Marshal(s) | 	xmlBytes, err := xml.Marshal(s) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return "", err | 		return "", err | ||||||
| @@ -1,13 +1,17 @@ | |||||||
| package model | package model | ||||||
| 
 | 
 | ||||||
|  | type ChaperContent struct { | ||||||
|  | 	Html   string | ||||||
|  | 	Images map[string][]byte | ||||||
|  | } | ||||||
|  | 
 | ||||||
| type Chapter struct { | type Chapter struct { | ||||||
| 	Title           string | 	Id       int | ||||||
| 	Url             string | 	NovelId  int | ||||||
| 	Content         string | 	VolumeId int | ||||||
| 	ImageOEBPSPaths []string | 	Title    string | ||||||
| 	ImageFullPaths  []string | 	Url      string | ||||||
| 	TextOEBPSPath   string | 	Content  *ChaperContent | ||||||
| 	TextFullPath    string |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| type Volume struct { | type Volume struct { | ||||||
| @@ -15,7 +19,8 @@ type Volume struct { | |||||||
| 	SeriesIdx   int | 	SeriesIdx   int | ||||||
| 	Title       string | 	Title       string | ||||||
| 	Url         string | 	Url         string | ||||||
| 	Cover       string | 	CoverUrl    string | ||||||
|  | 	Cover       []byte | ||||||
| 	Description string | 	Description string | ||||||
| 	Authors     []string | 	Authors     []string | ||||||
| 	Chapters    []*Chapter | 	Chapters    []*Chapter | ||||||
| @@ -1,47 +0,0 @@ | |||||||
| package model |  | ||||||
|  |  | ||||||
| import "encoding/xml" |  | ||||||
|  |  | ||||||
| type TocNCXHead struct { |  | ||||||
| 	XMLName xml.Name         `xml:"head"` |  | ||||||
| 	Meta    []TocNCXHeadMeta `xml:"meta"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type TocNCXHeadMeta struct { |  | ||||||
| 	XMLName xml.Name `xml:"meta"` |  | ||||||
| 	Content string   `xml:"content,attr"` |  | ||||||
| 	Name    string   `xml:"name,attr"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (h *TocNCXHead) Marshal() (string, error) { |  | ||||||
| 	xmlBytes, err := xml.Marshal(h) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return "", err |  | ||||||
| 	} |  | ||||||
| 	return string(xmlBytes), nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type NavPoint struct { |  | ||||||
| 	Id        string          `xml:"id,attr"` |  | ||||||
| 	PlayOrder int             `xml:"playOrder,attr"` |  | ||||||
| 	Label     string          `xml:"navLabel>text"` |  | ||||||
| 	Content   NavPointContent `xml:"content"` |  | ||||||
| 	NavPoints []*NavPoint     `xml:"navPoint"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type NavPointContent struct { |  | ||||||
| 	Src string `xml:"src,attr"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type NavMap struct { |  | ||||||
| 	XMLName xml.Name    `xml:"navMap"` |  | ||||||
| 	Points  []*NavPoint `xml:"navPoint"` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (n *NavMap) Marshal() (string, error) { |  | ||||||
| 	xmlBytes, err := xml.Marshal(n) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return "", err |  | ||||||
| 	} |  | ||||||
| 	return string(xmlBytes), nil |  | ||||||
| } |  | ||||||
| @@ -1,10 +1,10 @@ | |||||||
| package template | package template | ||||||
|  |  | ||||||
| templ ContainerXML() { | templ ContainerXML() { | ||||||
| 	@templ.Raw(`<?xml version="1.0"  encoding="UTF-8"?>`) | 	@templ.Raw(`<?xml version='1.0' encoding='utf-8'?>`) | ||||||
| 	<container version="1.0" xmlns="urn:oasis:names:tc:opendocument:xmlns:container"> | 	<container xmlns="urn:oasis:names:tc:opendocument:xmlns:container" version="1.0"> | ||||||
| 		<rootfiles> | 		<rootfiles> | ||||||
| 			<rootfile full-path="OEBPS/content.opf" media-type="application/oebps-package+xml"></rootfile> | 			<rootfile full-path="content.opf" media-type="application/oebps-package+xml"></rootfile> | ||||||
| 		</rootfiles> | 		</rootfiles> | ||||||
| 	</container> | 	</container> | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| // Code generated by templ - DO NOT EDIT. | // Code generated by templ - DO NOT EDIT. | ||||||
|  |  | ||||||
| // templ: version: v0.3.857 | // templ: version: v0.3.943 | ||||||
| package template | package template | ||||||
|  |  | ||||||
| //lint:file-ignore SA4006 This context is only used if a nested component is present. | //lint:file-ignore SA4006 This context is only used if a nested component is present. | ||||||
| @@ -29,11 +29,11 @@ func ContainerXML() templ.Component { | |||||||
| 			templ_7745c5c3_Var1 = templ.NopComponent | 			templ_7745c5c3_Var1 = templ.NopComponent | ||||||
| 		} | 		} | ||||||
| 		ctx = templ.ClearChildren(ctx) | 		ctx = templ.ClearChildren(ctx) | ||||||
| 		templ_7745c5c3_Err = templ.Raw(`<?xml version="1.0"  encoding="UTF-8"?>`).Render(ctx, templ_7745c5c3_Buffer) | 		templ_7745c5c3_Err = templ.Raw(`<?xml version='1.0' encoding='utf-8'?>`).Render(ctx, templ_7745c5c3_Buffer) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "<container version=\"1.0\" xmlns=\"urn:oasis:names:tc:opendocument:xmlns:container\"><rootfiles><rootfile full-path=\"OEBPS/content.opf\" media-type=\"application/oebps-package+xml\"></rootfile></rootfiles></container>") | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "<container xmlns=\"urn:oasis:names:tc:opendocument:xmlns:container\" version=\"1.0\"><rootfiles><rootfile full-path=\"content.opf\" media-type=\"application/oebps-package+xml\"></rootfile></rootfiles></container>") | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ package template | |||||||
| import "bilinovel-downloader/model" | import "bilinovel-downloader/model" | ||||||
|  |  | ||||||
| templ ContentOPF(uniqueIdentifier string, dc *model.DublinCoreMetadata, manifest *model.Manifest, spine *model.Spine, guide *model.Guide) { | templ ContentOPF(uniqueIdentifier string, dc *model.DublinCoreMetadata, manifest *model.Manifest, spine *model.Spine, guide *model.Guide) { | ||||||
| 	@templ.Raw(`<?xml version="1.0"  encoding="UTF-8"?>`) | 	@templ.Raw(`<?xml version='1.0' encoding='utf-8'?>`) | ||||||
| 	<package version="3.0" xmlns="http://www.idpf.org/2007/opf" xmlns:dc="http://purl.org/dc/elements/1.1/" unique-identifier={ uniqueIdentifier }> | 	<package version="3.0" xmlns="http://www.idpf.org/2007/opf" xmlns:dc="http://purl.org/dc/elements/1.1/" unique-identifier={ uniqueIdentifier }> | ||||||
| 		if dc != nil { | 		if dc != nil { | ||||||
| 			{{ metadata, err := dc.Marshal() }} | 			{{ metadata, err := dc.Marshal() }} | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| // Code generated by templ - DO NOT EDIT. | // Code generated by templ - DO NOT EDIT. | ||||||
|  |  | ||||||
| // templ: version: v0.3.857 | // templ: version: v0.3.943 | ||||||
| package template | package template | ||||||
|  |  | ||||||
| //lint:file-ignore SA4006 This context is only used if a nested component is present. | //lint:file-ignore SA4006 This context is only used if a nested component is present. | ||||||
| @@ -31,7 +31,7 @@ func ContentOPF(uniqueIdentifier string, dc *model.DublinCoreMetadata, manifest | |||||||
| 			templ_7745c5c3_Var1 = templ.NopComponent | 			templ_7745c5c3_Var1 = templ.NopComponent | ||||||
| 		} | 		} | ||||||
| 		ctx = templ.ClearChildren(ctx) | 		ctx = templ.ClearChildren(ctx) | ||||||
| 		templ_7745c5c3_Err = templ.Raw(`<?xml version="1.0"  encoding="UTF-8"?>`).Render(ctx, templ_7745c5c3_Buffer) | 		templ_7745c5c3_Err = templ.Raw(`<?xml version='1.0' encoding='utf-8'?>`).Render(ctx, templ_7745c5c3_Buffer) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -1,21 +1,19 @@ | |||||||
| package template | package template | ||||||
|  |  | ||||||
| import "bilinovel-downloader/model" | templ ContentXHTML(title, content string) { | ||||||
|  | 	@templ.Raw(`<?xml version='1.0' encoding='utf-8'?>`) | ||||||
| templ ContentXHTML(content *model.Chapter) { | 	// @templ.Raw(`<!DOCTYPE html>`) | ||||||
| 	@templ.Raw(`<?xml version="1.0" encoding="utf-8" standalone="no"?>`) | 	<html xmlns="http://www.w3.org/1999/xhtml" xmlns:epub="http://www.idpf.org/2007/ops" xml:lang="zh-CN"> | ||||||
| 	@templ.Raw(`<!DOCTYPE html>`) |  | ||||||
| 	<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="zh-CN" xmlns:epub="http://www.idpf.org/2007/ops" xmlns:xml="http://www.w3.org/XML/1998/namespace"> |  | ||||||
| 		<head> | 		<head> | ||||||
| 			<title>{ content.Title }</title> | 			<title>{ title }</title> | ||||||
| 			@templ.Raw(`<link href="../Styles/style.css" rel="stylesheet" type="text/css"/>`) | 			@templ.Raw(`<link href="../../style.css" rel="stylesheet" type="text/css"/>`) | ||||||
| 		</head> | 		</head> | ||||||
| 		<body> | 		<body> | ||||||
| 			<div class="chapter"> | 			<div class="chapter"> | ||||||
| 				<h1>{ content.Title }</h1> | 				<h1>{ title }</h1> | ||||||
| 				@templ.Raw(`<hr/>`) | 				@templ.Raw(`<hr/>`) | ||||||
| 				<div class="content"> | 				<div class="content"> | ||||||
| 					@templ.Raw(content.Content) | 					@templ.Raw(content) | ||||||
| 				</div> | 				</div> | ||||||
| 			</div> | 			</div> | ||||||
| 		</body> | 		</body> | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| // Code generated by templ - DO NOT EDIT. | // Code generated by templ - DO NOT EDIT. | ||||||
|  |  | ||||||
| // templ: version: v0.3.857 | // templ: version: v0.3.943 | ||||||
| package template | package template | ||||||
|  |  | ||||||
| //lint:file-ignore SA4006 This context is only used if a nested component is present. | //lint:file-ignore SA4006 This context is only used if a nested component is present. | ||||||
| @@ -8,9 +8,7 @@ package template | |||||||
| import "github.com/a-h/templ" | import "github.com/a-h/templ" | ||||||
| import templruntime "github.com/a-h/templ/runtime" | import templruntime "github.com/a-h/templ/runtime" | ||||||
|  |  | ||||||
| import "bilinovel-downloader/model" | func ContentXHTML(title, content string) templ.Component { | ||||||
|  |  | ||||||
| func ContentXHTML(content *model.Chapter) templ.Component { |  | ||||||
| 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | ||||||
| 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | ||||||
| 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | ||||||
| @@ -31,22 +29,18 @@ func ContentXHTML(content *model.Chapter) templ.Component { | |||||||
| 			templ_7745c5c3_Var1 = templ.NopComponent | 			templ_7745c5c3_Var1 = templ.NopComponent | ||||||
| 		} | 		} | ||||||
| 		ctx = templ.ClearChildren(ctx) | 		ctx = templ.ClearChildren(ctx) | ||||||
| 		templ_7745c5c3_Err = templ.Raw(`<?xml version="1.0" encoding="utf-8" standalone="no"?>`).Render(ctx, templ_7745c5c3_Buffer) | 		templ_7745c5c3_Err = templ.Raw(`<?xml version='1.0' encoding='utf-8'?>`).Render(ctx, templ_7745c5c3_Buffer) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = templ.Raw(`<!DOCTYPE html>`).Render(ctx, templ_7745c5c3_Buffer) | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "<html xmlns=\"http://www.w3.org/1999/xhtml\" xmlns:epub=\"http://www.idpf.org/2007/ops\" xml:lang=\"zh-CN\"><head><title>") | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "<html xmlns=\"http://www.w3.org/1999/xhtml\" xml:lang=\"zh-CN\" xmlns:epub=\"http://www.idpf.org/2007/ops\" xmlns:xml=\"http://www.w3.org/XML/1998/namespace\"><head><title>") |  | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		var templ_7745c5c3_Var2 string | 		var templ_7745c5c3_Var2 string | ||||||
| 		templ_7745c5c3_Var2, templ_7745c5c3_Err = templ.JoinStringErrs(content.Title) | 		templ_7745c5c3_Var2, templ_7745c5c3_Err = templ.JoinStringErrs(title) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `template/content.xhtml.templ`, Line: 10, Col: 25} | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `template/content.xhtml.templ`, Line: 8, Col: 17} | ||||||
| 		} | 		} | ||||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var2)) | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var2)) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| @@ -56,7 +50,7 @@ func ContentXHTML(content *model.Chapter) templ.Component { | |||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = templ.Raw(`<link href="../Styles/style.css" rel="stylesheet" type="text/css"/>`).Render(ctx, templ_7745c5c3_Buffer) | 		templ_7745c5c3_Err = templ.Raw(`<link href="../../style.css" rel="stylesheet" type="text/css"/>`).Render(ctx, templ_7745c5c3_Buffer) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| @@ -65,9 +59,9 @@ func ContentXHTML(content *model.Chapter) templ.Component { | |||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		var templ_7745c5c3_Var3 string | 		var templ_7745c5c3_Var3 string | ||||||
| 		templ_7745c5c3_Var3, templ_7745c5c3_Err = templ.JoinStringErrs(content.Title) | 		templ_7745c5c3_Var3, templ_7745c5c3_Err = templ.JoinStringErrs(title) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `template/content.xhtml.templ`, Line: 15, Col: 23} | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `template/content.xhtml.templ`, Line: 13, Col: 15} | ||||||
| 		} | 		} | ||||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var3)) | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var3)) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| @@ -85,7 +79,7 @@ func ContentXHTML(content *model.Chapter) templ.Component { | |||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = templ.Raw(content.Content).Render(ctx, templ_7745c5c3_Buffer) | 		templ_7745c5c3_Err = templ.Raw(content).Render(ctx, templ_7745c5c3_Buffer) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
|   | |||||||
							
								
								
									
										37
									
								
								template/cover.xhtml.templ
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								template/cover.xhtml.templ
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | |||||||
|  | package template | ||||||
|  |  | ||||||
|  | templ CoverXHTML(coverPath string) { | ||||||
|  | 	@templ.Raw(`<?xml version='1.0' encoding='utf-8'?>`) | ||||||
|  | 	<html xmlns="http://www.w3.org/1999/xhtml" xmlns:epub="http://www.idpf.org/2007/ops" | ||||||
|  | 	xml:lang="zh-CN"> | ||||||
|  | 	<head> | ||||||
|  | 		<title>Cover</title> | ||||||
|  | 	</head> | ||||||
|  | 	<style type="text/css"> | ||||||
|  | 		@page { | ||||||
|  | 		padding: 0pt; | ||||||
|  | 		margin: 0pt | ||||||
|  | 		} | ||||||
|  | 		body { | ||||||
|  | 		text-align: center; | ||||||
|  | 		padding: 0pt; | ||||||
|  | 		margin: 0pt; | ||||||
|  | 		} | ||||||
|  | 	</style> | ||||||
|  | 	<body> | ||||||
|  | 		<div> | ||||||
|  | 			<svg | ||||||
|  | 				xmlns="http://www.w3.org/2000/svg" | ||||||
|  | 				xmlns:xlink="http://www.w3.org/1999/xlink" | ||||||
|  | 				version="1.1" | ||||||
|  | 				width="100%" | ||||||
|  | 				height="100%" | ||||||
|  | 				viewBox="0 0 400 581" | ||||||
|  | 				preserveAspectRatio="none" | ||||||
|  | 			> | ||||||
|  | 				<image width="400" height="581" xlink:href={ coverPath }></image> | ||||||
|  | 			</svg> | ||||||
|  | 		</div> | ||||||
|  | 	</body> | ||||||
|  | </html> | ||||||
|  | } | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| // Code generated by templ - DO NOT EDIT. | // Code generated by templ - DO NOT EDIT. | ||||||
| 
 | 
 | ||||||
| // templ: version: v0.3.857 | // templ: version: v0.3.943 | ||||||
| package template | package template | ||||||
| 
 | 
 | ||||||
| //lint:file-ignore SA4006 This context is only used if a nested component is present. | //lint:file-ignore SA4006 This context is only used if a nested component is present. | ||||||
| @@ -8,9 +8,7 @@ package template | |||||||
| import "github.com/a-h/templ" | import "github.com/a-h/templ" | ||||||
| import templruntime "github.com/a-h/templ/runtime" | import templruntime "github.com/a-h/templ/runtime" | ||||||
| 
 | 
 | ||||||
| import "bilinovel-downloader/model" | func CoverXHTML(coverPath string) templ.Component { | ||||||
| 
 |  | ||||||
| func TocNCX(title string, head *model.TocNCXHead, navMap *model.NavMap) templ.Component { |  | ||||||
| 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | ||||||
| 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | ||||||
| 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | ||||||
| @@ -31,54 +29,24 @@ func TocNCX(title string, head *model.TocNCXHead, navMap *model.NavMap) templ.Co | |||||||
| 			templ_7745c5c3_Var1 = templ.NopComponent | 			templ_7745c5c3_Var1 = templ.NopComponent | ||||||
| 		} | 		} | ||||||
| 		ctx = templ.ClearChildren(ctx) | 		ctx = templ.ClearChildren(ctx) | ||||||
| 		templ_7745c5c3_Err = templ.Raw(`<?xml version="1.0" encoding="UTF-8"?>`).Render(ctx, templ_7745c5c3_Buffer) | 		templ_7745c5c3_Err = templ.Raw(`<?xml version='1.0' encoding='utf-8'?>`).Render(ctx, templ_7745c5c3_Buffer) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = templ.Raw(`<!DOCTYPE ncx PUBLIC "-//NISO//DTD ncx 2005-1//EN" "http://www.daisy.org/z3986/2005/ncx-2005-1.dtd">`).Render(ctx, templ_7745c5c3_Buffer) | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "<html xmlns=\"http://www.w3.org/1999/xhtml\" xmlns:epub=\"http://www.idpf.org/2007/ops\" xml:lang=\"zh-CN\"><head><title>Cover</title></head><style type=\"text/css\">\n\t\t@page {\n\t\tpadding: 0pt;\n\t\tmargin: 0pt\n\t\t}\n\t\tbody {\n\t\ttext-align: center;\n\t\tpadding: 0pt;\n\t\tmargin: 0pt;\n\t\t}\n\t</style><body><div><svg xmlns=\"http://www.w3.org/2000/svg\" xmlns:xlink=\"http://www.w3.org/1999/xlink\" version=\"1.1\" width=\"100%\" height=\"100%\" viewBox=\"0 0 400 581\" preserveAspectRatio=\"none\"><image width=\"400\" height=\"581\" xlink:href=\"") | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "<ncx xmlns=\"http://www.daisy.org/z3986/2005/ncx/\" version=\"2005-1\">") |  | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		if head != nil { |  | ||||||
| 			head, err := head.Marshal() |  | ||||||
| 			if err == nil { |  | ||||||
| 				templ_7745c5c3_Err = templ.Raw(head).Render(ctx, templ_7745c5c3_Buffer) |  | ||||||
| 				if templ_7745c5c3_Err != nil { |  | ||||||
| 					return templ_7745c5c3_Err |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, "<docTitle><text>") |  | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		var templ_7745c5c3_Var2 string | 		var templ_7745c5c3_Var2 string | ||||||
| 		templ_7745c5c3_Var2, templ_7745c5c3_Err = templ.JoinStringErrs(title) | 		templ_7745c5c3_Var2, templ_7745c5c3_Err = templ.JoinStringErrs(coverPath) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `template/toc.ncx.templ`, Line: 16, Col: 16} | 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `template/cover.xhtml.templ`, Line: 32, Col: 58} | ||||||
| 		} | 		} | ||||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var2)) | 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var2)) | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 3, "</text></docTitle> ") | 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, "\"></image></svg></div></body></html>") | ||||||
| 		if templ_7745c5c3_Err != nil { |  | ||||||
| 			return templ_7745c5c3_Err |  | ||||||
| 		} |  | ||||||
| 		if navMap != nil { |  | ||||||
| 			navMap, err := navMap.Marshal() |  | ||||||
| 			if err == nil { |  | ||||||
| 				templ_7745c5c3_Err = templ.Raw(navMap).Render(ctx, templ_7745c5c3_Buffer) |  | ||||||
| 				if templ_7745c5c3_Err != nil { |  | ||||||
| 					return templ_7745c5c3_Err |  | ||||||
| 				} |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 4, "</ncx>") |  | ||||||
| 		if templ_7745c5c3_Err != nil { | 		if templ_7745c5c3_Err != nil { | ||||||
| 			return templ_7745c5c3_Err | 			return templ_7745c5c3_Err | ||||||
| 		} | 		} | ||||||
| @@ -1,25 +0,0 @@ | |||||||
| package template |  | ||||||
|  |  | ||||||
| import "bilinovel-downloader/model" |  | ||||||
|  |  | ||||||
| templ TocNCX(title string, head *model.TocNCXHead, navMap *model.NavMap) { |  | ||||||
| 	@templ.Raw(`<?xml version="1.0" encoding="UTF-8"?>`) |  | ||||||
| 	@templ.Raw(`<!DOCTYPE ncx PUBLIC "-//NISO//DTD ncx 2005-1//EN" "http://www.daisy.org/z3986/2005/ncx-2005-1.dtd">`) |  | ||||||
| 	<ncx xmlns="http://www.daisy.org/z3986/2005/ncx/" version="2005-1"> |  | ||||||
| 		if head != nil { |  | ||||||
| 			{{ head, err := head.Marshal() }} |  | ||||||
| 			if err == nil { |  | ||||||
| 				@templ.Raw(head) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		<docTitle> |  | ||||||
| 			<text>{ title }</text> |  | ||||||
| 		</docTitle> |  | ||||||
| 		if navMap != nil { |  | ||||||
| 			{{ navMap, err := navMap.Marshal() }} |  | ||||||
| 			if err == nil { |  | ||||||
| 				@templ.Raw(navMap) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	</ncx> |  | ||||||
| } |  | ||||||
							
								
								
									
										59
									
								
								test/bilinovel_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								test/bilinovel_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,59 @@ | |||||||
|  | package test | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"bilinovel-downloader/downloader/bilinovel" | ||||||
|  | 	"encoding/json" | ||||||
|  | 	"fmt" | ||||||
|  | 	"testing" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func TestBilinovel_GetNovel(t *testing.T) { | ||||||
|  | 	bilinovel, err := bilinovel.New(bilinovel.BilinovelNewOption{Headless: false, Concurrency: 5}) | ||||||
|  | 	bilinovel.SetTextOnly(true) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("failed to create bilinovel: %v", err) | ||||||
|  | 	} | ||||||
|  | 	novel, err := bilinovel.GetNovel(2727, false, nil) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("failed to get novel: %v", err) | ||||||
|  | 	} | ||||||
|  | 	jsonBytes, err := json.Marshal(novel) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("failed to marshal novel: %v", err) | ||||||
|  | 	} | ||||||
|  | 	fmt.Println(string(jsonBytes)) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestBilinovel_GetVolume(t *testing.T) { | ||||||
|  | 	bilinovel, err := bilinovel.New(bilinovel.BilinovelNewOption{Headless: false, Concurrency: 1}) | ||||||
|  | 	bilinovel.SetTextOnly(true) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("failed to create bilinovel: %v", err) | ||||||
|  | 	} | ||||||
|  | 	volume, err := bilinovel.GetVolume(2727, 129092, false) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("failed to get volume: %v", err) | ||||||
|  | 	} | ||||||
|  | 	jsonBytes, err := json.Marshal(volume) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("failed to marshal volume: %v", err) | ||||||
|  | 	} | ||||||
|  | 	fmt.Println(string(jsonBytes)) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestBilinovel_GetChapter(t *testing.T) { | ||||||
|  | 	bilinovel, err := bilinovel.New(bilinovel.BilinovelNewOption{Headless: false, Concurrency: 1}) | ||||||
|  | 	bilinovel.SetTextOnly(true) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("failed to create bilinovel: %v", err) | ||||||
|  | 	} | ||||||
|  | 	chapter, err := bilinovel.GetChapter(2727, 129092, 129094) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("failed to get chapter: %v", err) | ||||||
|  | 	} | ||||||
|  | 	jsonBytes, err := json.Marshal(chapter) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("failed to marshal chapter: %v", err) | ||||||
|  | 	} | ||||||
|  | 	fmt.Println(string(jsonBytes)) | ||||||
|  | } | ||||||
							
								
								
									
										55
									
								
								text/wrapper.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								text/wrapper.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,55 @@ | |||||||
|  | package text | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"bilinovel-downloader/model" | ||||||
|  | 	"bilinovel-downloader/utils" | ||||||
|  | 	"fmt" | ||||||
|  | 	"os" | ||||||
|  | 	"path/filepath" | ||||||
|  | 	"strings" | ||||||
|  |  | ||||||
|  | 	"github.com/PuerkitoBio/goquery" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func PackVolumeToText(volume *model.Volume, outputPath string) error { | ||||||
|  | 	outputPath = filepath.Join(outputPath, utils.CleanDirName(volume.Title)) | ||||||
|  | 	_, err := os.Stat(outputPath) | ||||||
|  | 	if err != nil { | ||||||
|  | 		if os.IsNotExist(err) { | ||||||
|  | 			err = os.MkdirAll(outputPath, 0755) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return fmt.Errorf("failed to create output directory: %v", err) | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  | 			return fmt.Errorf("failed to get output directory: %v", err) | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		err = os.RemoveAll(outputPath) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("failed to remove output directory: %v", err) | ||||||
|  | 		} | ||||||
|  | 		err = os.MkdirAll(outputPath, 0755) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("failed to create output directory: %v", err) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	for i, chapter := range volume.Chapters { | ||||||
|  | 		chapterPath := filepath.Join(outputPath, fmt.Sprintf("%03d-%s.txt", i, chapter.Title)) | ||||||
|  | 		chapterFile, err := os.Create(chapterPath) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("failed to create chapter file: %v", err) | ||||||
|  | 		} | ||||||
|  | 		defer chapterFile.Close() | ||||||
|  | 		doc, err := goquery.NewDocumentFromReader(strings.NewReader(chapter.Content.Html)) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("failed to create chapter file: %v", err) | ||||||
|  | 		} | ||||||
|  | 		doc.Find("img").Remove() | ||||||
|  | 		text := doc.Text() | ||||||
|  | 		_, err = chapterFile.WriteString(strings.TrimSpace(text)) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return fmt.Errorf("failed to write chapter file: %v", err) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
							
								
								
									
										117
									
								
								utils/epub.go
									
									
									
									
									
								
							
							
						
						
									
										117
									
								
								utils/epub.go
									
									
									
									
									
								
							| @@ -1,117 +0,0 @@ | |||||||
| package utils |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"archive/zip" |  | ||||||
| 	"bilinovel-downloader/template" |  | ||||||
| 	"io" |  | ||||||
| 	"os" |  | ||||||
| 	"path/filepath" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func CreateEpub(path string) error { |  | ||||||
| 	savePath := path + ".epub" |  | ||||||
| 	zipFile, err := os.Create(savePath) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
| 	defer zipFile.Close() |  | ||||||
|  |  | ||||||
| 	zipWriter := zip.NewWriter(zipFile) |  | ||||||
| 	defer zipWriter.Close() |  | ||||||
|  |  | ||||||
| 	err = addStringToZip(zipWriter, "mimetype", "application/epub+zip", zip.Store) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	err = addDirContentToZip(zipWriter, path, zip.Deflate) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	err = addStringToZip(zipWriter, "OEBPS/Styles/style.css", template.StyleCSS, zip.Deflate) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // func addFileToZip(zipWriter *zip.Writer, filename string, relPath string, method uint16) error { |  | ||||||
| // 	file, err := os.Open(filename) |  | ||||||
| // 	if err != nil { |  | ||||||
| // 		return err |  | ||||||
| // 	} |  | ||||||
| // 	defer file.Close() |  | ||||||
|  |  | ||||||
| // 	info, err := file.Stat() |  | ||||||
| // 	if err != nil { |  | ||||||
| // 		return err |  | ||||||
| // 	} |  | ||||||
|  |  | ||||||
| // 	header, err := zip.FileInfoHeader(info) |  | ||||||
| // 	if err != nil { |  | ||||||
| // 		return err |  | ||||||
| // 	} |  | ||||||
| // 	header.Name = relPath |  | ||||||
| // 	header.Method = method |  | ||||||
|  |  | ||||||
| // 	writer, err := zipWriter.CreateHeader(header) |  | ||||||
| // 	if err != nil { |  | ||||||
| // 		return err |  | ||||||
| // 	} |  | ||||||
|  |  | ||||||
| // 	_, err = io.Copy(writer, file) |  | ||||||
| // 	return err |  | ||||||
| // } |  | ||||||
|  |  | ||||||
| func addStringToZip(zipWriter *zip.Writer, relPath, content string, method uint16) error { |  | ||||||
| 	header := &zip.FileHeader{ |  | ||||||
| 		Name:   relPath, |  | ||||||
| 		Method: method, |  | ||||||
| 	} |  | ||||||
| 	writer, err := zipWriter.CreateHeader(header) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return err |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	_, err = writer.Write([]byte(content)) |  | ||||||
| 	return err |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func addDirContentToZip(zipWriter *zip.Writer, dirPath string, method uint16) error { |  | ||||||
| 	return filepath.Walk(dirPath, func(filePath string, info os.FileInfo, err error) error { |  | ||||||
| 		if err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 		if info.IsDir() { |  | ||||||
| 			return nil |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		relPath, err := filepath.Rel(dirPath, filePath) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		file, err := os.Open(filePath) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 		defer file.Close() |  | ||||||
|  |  | ||||||
| 		header, err := zip.FileInfoHeader(info) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 		header.Name = relPath |  | ||||||
| 		header.Method = method |  | ||||||
|  |  | ||||||
| 		writer, err := zipWriter.CreateHeader(header) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		_, err = io.Copy(writer, file) |  | ||||||
| 		return err |  | ||||||
| 	}) |  | ||||||
| } |  | ||||||
| @@ -9,11 +9,19 @@ import ( | |||||||
| 	"github.com/go-resty/resty/v2" | 	"github.com/go-resty/resty/v2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var client *resty.Client | type RestyClient struct { | ||||||
|  | 	client      *resty.Client | ||||||
|  | 	concurrency int | ||||||
|  | 	sem         chan struct{} | ||||||
|  | } | ||||||
|  |  | ||||||
| func init() { | func NewRestyClient(concurrency int) *RestyClient { | ||||||
| 	client = resty.New() | 	client := &RestyClient{ | ||||||
| 	client.SetTransport(&http.Transport{ | 		client:      resty.New(), | ||||||
|  | 		concurrency: concurrency, | ||||||
|  | 		sem:         make(chan struct{}, concurrency), | ||||||
|  | 	} | ||||||
|  | 	client.client.SetTransport(&http.Transport{ | ||||||
| 		DialContext: func(ctx context.Context, network, addr string) (net.Conn, error) { | 		DialContext: func(ctx context.Context, network, addr string) (net.Conn, error) { | ||||||
| 			if addr == "www.bilinovel.com:443" { | 			if addr == "www.bilinovel.com:443" { | ||||||
| 				addr = "64.140.161.52:443" | 				addr = "64.140.161.52:443" | ||||||
| @@ -24,7 +32,16 @@ func init() { | |||||||
| 		}, | 		}, | ||||||
| 		TLSHandshakeTimeout: 10 * time.Second, | 		TLSHandshakeTimeout: 10 * time.Second, | ||||||
| 	}) | 	}) | ||||||
| 	client.SetRetryCount(10). | 	client.client. | ||||||
|  | 		OnBeforeRequest(func(c *resty.Client, req *resty.Request) error { | ||||||
|  | 			client.sem <- struct{}{} | ||||||
|  | 			return nil | ||||||
|  | 		}). | ||||||
|  | 		OnAfterResponse(func(c *resty.Client, resp *resty.Response) error { | ||||||
|  | 			<-client.sem | ||||||
|  | 			return nil | ||||||
|  | 		}) | ||||||
|  | 	client.client.SetRetryCount(10). | ||||||
| 		SetRetryWaitTime(3 * time.Second). | 		SetRetryWaitTime(3 * time.Second). | ||||||
| 		SetRetryAfter(func(client *resty.Client, resp *resty.Response) (time.Duration, error) { | 		SetRetryAfter(func(client *resty.Client, resp *resty.Response) (time.Duration, error) { | ||||||
| 			if resp.StatusCode() == http.StatusTooManyRequests { | 			if resp.StatusCode() == http.StatusTooManyRequests { | ||||||
| @@ -43,10 +60,13 @@ func init() { | |||||||
| 		AddRetryCondition(func(r *resty.Response, err error) bool { | 		AddRetryCondition(func(r *resty.Response, err error) bool { | ||||||
| 			return err != nil || r.StatusCode() == http.StatusTooManyRequests | 			return err != nil || r.StatusCode() == http.StatusTooManyRequests | ||||||
| 		}) | 		}) | ||||||
|  |  | ||||||
|  | 	client.client.SetLogger(disableLogger{}).SetHeader("Accept-Charset", "utf-8").SetHeader("User-Agent", "Mozilla/5.0 (X11; Linux x86_64; rv:133.0) Gecko/20100101 Firefox/133.0") | ||||||
|  | 	return client | ||||||
| } | } | ||||||
|  |  | ||||||
| func Request() *resty.Request { | func (c *RestyClient) R() *resty.Request { | ||||||
| 	return client.R().SetLogger(disableLogger{}).SetHeader("Accept-Charset", "utf-8").SetHeader("User-Agent", "Mozilla/5.0 (X11; Linux x86_64; rv:133.0) Gecko/20100101 Firefox/133.0") | 	return c.client.R() | ||||||
| } | } | ||||||
|  |  | ||||||
| type disableLogger struct{} | type disableLogger struct{} | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user