├── mangodl
├── assets
├── logo.png
├── example.png
└── opencomic.png
├── .idea
├── vcs.xml
├── .gitignore
├── modules.xml
└── mangodl-git.iml
├── main.go
├── install.sh
├── go.mod
├── .gitignore
├── mangodl.rb
├── download
└── download.go
├── output
├── cbz.go
└── pdf.go
├── go.sum
├── utils
├── conf.go
└── utils.go
├── README.md
└── LICENSE
/mangodl:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/liamtoaldo/mangodl/HEAD/mangodl
--------------------------------------------------------------------------------
/assets/logo.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/liamtoaldo/mangodl/HEAD/assets/logo.png
--------------------------------------------------------------------------------
/assets/example.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/liamtoaldo/mangodl/HEAD/assets/example.png
--------------------------------------------------------------------------------
/assets/opencomic.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/liamtoaldo/mangodl/HEAD/assets/opencomic.png
--------------------------------------------------------------------------------
/.idea/vcs.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
--------------------------------------------------------------------------------
/.idea/.gitignore:
--------------------------------------------------------------------------------
1 | # Default ignored files
2 | /shelf/
3 | /workspace.xml
4 | # Datasource local storage ignored files
5 | /dataSources/
6 | /dataSources.local.xml
7 | # Editor-based HTTP Client requests
8 | /httpRequests/
9 |
--------------------------------------------------------------------------------
/.idea/modules.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
--------------------------------------------------------------------------------
/main.go:
--------------------------------------------------------------------------------
1 | package main
2 |
3 | import (
4 | "mangodl-git/utils"
5 | )
6 |
7 | func main() {
8 | //start the program using the internal "utils" package
9 | utils.Execute()
10 | }
11 |
12 | //TODO add option to delete a downloaded manga
13 | //TODO (long future) change host to https://www.mangahere.cc/
14 |
--------------------------------------------------------------------------------
/.idea/mangodl-git.iml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
--------------------------------------------------------------------------------
/install.sh:
--------------------------------------------------------------------------------
1 | #!/bin/bash
2 |
3 | ## UNCOMMENT THESE LINES IF YOU WANT TO BUILD FROM SOURCE
4 | #echo Removing pre-built binary
5 | #rm mangodl
6 | #echo Installing the needed dependencies
7 | #go get -d ./...
8 | #echo Building...
9 | #go build
10 |
11 | # make the file executable
12 | chmod +x mangodl
13 | # install the executable to /usr/bin
14 | sudo install -m755 mangodl /usr/bin/mangodl
15 |
--------------------------------------------------------------------------------
/go.mod:
--------------------------------------------------------------------------------
1 | module mangodl-git
2 |
3 | go 1.17
4 |
5 | require (
6 | github.com/PuerkitoBio/goquery v1.8.0
7 | github.com/signintech/gopdf v0.10.6
8 | )
9 |
10 | require (
11 | github.com/andybalholm/cascadia v1.3.1 // indirect
12 | github.com/phpdave11/gofpdi v1.0.11 // indirect
13 | github.com/pkg/errors v0.8.1 // indirect
14 | golang.org/x/net v0.0.0-20210916014120-12bc252f5db8 // indirect
15 | )
16 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | # Binaries for programs and plugins
2 | *.exe
3 | *.exe~
4 | *.dll
5 | *.so
6 | *.dylib
7 |
8 | # Test binary, built with `go test -c`
9 | *.test
10 |
11 | # Output of the go coverage tool, specifically when used with LiteIDE
12 | *.out
13 |
14 | # Dependency directories (remove the comment below to include it)
15 | # vendor/
16 |
17 | # personal script to create DMGs
18 | dmg-creator.sh
19 | mangodl.dmg
20 | mangodl-darwin
21 | mangodl.tar.gz
22 |
23 | .idea/workspace.xml
24 |
--------------------------------------------------------------------------------
/mangodl.rb:
--------------------------------------------------------------------------------
1 | class Mangodl < Formula
2 | desc "An easy-to-use cli tool for downloading manga"
3 | homepage "https://github.com/liamtoaldo/mangodl"
4 | url "https://github.com/liamtoaldo/mangodl/releases/download/mangodl-v1.3-mac/mangodl.tar.gz"
5 | sha256 "5213bd5b4a3faa2246ec98ad0a3e772989bba9957294ffd5c81b2b48b87d243f"
6 | license "GPL-3.0-only"
7 |
8 | depends_on "go" => :build
9 |
10 | def install
11 | system "go", "build", "-o", "/usr/local/bin"
12 | end
13 |
14 | test do
15 | system "mangodl", "--help"
16 | end
17 | end
18 |
--------------------------------------------------------------------------------
/download/download.go:
--------------------------------------------------------------------------------
1 | package download
2 |
3 | import (
4 | "errors"
5 | "io"
6 | "net/http"
7 | "os"
8 | )
9 |
10 | //This function is used to download a file in the given url
11 | func DownloadFile(URL, fileName string) error {
12 | if URL == "" {
13 | return errors.New("couldn't download this image, url not existing")
14 | }
15 | response, err := http.Get(URL)
16 | if err != nil {
17 | return err
18 | }
19 | defer response.Body.Close()
20 |
21 | if response.StatusCode != 200 {
22 | return errors.New("received non 200 response code")
23 | }
24 | file, err := os.Create(fileName)
25 | if err != nil {
26 | return err
27 | }
28 | defer file.Close()
29 |
30 | _, err = io.Copy(file, response.Body)
31 | if err != nil {
32 | return err
33 | }
34 |
35 | return nil
36 | }
37 |
--------------------------------------------------------------------------------
/output/cbz.go:
--------------------------------------------------------------------------------
1 | package output
2 |
3 | import (
4 | "archive/zip"
5 | "io"
6 | "log"
7 | "os"
8 | )
9 |
10 | func ConvertToCBZ(inputImages []string, outputCBZ string) {
11 | cbz, err := os.Create(outputCBZ)
12 | if err != nil {
13 | log.Println(err)
14 | }
15 | defer cbz.Close()
16 |
17 | zipStream := zip.NewWriter(cbz)
18 | defer zipStream.Close()
19 | for _, page := range inputImages {
20 | if AddPage(zipStream, page) != nil {
21 | log.Println(err)
22 | }
23 | }
24 | }
25 | func AddPage(zipStream *zip.Writer, filename string) error {
26 |
27 | fileToZip, err := os.Open(filename)
28 | if err != nil {
29 | return err
30 | }
31 | defer func(fileToZip *os.File) {
32 | err := fileToZip.Close()
33 | if err != nil {
34 | log.Println(err)
35 | }
36 | }(fileToZip)
37 |
38 | info, err := fileToZip.Stat()
39 | if err != nil {
40 | return err
41 | }
42 |
43 | header, err := zip.FileInfoHeader(info)
44 | if err != nil {
45 | return err
46 | }
47 | header.Method = zip.Deflate
48 |
49 | writer, err := zipStream.CreateHeader(header)
50 | if err != nil {
51 | return err
52 | }
53 | _, err = io.Copy(writer, fileToZip)
54 | return err
55 | }
56 |
--------------------------------------------------------------------------------
/go.sum:
--------------------------------------------------------------------------------
1 | github.com/PuerkitoBio/goquery v1.8.0 h1:PJTF7AmFCFKk1N6V6jmKfrNH9tV5pNE6lZMkG0gta/U=
2 | github.com/PuerkitoBio/goquery v1.8.0/go.mod h1:ypIiRMtY7COPGk+I/YbZLbxsxn9g5ejnI2HSMtkjZvI=
3 | github.com/andybalholm/cascadia v1.3.1 h1:nhxRkql1kdYCc8Snf7D5/D3spOX+dBgjA6u8x004T2c=
4 | github.com/andybalholm/cascadia v1.3.1/go.mod h1:R4bJ1UQfqADjvDa4P6HZHLh/3OxWWEqc0Sk8XGwHqvA=
5 | github.com/phpdave11/gofpdi v1.0.11 h1:wsBNx+3S0wy1dEp6fzv281S74ogZGgIdYWV2PugWgho=
6 | github.com/phpdave11/gofpdi v1.0.11/go.mod h1:vBmVV0Do6hSBHC8uKUQ71JGW+ZGQq74llk/7bXwjDoI=
7 | github.com/pkg/errors v0.8.1 h1:iURUrRGxPUNPdy5/HRSm+Yj6okJ6UtLINN0Q9M4+h3I=
8 | github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
9 | github.com/signintech/gopdf v0.10.6 h1:E4aQqEjgCABH9ved+VbU5cneJPQ6m0PRHxJjGE2rJMI=
10 | github.com/signintech/gopdf v0.10.6/go.mod h1:PXwitUSeFWEWs+wHVjSS3cUmD4PTXB686ozqfDIQQoQ=
11 | golang.org/x/net v0.0.0-20210916014120-12bc252f5db8 h1:/6y1LfuqNuQdHAm0jjtPtgRcxIxjVZgm5OTu8/QhZvk=
12 | golang.org/x/net v0.0.0-20210916014120-12bc252f5db8/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y=
13 | golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
14 | golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
15 | golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
16 | golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
17 | golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
18 |
--------------------------------------------------------------------------------
/output/pdf.go:
--------------------------------------------------------------------------------
1 | package output
2 |
3 | import (
4 | "bufio"
5 | "github.com/signintech/gopdf"
6 | "image"
7 | "io/ioutil"
8 | "log"
9 | "os"
10 | "strings"
11 | )
12 |
13 | //ConvertToPDF takes as input the path of the files and creates a PDF with the images of the pages
14 | func ConvertToPDF(inputImages []string, outputPDF string) {
15 | pdf := gopdf.GoPdf{}
16 | pagesWidth, pagesHeight := getSizes(inputImages[0]) //GetSizes is not precise by default, so dividing the sizes by 1.8 the page should return to the normal size
17 | pdf.Start(gopdf.Config{PageSize: gopdf.Rect{W: float64(pagesWidth) / 1.8, H: float64(pagesHeight) / 1.8}}) //595.28, 841.89 = A4
18 | for _, page := range inputImages {
19 | currentPageWidth, currentPageHeight := getSizes(page)
20 | pdf.AddPageWithOption(gopdf.PageOption{
21 | PageSize: &gopdf.Rect{W: float64(currentPageWidth) / 1.8, H: float64(currentPageHeight) / 1.8},
22 | })
23 | err := pdf.Image(page, 0, 0, nil)
24 | if err != nil {
25 | log.Print(err)
26 | }
27 | }
28 | err := pdf.WritePdf(outputPDF)
29 | if err != nil {
30 | log.Print(err)
31 | }
32 | }
33 |
34 | func getSizes(file string) (width int, height int) {
35 | //open the file and create a reader
36 | inputFile, _ := os.Open(file)
37 | reader := bufio.NewReader(inputFile)
38 |
39 | // Decode image.
40 | img, _, _ := image.DecodeConfig(reader)
41 |
42 | return img.Width, img.Height
43 | }
44 |
45 | func GetNumberOfPages(path string) int {
46 | files, _ := ioutil.ReadDir(path)
47 | length := 0
48 | for _, file := range files {
49 | if strings.Contains(file.Name(), "jpg") {
50 | length++
51 | }
52 | }
53 | return length
54 | }
55 |
--------------------------------------------------------------------------------
/utils/conf.go:
--------------------------------------------------------------------------------
1 | package utils
2 |
3 | import (
4 | "encoding/json"
5 | "io/ioutil"
6 | "log"
7 | "os"
8 | "runtime"
9 | )
10 |
11 | var home, _ = os.UserHomeDir()
12 | var configFile = home + "/.config/mangodl.conf"
13 |
14 | type Config struct {
15 | Directory string `json:"directory"`
16 | Output string `json:"output"`
17 | }
18 |
19 | func check() {
20 | if runtime.GOOS == "windows" {
21 | configFile = home + "/mangodl.conf"
22 | }
23 | _, err := os.Stat(configFile)
24 | if os.IsNotExist(err) {
25 | os.Create(configFile)
26 | defaultJson()
27 | }
28 | }
29 | func ReadJSON() Config {
30 | check()
31 | var config Config
32 | configData, _ := ioutil.ReadFile(configFile)
33 | json.Unmarshal(configData, &config)
34 | return config
35 | }
36 | func WriteJson(dir string, out string) {
37 | check()
38 | //check if the user has put an '/' which is required, otherwise add it to the string
39 | if dir[len(dir)-1] != '/' {
40 | dir += "/"
41 | }
42 | _, err := ioutil.ReadFile(configFile)
43 | var config Config
44 | config.Directory = dir
45 | //check if output is one of the available output files, otherwise, put the default (img)
46 | //TODO if a new one is added, update this:
47 | if out != "pdf" && out != "cbz" {
48 | config.Output = "img"
49 | } else {
50 | config.Output = out
51 | }
52 |
53 | newData, _ := json.MarshalIndent(config, "", " ")
54 | err = ioutil.WriteFile(configFile, newData, 0777)
55 | if err != nil {
56 | log.Println(err)
57 | }
58 | }
59 | func defaultJson() {
60 | check()
61 | configData, err := ioutil.ReadFile(configFile)
62 | var config Config
63 | json.Unmarshal(configData, &config)
64 | if config.Directory != home+"/Downloaded Manga/" && config.Directory != "" {
65 | return
66 | }
67 | //if it's windows, then redirect the downloaded manga to the Desktop, for better usability
68 | if runtime.GOOS == "windows" {
69 | config.Directory = home + "/Desktop/Downloaded Manga/"
70 | } else {
71 | config.Directory = home + "/Downloaded Manga/"
72 | }
73 | config.Output = "img"
74 | //write to the config struct and then write the struct to the file
75 | newData, _ := json.MarshalIndent(config, "", " ")
76 | err = ioutil.WriteFile(configFile, newData, 0777)
77 | if err != nil {
78 | log.Println(err)
79 | }
80 | }
81 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | [![Contributors][contributors-shield]][contributors-url]
2 | [![Forks][forks-shield]][forks-url]
3 | [![Stargazers][stars-shield]][stars-url]
4 | [![Issues][issues-shield]][issues-url]
5 | [![MIT License][license-shield]][license-url]
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 | mangodl
15 |
16 |
17 | Download and search manga right from the terminal!
18 |
19 |
20 | Report Bug ||
21 | Request Feature
22 |
23 |
24 |
25 |
26 | ## Table of Contents
27 |
28 | - [About the Project](#about-the-project)
29 | - [Built With](#built-with)
30 | - [Getting Started](#getting-started)
31 | - [Prerequisites](#prerequisites)
32 | - [Installation](#installation)
33 | - [Linux](#linux)
34 | - [Linux (Build from Source)](#linux-build-from-source)
35 | - [Arch Linux and derivatives](#arch-linux-and-derivatives)
36 | - [Windows](#windows)
37 | - [macOS](#macos)
38 | - [macOS (alternative)](#macos-alternative)
39 | - [Usage](#usage)
40 | - [Reading](#reading)
41 | - [Roadmap](#roadmap)
42 | - [Contributing](#contributing)
43 | - [License](#license)
44 | - [Contact](#contact)
45 | - [Acknowledgements](#acknowledgements)
46 |
47 | ## About The Project
48 |
49 |
50 |
51 | An easy-to-use cli tool for downloading manga
52 |
53 |
54 |
55 |
56 |
57 | ### Built With
58 |
59 | - [Go](https://golang.org)
60 | - [Goquery](https://github.com/PuerkitoBio/goquery)
61 |
62 |
63 |
64 | ## Getting Started
65 |
66 | To get a local copy up and running follow these simple steps.
67 |
68 | ### Prerequisites
69 |
70 | - Go compiler (if you want to build from source)
71 | - Linux, Windows or Mac
72 |
73 | ### Installation
74 |
75 | #### Linux
76 | Download the files _mangodl_ and _install.sh_ from the latest Linux version in the [Releases](https://github.com/liamtoaldo/mangodl/releases)
77 | ```sh
78 | # run the installation script
79 | chmod +x install.sh
80 | ./install.sh
81 | ```
82 |
83 | #### Linux (build from source)
84 | ```sh
85 | # clone and go into the repository
86 | git clone https://github.com/liamtoaldo/mangodl.git
87 | cd mangodl
88 |
89 | # NOW JUST OPEN THE INSTALL.SH SCRIPT AND UNCOMMENT THE COMMENTED LINES
90 | # then run the installation script
91 | chmod +x install.sh
92 | ./install.sh
93 | ```
94 | #### Arch Linux and derivatives
95 | An AUR package is now available.
96 |
97 | Just `yay -S mangodl` or, if you use paru `paru -S mangodl`
98 | #### Windows
99 | Download the executable (mangodl.exe) from the latest Windows version in the [Releases](https://github.com/liamtoaldo/mangodl/releases)
100 | If you just want to use it without installing it, just run mangodl.exe everytime and skip these steps below
101 | ```sh
102 | Open start menu,
103 | 1. Type Edit environment variables
104 | 2. Open the option Edit the system environment variables
105 | 3. Click Environment variables... button
106 | 4. There you see two boxes, in System Variables box find path variable
107 | 5. Click Edit
108 | 6. a window pops up, click New
109 | 7. Type the Directory path of mangodl.exe (Directory means exclude the file name from path)
110 | 8. Click Ok on all open windows and restart the command prompt.
111 | ```
112 |
113 | #### macOS
114 | If you haven't already given the terminal access to the disk, then do it, for further help see [THIS](https://osxdaily.com/2018/10/09/fix-operation-not-permitted-terminal-error-macos/)
115 | Installing via brew (assuming that `/usr/local/bin` is already in the $PATH variable):
116 | ```bash
117 | brew tap liamtoaldo/mangodl
118 | brew install --build-from-source mangodl
119 | ```
120 |
121 | #### macOS (alternative)
122 | Download the executable mangodl-darwin from the latest macOS version in the [Releases](https://github.com/liamtoaldo/mangodl/releases)
123 | ```sh
124 | # rename the executable
125 | mv mangodl-darwin mangodl
126 | chmod +x mangodl
127 | # move the executable to the /usr/local/bin/ path, be aware of not deleting the directory!
128 | sudo mv mangodl /usr/local/bin/mangodl
129 | ```
130 | ## Usage
131 | Usage: mangodl [FLAGS]...
132 |
133 | Arguments and flags:
134 |
135 | -h, --help shows this message and exit
136 |
137 | Needed (one of them):
138 | -D, --download downloads the manga specified after -D (e.g. mangodl -D jojo will search for 10 manga with that name and ask you which one to download)
139 | -S, --search searches for the manga specified after this flag (e.g. mangodl -S "kanojo x kanojo" will search and display the manga found with that name)
140 | -Q, --query show downloaded manga
141 | -Dir, --directory sets the default directory to download manga (e.g. mangodl -Dir "$HOME/Documents/manga/"), otherwise the default one would be "$HOME/Downloaded Manga/" and the Desktop for Windows
142 | -v, --version prints the version number (this was implemented from v1.6)
143 |
144 | Optional:
145 | For -D:
146 | -c, --chapter used to specify the chapter to download (if omitted it will download them all)
147 | -cr, --chapterrange used to specify a range of chapters to download (e.g. mangodl -D "Martial Peak" -cr 1 99 will download chapters from 1 to 99 (included)
148 | -o, --output used to specify the file output of the pages (img, pdf or cbz), e.g. mangodl -D "Tokyo Revengers" -o pdf will create a pdf for every chapter. By default, it's images.
149 | Remember that this flag and any other flags must be used before the chapter selection flag, otherwise they wouldn't be detected
150 | -s, --special used to download "special" chapters too, the ones with floating point values (13.1, 14.7, 99.3, etc). Makes the downloads slower, so use this only if needed
151 | -f, --first used to skip the selection phase and select the first manga found. (e.g. mangodl -D "Chainsaw" -f)
152 |
153 | For -S:
154 | -n, --noplot do not print the plot of searched manga
155 |
156 |
157 |
158 | ## Reading
159 | To read the downloaded pages, I really suggest this free piece of software, which is lightweight and flexible:
160 | ### [OpenComic](https://github.com/ollm/OpenComic)
161 |
162 |
163 | You can simply add the folder "Downloaded Manga" to OpenComic, which is the most recommended thing to do.
164 |
165 | And then it'll open all your manga, divided into chapters.
166 | ## Roadmap
167 |
168 | See the [open issues](https://github.com/liamtoaldo/mangodl/issues) for a list of proposed features (and known issues).
169 |
170 |
171 |
172 | ## Contributing
173 |
174 | Contributions are what make the open source community such an amazing place to be learn, inspire, and create. Any contributions you make are **greatly appreciated**.
175 |
176 | 1. Fork the Project
177 | 2. Create your Feature Branch (`git checkout -b feature/AmazingFeature`)
178 | 3. Commit your Changes (`git commit -m 'Add some AmazingFeature'`)
179 | 4. Push to the Branch (`git push origin feature/AmazingFeature`)
180 | 5. Open a Pull Request
181 |
182 |
183 |
184 | ## License
185 |
186 | Distributed under the GPL 3.0 License. See `LICENSE` for more information.
187 |
188 |
189 |
190 | ## Contact
191 |
192 | Me - [liamtoaldo+gh@gmail.com](mailto:liamtoaldo+gh@gmail.com)
193 |
194 | Project Link: [https://github.com/liamtoaldo/mangodl](https://github.com/liamtoaldo/mangodl)
195 |
196 |
197 |
198 | ## Acknowledgements
199 |
200 | - [Myself for doing everything.](https://github.com/liamtoaldo)
201 |
202 |
203 |
204 |
205 | [contributors-shield]: https://img.shields.io/github/contributors/liamtoaldo/mangodl.svg?style=flat-square
206 | [contributors-url]: https://github.com/liamtoaldo/mangodl/graphs/contributors
207 | [forks-shield]: https://img.shields.io/github/forks/liamtoaldo/mangodl.svg?style=flat-square
208 | [forks-url]: https://github.com/liamtoaldo/mangodl/network/members
209 | [stars-shield]: https://img.shields.io/github/stars/liamtoaldo/mangodl.svg?style=flat-square
210 | [stars-url]: https://github.com/liamtoaldo/mangodl/stargazers
211 | [issues-shield]: https://img.shields.io/github/issues/liamtoaldo/mangodl.svg?style=flat-square
212 | [issues-url]: https://github.com/liamtoaldo/mangodl/issues
213 | [license-shield]: https://img.shields.io/github/license/liamtoaldo/mangodl.svg?style=flat-square
214 | [license-url]: https://github.com/liamtoaldo/mangodl/blob/main/LICENSE
215 |
216 |
--------------------------------------------------------------------------------
/utils/utils.go:
--------------------------------------------------------------------------------
1 | package utils
2 |
3 | import (
4 | "fmt"
5 | "github.com/PuerkitoBio/goquery"
6 | "io/ioutil"
7 | "log"
8 | dl "mangodl-git/download"
9 | outl "mangodl-git/output"
10 | "math"
11 | "net/http"
12 | "os"
13 | "runtime"
14 | "sort"
15 | "strconv"
16 | "strings"
17 | "time"
18 | )
19 |
20 | const (
21 | HELPFLAG = "-h"
22 | HELPFLAGALT = "--help"
23 | DOWNLOADFLAG = "-D"
24 | DOWNLOADFLAGALT = "--download"
25 | SEARCHFLAG = "-S"
26 | SEARCHFLAGALT = "--search"
27 | QUERYFLAG = "-Q"
28 | QUERYFLAGALT = "--query"
29 | DIRECTORYFLAG = "-Dir"
30 | DIRECTORYFLAGALT = "--directory"
31 | CHAPTERFLAG = "-c"
32 | CHAPTERFLAGALT = "--chapter"
33 | CHAPTERRANGEFLAG = "-cr"
34 | CHAPTERRANGEFLAGALT = "--chapterrange"
35 | NOPLOTFLAG = "-n"
36 | NOPLOTFLAGALT = "--noplot"
37 | OUTPUTFLAG = "-o"
38 | OUTPUTFLAGALT = "--output"
39 | SPECIALFLAG = "-s"
40 | SPECIALFLAGALT = "--special"
41 | SELECTFIRSTFLAG = "-f"
42 | SELECTFIRSTFLAGALT = "--first"
43 | VERSIONFLAG = "-v"
44 | VERSIONFLAGALT = "--version"
45 | )
46 |
47 | //the variables based on the user's results
48 | var (
49 | mangaName string
50 | selectedMangaID string
51 | realMangaName string
52 | currentState byte //D for downloading, S for searching, Q for querying, H for help, E for error, F for directory/folder, V for version, O for changing output in config
53 | foundMangaIDs []string
54 | foundMangaNames []string
55 | chosenDirectory string
56 | alreadyChecked = false
57 |
58 | //optional
59 | chapterState string //single or multiple or all
60 | singleChapter string
61 | chapterBegin string
62 | chapterEnd string
63 | plotState string //no or yes
64 | output = ReadJSON().Output //img, pdf or cbz, default is image
65 | special = false
66 | selectFirst = false
67 | //TODO always change this value when the version is updated
68 | version = 1.
69 | )
70 |
71 | type DownloadedManga struct {
72 | title string
73 | chapters []float64
74 | }
75 |
76 | func showHelp() {
77 | fmt.Println(`Usage: mangodl [FLAGS]...
78 | Download manga using the terminal. The manga list is really big.
79 |
80 | Arguments and flags:
81 |
82 | -h, --help shows this message and exit
83 |
84 | Needed (one of them):
85 | -D, --download downloads the manga specified after -D (e.g. mangodl -D jojo will search for 10 manga with that name and ask you which one to download)
86 | -S, --search searches for the manga specified after this flag (e.g. mangodl -S "kanojo x kanojo" will search and display the manga found with that name)
87 | -Q, --query show downloaded manga
88 | -Dir, --directory sets the default directory to download manga (e.g. mangodl -Dir "$HOME/Documents/manga/"), otherwise the default one would be "$HOME/Downloaded Manga/" and the Desktop for Windows
89 | -v, --version prints the version number (this was implemented from v1.6)
90 |
91 | Optional:
92 | For -D:
93 | -c, --chapter used to specify the chapter to download (if omitted it will download them all)
94 | -cr, --chapterrange used to specify a range of chapters to download (e.g. mangodl -D "Martial Peak" -cr 1 99 will download chapters from 1 to 99 (included)
95 | -o, --output used to specify the file output of the pages (img, pdf or cbz), e.g. mangodl -D "Tokyo Revengers" -o pdf will create a pdf for every chapter. By default, it's images.
96 | This will also change the default output file format in the config.
97 | Remember that this flag and any other flags must be used before the chapter selection flag, otherwise they wouldn't be detected
98 | -s, --special used to download "special" chapters too, the ones with floating point values (13.1, 14.7, 99.3, etc). Makes the downloads slower, so use this only if needed
99 | -f, --first used to skip the selection phase and select the first manga found. (e.g. mangodl -D "Chainsaw" -f)
100 |
101 | For -S:
102 | -n, --noplot do not print the plot of searched manga
103 | `)
104 | }
105 |
106 | func showVersion() {
107 | fmt.Printf("mangodl v%v", version)
108 | }
109 |
110 | func isNextArg(index int) bool {
111 | if index >= len(os.Args) {
112 | return false
113 | }
114 | return true
115 | }
116 |
117 | //check and redirect the "states"
118 | func checkArgs() {
119 | if len(os.Args) <= 1 {
120 | currentState = 'H'
121 | return
122 | }
123 | //detect if the user is running as sudo
124 | home, _ := os.UserHomeDir()
125 | if home == "/root" {
126 | fmt.Println("Avoid running mangodl as sudo")
127 | return
128 | }
129 | for i, s := range os.Args {
130 | //Help
131 | if s == HELPFLAG || s == HELPFLAGALT {
132 | currentState = 'H'
133 | return
134 | }
135 | //Download
136 | if s == DOWNLOADFLAG || s == DOWNLOADFLAGALT {
137 | if !isNextArg(i + 1) {
138 | currentState = 'E'
139 | fmt.Println("You should specify the manga to download")
140 | return
141 | }
142 | currentState = 'D'
143 | mangaName = os.Args[i+1]
144 | fmt.Println("Attempting to download a manga with that name, for better results, first use -S and search")
145 | continue
146 | }
147 | //Search for existing manga
148 | if s == SEARCHFLAG || s == SEARCHFLAGALT {
149 | if !isNextArg(i + 1) {
150 | currentState = 'E'
151 | fmt.Println("You should specify the manga to search for")
152 | return
153 | }
154 | currentState = 'S'
155 | mangaName = os.Args[i+1]
156 | fmt.Println("Searching the manga")
157 | continue
158 | }
159 | //Query
160 | if s == QUERYFLAG || s == QUERYFLAGALT {
161 | currentState = 'Q'
162 | break
163 | }
164 | //Directory selection
165 | if s == DIRECTORYFLAG || s == DIRECTORYFLAGALT {
166 | currentState = 'F'
167 | chosenDirectory = os.Args[i+1]
168 | break
169 | }
170 | //Version
171 | if s == VERSIONFLAG || s == VERSIONFLAGALT {
172 | currentState = 'V'
173 | }
174 | //Skip selection (first)
175 | if (s == SELECTFIRSTFLAG || s == SELECTFIRSTFLAGALT) && currentState == 'D' {
176 | selectFirst = true
177 | }
178 | //chapters
179 | if s == CHAPTERFLAG || s == CHAPTERFLAGALT {
180 | if !isNextArg(i + 1) {
181 | currentState = 'E'
182 | fmt.Println("Not enough chapter arguments")
183 | return
184 | }
185 | chapterState = "single"
186 | singleChapter = os.Args[i+1]
187 | break
188 | } else if s == CHAPTERRANGEFLAG || s == CHAPTERRANGEFLAGALT {
189 |
190 | if !isNextArg(i+1) && !isNextArg(i+2) {
191 | currentState = 'E'
192 | fmt.Println("Not enough chapter arguments")
193 | return
194 | }
195 | chapterState = "multiple"
196 | chapterBegin = os.Args[i+1]
197 | chapterEnd = os.Args[i+2]
198 | break
199 | } else {
200 | chapterState = "all"
201 | }
202 |
203 | //manga plot
204 | if s == NOPLOTFLAG || s == NOPLOTFLAGALT {
205 | plotState = "no"
206 | } else {
207 | plotState = "yes"
208 | }
209 | //output
210 | if s == OUTPUTFLAG || s == OUTPUTFLAGALT {
211 | if !isNextArg(i + 1) {
212 | currentState = 'E'
213 | fmt.Println("You should specify the output (img or pdf)")
214 | return
215 | }
216 | output = os.Args[i+1]
217 | WriteJson(ReadJSON().Directory, output)
218 | }
219 | //special chapters
220 | if s == SPECIALFLAG || s == SPECIALFLAGALT {
221 | special = true
222 | }
223 | }
224 | }
225 |
226 | //looks for manga and displays them (10 for download and 10 for just searching)
227 | func search(howMany int) {
228 | URL := fmt.Sprintf("https://ww.mangakakalot.tv/search/" + mangaName)
229 | currentState = 'D'
230 | res, err := http.Get(URL)
231 | if err != nil {
232 | log.Println("Unable to connect to website, error: ", err)
233 | }
234 |
235 | doc, _ := goquery.NewDocumentFromReader(res.Body)
236 | fmt.Println("Found: ")
237 |
238 | counter := 1 //the variable that counts the number of manga listed in the search.
239 | previousI := 0 //the previous index, it's used to avoid doubles
240 | doc.Find("a").EachWithBreak(func(i int, selection *goquery.Selection) bool {
241 | selectedMangaID, _ = selection.Attr("href")
242 | //this condition checks if the current analyzed manga has the tag title (which means it's not a searched manga) and if the i is greater than the previous i + 1 (to avoid duplicates) but, because of this, the first option gets ignored, so I just added another condition
243 | if strings.Contains(selectedMangaID, "/manga/") && selection.AttrOr("title", "y") == "y" && counter <= howMany && ((counter > 1 && i > previousI+1) || counter == 1) {
244 | selectedMangaID = strings.Split(selectedMangaID, "/")[2]
245 | foundMangaIDs = append(foundMangaIDs, selectedMangaID)
246 |
247 | //search for the manga name, print the first 10 entries and let the user decide
248 | URL := fmt.Sprintf("https://ww.mangakakalot.tv/manga/" + selectedMangaID)
249 | currentState = 'D'
250 | res, err := http.Get(URL)
251 | if err != nil {
252 | log.Println("Unable to connect to website, error: ", err)
253 | }
254 | doc, _ := goquery.NewDocumentFromReader(res.Body)
255 |
256 | doc.Find("p").Each(func(j int, selection *goquery.Selection) {
257 | if strings.Contains(selection.Text(), "summary:") {
258 | realMangaName = strings.Trim(strings.Replace(selection.Text(), " summary:", "", -1), " ")
259 | foundMangaNames = append(foundMangaNames, realMangaName)
260 | //If the skip flag is selected, skip other manga. I placed this code just after the first manga selection.
261 | if selectFirst {
262 | return
263 | }
264 | fmt.Println("["+fmt.Sprint(counter)+"] -", realMangaName)
265 | counter++
266 | }
267 | })
268 | //If the skip flag is selected, skip other manga and just download the first possible.
269 | if selectFirst {
270 | return false
271 | }
272 | //added wait option because the host has become slow, and requests to it need to be slow as well
273 | time.Sleep(300 * time.Millisecond)
274 | //plot
275 | if plotState == "yes" {
276 | doc.Find("#noidungm").Each(func(j int, selection *goquery.Selection) {
277 | if strings.Contains(selection.Text(), "summary:") {
278 | fmt.Println(strings.Trim(strings.Replace(strings.Replace(strings.Replace(strings.Trim(selection.Text(), " "), realMangaName, "", -1), "\n", "", -1), " summary: ", "", -1), " "))
279 | fmt.Println()
280 | }
281 | })
282 | }
283 | previousI = i
284 | return true
285 | }
286 |
287 | return true
288 |
289 | })
290 | }
291 |
292 | //the function to let the user choose the manga
293 | func chooseManga() {
294 | //If the select first flag is true, skip the manga selection and take the first manga.
295 | if selectFirst {
296 | selectedMangaID = foundMangaIDs[0]
297 | realMangaName = foundMangaNames[0]
298 | } else {
299 | fmt.Println("Which Manga do you want to download?")
300 | var inputChoice int
301 | fmt.Scan(&inputChoice)
302 | switch inputChoice {
303 | case 1:
304 | selectedMangaID = foundMangaIDs[0]
305 | realMangaName = foundMangaNames[0]
306 | case 2:
307 | selectedMangaID = foundMangaIDs[1]
308 | realMangaName = foundMangaNames[1]
309 | case 3:
310 | selectedMangaID = foundMangaIDs[2]
311 | realMangaName = foundMangaNames[2]
312 | case 4:
313 | selectedMangaID = foundMangaIDs[3]
314 | realMangaName = foundMangaNames[3]
315 | case 5:
316 | selectedMangaID = foundMangaIDs[4]
317 | realMangaName = foundMangaNames[4]
318 | case 6:
319 | selectedMangaID = foundMangaIDs[5]
320 | realMangaName = foundMangaNames[5]
321 | case 7:
322 | selectedMangaID = foundMangaIDs[6]
323 | realMangaName = foundMangaNames[6]
324 | case 8:
325 | selectedMangaID = foundMangaIDs[7]
326 | realMangaName = foundMangaNames[7]
327 | case 9:
328 | selectedMangaID = foundMangaIDs[8]
329 | realMangaName = foundMangaNames[8]
330 | case 10:
331 | selectedMangaID = foundMangaIDs[9]
332 | realMangaName = foundMangaNames[9]
333 | }
334 | }
335 | if output != "img" {
336 | fmt.Println("Downloading images to be converted...")
337 | }
338 | }
339 |
340 | //download the chosen manga
341 | func download(chapter float32) bool {
342 | var chapterNumber string
343 | if chapter-float32(int(chapter)) > 0 {
344 | chapterNumber = strconv.FormatFloat(float64(chapter), 'f', 1, 64)
345 | } else {
346 | chapterNumber = strconv.FormatFloat(float64(chapter), 'f', -1, 64)
347 | }
348 | URL := fmt.Sprintf("https://ww.mangakakalot.tv/chapter/%s/chapter-%v", selectedMangaID, fmt.Sprint(chapterNumber))
349 | res, err := http.Get(URL)
350 | if err != nil {
351 | log.Println("Unable to connect to website, error: ", err)
352 | }
353 | doc, _ := goquery.NewDocumentFromReader(res.Body)
354 |
355 | if strings.Contains(realMangaName, ":") && runtime.GOOS == "windows" {
356 | realMangaName = strings.ReplaceAll(realMangaName, ":", " -")
357 | }
358 | dir := ReadJSON().Directory + realMangaName + "/Chapter " + fmt.Sprint(chapterNumber)
359 | err = os.MkdirAll(dir, 0777)
360 | if err != nil {
361 | log.Println(err)
362 | }
363 | selection := doc.Find("span")
364 | if strings.Contains(selection.Text(), "Error") {
365 | //wait for half a second, otherwise the checking would be too fast and would skip some chapters
366 | time.Sleep(300 * time.Millisecond)
367 | //if the chapter doesn't exist, delete the just created directory
368 | err := os.Remove(dir)
369 | if err != nil {
370 | log.Println(err)
371 | }
372 | return false
373 | }
374 |
375 | doc.Find("img").Each(func(i int, selection *goquery.Selection) {
376 | imageURL, _ := selection.Attr("data-src")
377 | fileName := fmt.Sprintf("%s/page%03d.jpg", dir, i)
378 |
379 | err = dl.DownloadFile(imageURL, fileName)
380 | if err == nil && output == "img" {
381 | fmt.Println("Downloading ::", fileName)
382 | }
383 | })
384 | return true
385 | }
386 |
387 | //the function used for --query (shows the downloaded manga)
388 | func showDownloaded() {
389 | var downloaded []DownloadedManga
390 | dir := ReadJSON().Directory
391 | files, err := ioutil.ReadDir(dir)
392 |
393 | if err != nil {
394 | fmt.Println("You don't have any manga downloaded or they couldn't be recognised")
395 | log.Println(err)
396 | }
397 |
398 | for _, f := range files {
399 | if f.IsDir() {
400 | downloaded = append(downloaded, DownloadedManga{
401 | title: f.Name(),
402 | chapters: make([]float64, 0),
403 | })
404 | dir += f.Name()
405 | files, err = ioutil.ReadDir(dir)
406 | for _, f := range files {
407 | if f.IsDir() {
408 | tmp, err := strconv.ParseFloat(strings.Split(f.Name(), " ")[1], 32)
409 | if err != nil {
410 | log.Println(err)
411 | }
412 | downloaded[len(downloaded)-1].chapters = append(downloaded[len(downloaded)-1].chapters, math.Ceil(tmp*100)/100) //rounding the number to the first decimal digit because some chapters have decimals
413 | }
414 | }
415 | dir = ReadJSON().Directory
416 | }
417 | }
418 | for _, m := range downloaded {
419 | //sort the chapters, so that they get displayed correctly ( 1 2 3 ) instead of ( 1 10 11 2 23 3 4 ...)
420 | sort.Float64s(m.chapters)
421 | fmt.Println(m.title, "- Chapters:", m.chapters)
422 | }
423 | }
424 |
425 | //this function checks if the output was redirected to pdf or cbz, and if it was, it takes care of it
426 | func prepareOutput(i float64) {
427 | if output != "img" {
428 | var chapterNumber string
429 | if i-float64(int(i)) > 0 {
430 | chapterNumber = strconv.FormatFloat(i, 'f', 1, 64)
431 | } else {
432 | chapterNumber = strconv.FormatFloat(i, 'f', -1, 64)
433 | }
434 | dir := ReadJSON().Directory + realMangaName + "/Chapter " + chapterNumber
435 | pageNumber := outl.GetNumberOfPages(dir)
436 | var pages []string
437 | for j := 1; j <= pageNumber; j++ {
438 | pages = append(pages, fmt.Sprintf("%s/page%03d.jpg", dir, j))
439 | }
440 | if output == "pdf" {
441 | fmt.Println("Converting the downloaded images to PDF in", fmt.Sprintf("%s/Chapter%v.pdf", dir, chapterNumber))
442 | outl.ConvertToPDF(pages, fmt.Sprintf("%s/Chapter%v.pdf", dir, chapterNumber))
443 | //move the chapter one directory up.
444 | err := os.Rename(fmt.Sprintf("%s/Chapter%v.pdf", dir, chapterNumber), fmt.Sprintf(strings.Replace(dir, fmt.Sprintf("Chapter %v", chapterNumber), "", 1))+fmt.Sprintf("Chapter%v.pdf", chapterNumber))
445 | if err != nil {
446 | log.Println(err)
447 | }
448 | } else if output == "cbz" {
449 | fmt.Println("Converting the downloaded images to CBZ in", fmt.Sprintf("%s/Chapter%v.cbz", dir, chapterNumber))
450 | outl.ConvertToCBZ(pages, fmt.Sprintf("%s/Chapter%v.cbz", dir, chapterNumber))
451 | //move the chapter one directory up.
452 | err := os.Rename(fmt.Sprintf("%s/Chapter%v.cbz", dir, chapterNumber), fmt.Sprintf(strings.Replace(dir, fmt.Sprintf("Chapter %v", chapterNumber), "", 1))+fmt.Sprintf("Chapter%v.cbz", chapterNumber))
453 | if err != nil {
454 | log.Println(err)
455 | }
456 | }
457 | //delete the folder and images, (issue #21)
458 | fmt.Println("Removing previously downloaded images...")
459 | os.RemoveAll(dir)
460 | fmt.Println("Done")
461 | }
462 | }
463 |
464 | //Execute is equivalent to a "main" since it does everything required to run and calls all other private functions
465 | func Execute() {
466 | //Read Output from config
467 | //Check if arguments are inputted correctly and change "states"
468 | checkArgs()
469 | //Put default output
470 | if currentState == 'D' {
471 |
472 | plotState = "no"
473 | search(10) //redirect search in another goroutine
474 | chooseManga()
475 | if chapterState == "all" {
476 | if special {
477 | i := 0
478 | for {
479 | i++
480 | tmpDownloaded := download(float32(i))
481 | if tmpDownloaded {
482 | prepareOutput(float64(i))
483 | }
484 | alreadyChecked = true
485 | for j := 0.1; j <= 0.9; j += 0.1 {
486 | if download(float32(float64(i) + j)) {
487 | alreadyChecked = false
488 | prepareOutput(float64(i) + j)
489 | }
490 | }
491 | if !tmpDownloaded && alreadyChecked {
492 | break
493 | }
494 | }
495 | } else {
496 | i := 0
497 | for {
498 | i++
499 | tmpDownloaded := download(float32(i))
500 | if tmpDownloaded {
501 | prepareOutput(float64(i))
502 | }
503 | if !tmpDownloaded && alreadyChecked {
504 | break
505 | } else if !tmpDownloaded && !alreadyChecked {
506 | fmt.Println("Checking for weird naming conventions...")
507 | for j := 0.1; j <= 0.9; j += 0.1 {
508 | if download(float32(float64(i) + j)) {
509 | prepareOutput(float64(i) + j)
510 | }
511 | }
512 | alreadyChecked = true
513 | }
514 | }
515 | }
516 | } else if chapterState == "multiple" {
517 | if special {
518 | tmp, _ := strconv.ParseFloat(chapterEnd, 32)
519 | for i, _ := strconv.ParseFloat(chapterBegin, 32); i <= tmp; i++ {
520 | tmpDownloaded := download(float32(i))
521 | if tmpDownloaded {
522 | prepareOutput(i)
523 | }
524 | alreadyChecked = true
525 | for j := i - float64(int(i)); j <= 0.9; j += 0.1 {
526 | //if the chapter is a whole number, it gets redownloaded, so add +0.1 to avoid that.
527 | if j == 0.0 {
528 | j = 0.1
529 | }
530 | //out of the loop if the chapter being downloaded is greater than the last chapter.
531 | if i+j >= tmp {
532 | break
533 | }
534 | if download(float32(i + j)) {
535 | alreadyChecked = false
536 | prepareOutput(i + j)
537 | }
538 | }
539 | if !tmpDownloaded && alreadyChecked {
540 | break
541 | }
542 | i = float64(int(i))
543 | }
544 | //checks if the chapterEnd (tmp) is a whole number, if it isn't, then it downloads the last chapter (which is decimal)
545 | if tmp != float64(int64(tmp)) {
546 | if download(float32(tmp)) {
547 | prepareOutput(tmp)
548 | }
549 | }
550 | } else {
551 | tmp, _ := strconv.ParseFloat(chapterEnd, 32)
552 | for i, _ := strconv.ParseFloat(chapterBegin, 32); i <= tmp; i++ {
553 | tmpDownloaded := download(float32(i))
554 | if tmpDownloaded {
555 | prepareOutput(i)
556 | }
557 | if !tmpDownloaded && alreadyChecked {
558 | break
559 | } else if !tmpDownloaded && !alreadyChecked {
560 | fmt.Println("Checking for weird naming conventions...")
561 | for j := 0.1; j <= 0.9; j += 0.1 {
562 | if download(float32(i + j)) {
563 | prepareOutput(i)
564 | }
565 | }
566 | alreadyChecked = true
567 | }
568 | i = float64(int(i))
569 | }
570 | //checks if the chapterEnd (tmp) is a whole number, if it isn't, then it downloads the last chapter (which is decimal)
571 | if tmp != float64(int64(tmp)) {
572 | if download(float32(tmp)) {
573 | prepareOutput(tmp)
574 | }
575 | }
576 | }
577 | } else if chapterState == "single" {
578 | tmp, _ := strconv.ParseFloat(singleChapter, 32)
579 | download(float32(tmp))
580 | prepareOutput(tmp)
581 | }
582 | } else if currentState == 'Q' {
583 | showDownloaded()
584 | } else if currentState == 'F' {
585 | WriteJson(chosenDirectory, ReadJSON().Output)
586 | fmt.Println("Set default directory to", ReadJSON().Directory)
587 | } else if currentState == 'S' {
588 | search(10)
589 | } else if currentState == 'H' {
590 | showHelp()
591 | } else if currentState == 'V' {
592 | showVersion()
593 | } else if currentState == 'E' {
594 | return
595 | } else {
596 | fmt.Println("Unknown command, try mangodl --help for help")
597 | }
598 |
599 | }
600 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | GNU GENERAL PUBLIC LICENSE
2 | Version 3, 29 June 2007
3 |
4 | Copyright (C) 2007 Free Software Foundation, Inc.
5 | Everyone is permitted to copy and distribute verbatim copies
6 | of this license document, but changing it is not allowed.
7 |
8 | Preamble
9 |
10 | The GNU General Public License is a free, copyleft license for
11 | software and other kinds of works.
12 |
13 | The licenses for most software and other practical works are designed
14 | to take away your freedom to share and change the works. By contrast,
15 | the GNU General Public License is intended to guarantee your freedom to
16 | share and change all versions of a program--to make sure it remains free
17 | software for all its users. We, the Free Software Foundation, use the
18 | GNU General Public License for most of our software; it applies also to
19 | any other work released this way by its authors. You can apply it to
20 | your programs, too.
21 |
22 | When we speak of free software, we are referring to freedom, not
23 | price. Our General Public Licenses are designed to make sure that you
24 | have the freedom to distribute copies of free software (and charge for
25 | them if you wish), that you receive source code or can get it if you
26 | want it, that you can change the software or use pieces of it in new
27 | free programs, and that you know you can do these things.
28 |
29 | To protect your rights, we need to prevent others from denying you
30 | these rights or asking you to surrender the rights. Therefore, you have
31 | certain responsibilities if you distribute copies of the software, or if
32 | you modify it: responsibilities to respect the freedom of others.
33 |
34 | For example, if you distribute copies of such a program, whether
35 | gratis or for a fee, you must pass on to the recipients the same
36 | freedoms that you received. You must make sure that they, too, receive
37 | or can get the source code. And you must show them these terms so they
38 | know their rights.
39 |
40 | Developers that use the GNU GPL protect your rights with two steps:
41 | (1) assert copyright on the software, and (2) offer you this License
42 | giving you legal permission to copy, distribute and/or modify it.
43 |
44 | For the developers' and authors' protection, the GPL clearly explains
45 | that there is no warranty for this free software. For both users' and
46 | authors' sake, the GPL requires that modified versions be marked as
47 | changed, so that their problems will not be attributed erroneously to
48 | authors of previous versions.
49 |
50 | Some devices are designed to deny users access to install or run
51 | modified versions of the software inside them, although the manufacturer
52 | can do so. This is fundamentally incompatible with the aim of
53 | protecting users' freedom to change the software. The systematic
54 | pattern of such abuse occurs in the area of products for individuals to
55 | use, which is precisely where it is most unacceptable. Therefore, we
56 | have designed this version of the GPL to prohibit the practice for those
57 | products. If such problems arise substantially in other domains, we
58 | stand ready to extend this provision to those domains in future versions
59 | of the GPL, as needed to protect the freedom of users.
60 |
61 | Finally, every program is threatened constantly by software patents.
62 | States should not allow patents to restrict development and use of
63 | software on general-purpose computers, but in those that do, we wish to
64 | avoid the special danger that patents applied to a free program could
65 | make it effectively proprietary. To prevent this, the GPL assures that
66 | patents cannot be used to render the program non-free.
67 |
68 | The precise terms and conditions for copying, distribution and
69 | modification follow.
70 |
71 | TERMS AND CONDITIONS
72 |
73 | 0. Definitions.
74 |
75 | "This License" refers to version 3 of the GNU General Public License.
76 |
77 | "Copyright" also means copyright-like laws that apply to other kinds of
78 | works, such as semiconductor masks.
79 |
80 | "The Program" refers to any copyrightable work licensed under this
81 | License. Each licensee is addressed as "you". "Licensees" and
82 | "recipients" may be individuals or organizations.
83 |
84 | To "modify" a work means to copy from or adapt all or part of the work
85 | in a fashion requiring copyright permission, other than the making of an
86 | exact copy. The resulting work is called a "modified version" of the
87 | earlier work or a work "based on" the earlier work.
88 |
89 | A "covered work" means either the unmodified Program or a work based
90 | on the Program.
91 |
92 | To "propagate" a work means to do anything with it that, without
93 | permission, would make you directly or secondarily liable for
94 | infringement under applicable copyright law, except executing it on a
95 | computer or modifying a private copy. Propagation includes copying,
96 | distribution (with or without modification), making available to the
97 | public, and in some countries other activities as well.
98 |
99 | To "convey" a work means any kind of propagation that enables other
100 | parties to make or receive copies. Mere interaction with a user through
101 | a computer network, with no transfer of a copy, is not conveying.
102 |
103 | An interactive user interface displays "Appropriate Legal Notices"
104 | to the extent that it includes a convenient and prominently visible
105 | feature that (1) displays an appropriate copyright notice, and (2)
106 | tells the user that there is no warranty for the work (except to the
107 | extent that warranties are provided), that licensees may convey the
108 | work under this License, and how to view a copy of this License. If
109 | the interface presents a list of user commands or options, such as a
110 | menu, a prominent item in the list meets this criterion.
111 |
112 | 1. Source Code.
113 |
114 | The "source code" for a work means the preferred form of the work
115 | for making modifications to it. "Object code" means any non-source
116 | form of a work.
117 |
118 | A "Standard Interface" means an interface that either is an official
119 | standard defined by a recognized standards body, or, in the case of
120 | interfaces specified for a particular programming language, one that
121 | is widely used among developers working in that language.
122 |
123 | The "System Libraries" of an executable work include anything, other
124 | than the work as a whole, that (a) is included in the normal form of
125 | packaging a Major Component, but which is not part of that Major
126 | Component, and (b) serves only to enable use of the work with that
127 | Major Component, or to implement a Standard Interface for which an
128 | implementation is available to the public in source code form. A
129 | "Major Component", in this context, means a major essential component
130 | (kernel, window system, and so on) of the specific operating system
131 | (if any) on which the executable work runs, or a compiler used to
132 | produce the work, or an object code interpreter used to run it.
133 |
134 | The "Corresponding Source" for a work in object code form means all
135 | the source code needed to generate, install, and (for an executable
136 | work) run the object code and to modify the work, including scripts to
137 | control those activities. However, it does not include the work's
138 | System Libraries, or general-purpose tools or generally available free
139 | programs which are used unmodified in performing those activities but
140 | which are not part of the work. For example, Corresponding Source
141 | includes interface definition files associated with source files for
142 | the work, and the source code for shared libraries and dynamically
143 | linked subprograms that the work is specifically designed to require,
144 | such as by intimate data communication or control flow between those
145 | subprograms and other parts of the work.
146 |
147 | The Corresponding Source need not include anything that users
148 | can regenerate automatically from other parts of the Corresponding
149 | Source.
150 |
151 | The Corresponding Source for a work in source code form is that
152 | same work.
153 |
154 | 2. Basic Permissions.
155 |
156 | All rights granted under this License are granted for the term of
157 | copyright on the Program, and are irrevocable provided the stated
158 | conditions are met. This License explicitly affirms your unlimited
159 | permission to run the unmodified Program. The output from running a
160 | covered work is covered by this License only if the output, given its
161 | content, constitutes a covered work. This License acknowledges your
162 | rights of fair use or other equivalent, as provided by copyright law.
163 |
164 | You may make, run and propagate covered works that you do not
165 | convey, without conditions so long as your license otherwise remains
166 | in force. You may convey covered works to others for the sole purpose
167 | of having them make modifications exclusively for you, or provide you
168 | with facilities for running those works, provided that you comply with
169 | the terms of this License in conveying all material for which you do
170 | not control copyright. Those thus making or running the covered works
171 | for you must do so exclusively on your behalf, under your direction
172 | and control, on terms that prohibit them from making any copies of
173 | your copyrighted material outside their relationship with you.
174 |
175 | Conveying under any other circumstances is permitted solely under
176 | the conditions stated below. Sublicensing is not allowed; section 10
177 | makes it unnecessary.
178 |
179 | 3. Protecting Users' Legal Rights From Anti-Circumvention Law.
180 |
181 | No covered work shall be deemed part of an effective technological
182 | measure under any applicable law fulfilling obligations under article
183 | 11 of the WIPO copyright treaty adopted on 20 December 1996, or
184 | similar laws prohibiting or restricting circumvention of such
185 | measures.
186 |
187 | When you convey a covered work, you waive any legal power to forbid
188 | circumvention of technological measures to the extent such circumvention
189 | is effected by exercising rights under this License with respect to
190 | the covered work, and you disclaim any intention to limit operation or
191 | modification of the work as a means of enforcing, against the work's
192 | users, your or third parties' legal rights to forbid circumvention of
193 | technological measures.
194 |
195 | 4. Conveying Verbatim Copies.
196 |
197 | You may convey verbatim copies of the Program's source code as you
198 | receive it, in any medium, provided that you conspicuously and
199 | appropriately publish on each copy an appropriate copyright notice;
200 | keep intact all notices stating that this License and any
201 | non-permissive terms added in accord with section 7 apply to the code;
202 | keep intact all notices of the absence of any warranty; and give all
203 | recipients a copy of this License along with the Program.
204 |
205 | You may charge any price or no price for each copy that you convey,
206 | and you may offer support or warranty protection for a fee.
207 |
208 | 5. Conveying Modified Source Versions.
209 |
210 | You may convey a work based on the Program, or the modifications to
211 | produce it from the Program, in the form of source code under the
212 | terms of section 4, provided that you also meet all of these conditions:
213 |
214 | a) The work must carry prominent notices stating that you modified
215 | it, and giving a relevant date.
216 |
217 | b) The work must carry prominent notices stating that it is
218 | released under this License and any conditions added under section
219 | 7. This requirement modifies the requirement in section 4 to
220 | "keep intact all notices".
221 |
222 | c) You must license the entire work, as a whole, under this
223 | License to anyone who comes into possession of a copy. This
224 | License will therefore apply, along with any applicable section 7
225 | additional terms, to the whole of the work, and all its parts,
226 | regardless of how they are packaged. This License gives no
227 | permission to license the work in any other way, but it does not
228 | invalidate such permission if you have separately received it.
229 |
230 | d) If the work has interactive user interfaces, each must display
231 | Appropriate Legal Notices; however, if the Program has interactive
232 | interfaces that do not display Appropriate Legal Notices, your
233 | work need not make them do so.
234 |
235 | A compilation of a covered work with other separate and independent
236 | works, which are not by their nature extensions of the covered work,
237 | and which are not combined with it such as to form a larger program,
238 | in or on a volume of a storage or distribution medium, is called an
239 | "aggregate" if the compilation and its resulting copyright are not
240 | used to limit the access or legal rights of the compilation's users
241 | beyond what the individual works permit. Inclusion of a covered work
242 | in an aggregate does not cause this License to apply to the other
243 | parts of the aggregate.
244 |
245 | 6. Conveying Non-Source Forms.
246 |
247 | You may convey a covered work in object code form under the terms
248 | of sections 4 and 5, provided that you also convey the
249 | machine-readable Corresponding Source under the terms of this License,
250 | in one of these ways:
251 |
252 | a) Convey the object code in, or embodied in, a physical product
253 | (including a physical distribution medium), accompanied by the
254 | Corresponding Source fixed on a durable physical medium
255 | customarily used for software interchange.
256 |
257 | b) Convey the object code in, or embodied in, a physical product
258 | (including a physical distribution medium), accompanied by a
259 | written offer, valid for at least three years and valid for as
260 | long as you offer spare parts or customer support for that product
261 | model, to give anyone who possesses the object code either (1) a
262 | copy of the Corresponding Source for all the software in the
263 | product that is covered by this License, on a durable physical
264 | medium customarily used for software interchange, for a price no
265 | more than your reasonable cost of physically performing this
266 | conveying of source, or (2) access to copy the
267 | Corresponding Source from a network server at no charge.
268 |
269 | c) Convey individual copies of the object code with a copy of the
270 | written offer to provide the Corresponding Source. This
271 | alternative is allowed only occasionally and noncommercially, and
272 | only if you received the object code with such an offer, in accord
273 | with subsection 6b.
274 |
275 | d) Convey the object code by offering access from a designated
276 | place (gratis or for a charge), and offer equivalent access to the
277 | Corresponding Source in the same way through the same place at no
278 | further charge. You need not require recipients to copy the
279 | Corresponding Source along with the object code. If the place to
280 | copy the object code is a network server, the Corresponding Source
281 | may be on a different server (operated by you or a third party)
282 | that supports equivalent copying facilities, provided you maintain
283 | clear directions next to the object code saying where to find the
284 | Corresponding Source. Regardless of what server hosts the
285 | Corresponding Source, you remain obligated to ensure that it is
286 | available for as long as needed to satisfy these requirements.
287 |
288 | e) Convey the object code using peer-to-peer transmission, provided
289 | you inform other peers where the object code and Corresponding
290 | Source of the work are being offered to the general public at no
291 | charge under subsection 6d.
292 |
293 | A separable portion of the object code, whose source code is excluded
294 | from the Corresponding Source as a System Library, need not be
295 | included in conveying the object code work.
296 |
297 | A "User Product" is either (1) a "consumer product", which means any
298 | tangible personal property which is normally used for personal, family,
299 | or household purposes, or (2) anything designed or sold for incorporation
300 | into a dwelling. In determining whether a product is a consumer product,
301 | doubtful cases shall be resolved in favor of coverage. For a particular
302 | product received by a particular user, "normally used" refers to a
303 | typical or common use of that class of product, regardless of the status
304 | of the particular user or of the way in which the particular user
305 | actually uses, or expects or is expected to use, the product. A product
306 | is a consumer product regardless of whether the product has substantial
307 | commercial, industrial or non-consumer uses, unless such uses represent
308 | the only significant mode of use of the product.
309 |
310 | "Installation Information" for a User Product means any methods,
311 | procedures, authorization keys, or other information required to install
312 | and execute modified versions of a covered work in that User Product from
313 | a modified version of its Corresponding Source. The information must
314 | suffice to ensure that the continued functioning of the modified object
315 | code is in no case prevented or interfered with solely because
316 | modification has been made.
317 |
318 | If you convey an object code work under this section in, or with, or
319 | specifically for use in, a User Product, and the conveying occurs as
320 | part of a transaction in which the right of possession and use of the
321 | User Product is transferred to the recipient in perpetuity or for a
322 | fixed term (regardless of how the transaction is characterized), the
323 | Corresponding Source conveyed under this section must be accompanied
324 | by the Installation Information. But this requirement does not apply
325 | if neither you nor any third party retains the ability to install
326 | modified object code on the User Product (for example, the work has
327 | been installed in ROM).
328 |
329 | The requirement to provide Installation Information does not include a
330 | requirement to continue to provide support service, warranty, or updates
331 | for a work that has been modified or installed by the recipient, or for
332 | the User Product in which it has been modified or installed. Access to a
333 | network may be denied when the modification itself materially and
334 | adversely affects the operation of the network or violates the rules and
335 | protocols for communication across the network.
336 |
337 | Corresponding Source conveyed, and Installation Information provided,
338 | in accord with this section must be in a format that is publicly
339 | documented (and with an implementation available to the public in
340 | source code form), and must require no special password or key for
341 | unpacking, reading or copying.
342 |
343 | 7. Additional Terms.
344 |
345 | "Additional permissions" are terms that supplement the terms of this
346 | License by making exceptions from one or more of its conditions.
347 | Additional permissions that are applicable to the entire Program shall
348 | be treated as though they were included in this License, to the extent
349 | that they are valid under applicable law. If additional permissions
350 | apply only to part of the Program, that part may be used separately
351 | under those permissions, but the entire Program remains governed by
352 | this License without regard to the additional permissions.
353 |
354 | When you convey a copy of a covered work, you may at your option
355 | remove any additional permissions from that copy, or from any part of
356 | it. (Additional permissions may be written to require their own
357 | removal in certain cases when you modify the work.) You may place
358 | additional permissions on material, added by you to a covered work,
359 | for which you have or can give appropriate copyright permission.
360 |
361 | Notwithstanding any other provision of this License, for material you
362 | add to a covered work, you may (if authorized by the copyright holders of
363 | that material) supplement the terms of this License with terms:
364 |
365 | a) Disclaiming warranty or limiting liability differently from the
366 | terms of sections 15 and 16 of this License; or
367 |
368 | b) Requiring preservation of specified reasonable legal notices or
369 | author attributions in that material or in the Appropriate Legal
370 | Notices displayed by works containing it; or
371 |
372 | c) Prohibiting misrepresentation of the origin of that material, or
373 | requiring that modified versions of such material be marked in
374 | reasonable ways as different from the original version; or
375 |
376 | d) Limiting the use for publicity purposes of names of licensors or
377 | authors of the material; or
378 |
379 | e) Declining to grant rights under trademark law for use of some
380 | trade names, trademarks, or service marks; or
381 |
382 | f) Requiring indemnification of licensors and authors of that
383 | material by anyone who conveys the material (or modified versions of
384 | it) with contractual assumptions of liability to the recipient, for
385 | any liability that these contractual assumptions directly impose on
386 | those licensors and authors.
387 |
388 | All other non-permissive additional terms are considered "further
389 | restrictions" within the meaning of section 10. If the Program as you
390 | received it, or any part of it, contains a notice stating that it is
391 | governed by this License along with a term that is a further
392 | restriction, you may remove that term. If a license document contains
393 | a further restriction but permits relicensing or conveying under this
394 | License, you may add to a covered work material governed by the terms
395 | of that license document, provided that the further restriction does
396 | not survive such relicensing or conveying.
397 |
398 | If you add terms to a covered work in accord with this section, you
399 | must place, in the relevant source files, a statement of the
400 | additional terms that apply to those files, or a notice indicating
401 | where to find the applicable terms.
402 |
403 | Additional terms, permissive or non-permissive, may be stated in the
404 | form of a separately written license, or stated as exceptions;
405 | the above requirements apply either way.
406 |
407 | 8. Termination.
408 |
409 | You may not propagate or modify a covered work except as expressly
410 | provided under this License. Any attempt otherwise to propagate or
411 | modify it is void, and will automatically terminate your rights under
412 | this License (including any patent licenses granted under the third
413 | paragraph of section 11).
414 |
415 | However, if you cease all violation of this License, then your
416 | license from a particular copyright holder is reinstated (a)
417 | provisionally, unless and until the copyright holder explicitly and
418 | finally terminates your license, and (b) permanently, if the copyright
419 | holder fails to notify you of the violation by some reasonable means
420 | prior to 60 days after the cessation.
421 |
422 | Moreover, your license from a particular copyright holder is
423 | reinstated permanently if the copyright holder notifies you of the
424 | violation by some reasonable means, this is the first time you have
425 | received notice of violation of this License (for any work) from that
426 | copyright holder, and you cure the violation prior to 30 days after
427 | your receipt of the notice.
428 |
429 | Termination of your rights under this section does not terminate the
430 | licenses of parties who have received copies or rights from you under
431 | this License. If your rights have been terminated and not permanently
432 | reinstated, you do not qualify to receive new licenses for the same
433 | material under section 10.
434 |
435 | 9. Acceptance Not Required for Having Copies.
436 |
437 | You are not required to accept this License in order to receive or
438 | run a copy of the Program. Ancillary propagation of a covered work
439 | occurring solely as a consequence of using peer-to-peer transmission
440 | to receive a copy likewise does not require acceptance. However,
441 | nothing other than this License grants you permission to propagate or
442 | modify any covered work. These actions infringe copyright if you do
443 | not accept this License. Therefore, by modifying or propagating a
444 | covered work, you indicate your acceptance of this License to do so.
445 |
446 | 10. Automatic Licensing of Downstream Recipients.
447 |
448 | Each time you convey a covered work, the recipient automatically
449 | receives a license from the original licensors, to run, modify and
450 | propagate that work, subject to this License. You are not responsible
451 | for enforcing compliance by third parties with this License.
452 |
453 | An "entity transaction" is a transaction transferring control of an
454 | organization, or substantially all assets of one, or subdividing an
455 | organization, or merging organizations. If propagation of a covered
456 | work results from an entity transaction, each party to that
457 | transaction who receives a copy of the work also receives whatever
458 | licenses to the work the party's predecessor in interest had or could
459 | give under the previous paragraph, plus a right to possession of the
460 | Corresponding Source of the work from the predecessor in interest, if
461 | the predecessor has it or can get it with reasonable efforts.
462 |
463 | You may not impose any further restrictions on the exercise of the
464 | rights granted or affirmed under this License. For example, you may
465 | not impose a license fee, royalty, or other charge for exercise of
466 | rights granted under this License, and you may not initiate litigation
467 | (including a cross-claim or counterclaim in a lawsuit) alleging that
468 | any patent claim is infringed by making, using, selling, offering for
469 | sale, or importing the Program or any portion of it.
470 |
471 | 11. Patents.
472 |
473 | A "contributor" is a copyright holder who authorizes use under this
474 | License of the Program or a work on which the Program is based. The
475 | work thus licensed is called the contributor's "contributor version".
476 |
477 | A contributor's "essential patent claims" are all patent claims
478 | owned or controlled by the contributor, whether already acquired or
479 | hereafter acquired, that would be infringed by some manner, permitted
480 | by this License, of making, using, or selling its contributor version,
481 | but do not include claims that would be infringed only as a
482 | consequence of further modification of the contributor version. For
483 | purposes of this definition, "control" includes the right to grant
484 | patent sublicenses in a manner consistent with the requirements of
485 | this License.
486 |
487 | Each contributor grants you a non-exclusive, worldwide, royalty-free
488 | patent license under the contributor's essential patent claims, to
489 | make, use, sell, offer for sale, import and otherwise run, modify and
490 | propagate the contents of its contributor version.
491 |
492 | In the following three paragraphs, a "patent license" is any express
493 | agreement or commitment, however denominated, not to enforce a patent
494 | (such as an express permission to practice a patent or covenant not to
495 | sue for patent infringement). To "grant" such a patent license to a
496 | party means to make such an agreement or commitment not to enforce a
497 | patent against the party.
498 |
499 | If you convey a covered work, knowingly relying on a patent license,
500 | and the Corresponding Source of the work is not available for anyone
501 | to copy, free of charge and under the terms of this License, through a
502 | publicly available network server or other readily accessible means,
503 | then you must either (1) cause the Corresponding Source to be so
504 | available, or (2) arrange to deprive yourself of the benefit of the
505 | patent license for this particular work, or (3) arrange, in a manner
506 | consistent with the requirements of this License, to extend the patent
507 | license to downstream recipients. "Knowingly relying" means you have
508 | actual knowledge that, but for the patent license, your conveying the
509 | covered work in a country, or your recipient's use of the covered work
510 | in a country, would infringe one or more identifiable patents in that
511 | country that you have reason to believe are valid.
512 |
513 | If, pursuant to or in connection with a single transaction or
514 | arrangement, you convey, or propagate by procuring conveyance of, a
515 | covered work, and grant a patent license to some of the parties
516 | receiving the covered work authorizing them to use, propagate, modify
517 | or convey a specific copy of the covered work, then the patent license
518 | you grant is automatically extended to all recipients of the covered
519 | work and works based on it.
520 |
521 | A patent license is "discriminatory" if it does not include within
522 | the scope of its coverage, prohibits the exercise of, or is
523 | conditioned on the non-exercise of one or more of the rights that are
524 | specifically granted under this License. You may not convey a covered
525 | work if you are a party to an arrangement with a third party that is
526 | in the business of distributing software, under which you make payment
527 | to the third party based on the extent of your activity of conveying
528 | the work, and under which the third party grants, to any of the
529 | parties who would receive the covered work from you, a discriminatory
530 | patent license (a) in connection with copies of the covered work
531 | conveyed by you (or copies made from those copies), or (b) primarily
532 | for and in connection with specific products or compilations that
533 | contain the covered work, unless you entered into that arrangement,
534 | or that patent license was granted, prior to 28 March 2007.
535 |
536 | Nothing in this License shall be construed as excluding or limiting
537 | any implied license or other defenses to infringement that may
538 | otherwise be available to you under applicable patent law.
539 |
540 | 12. No Surrender of Others' Freedom.
541 |
542 | If conditions are imposed on you (whether by court order, agreement or
543 | otherwise) that contradict the conditions of this License, they do not
544 | excuse you from the conditions of this License. If you cannot convey a
545 | covered work so as to satisfy simultaneously your obligations under this
546 | License and any other pertinent obligations, then as a consequence you may
547 | not convey it at all. For example, if you agree to terms that obligate you
548 | to collect a royalty for further conveying from those to whom you convey
549 | the Program, the only way you could satisfy both those terms and this
550 | License would be to refrain entirely from conveying the Program.
551 |
552 | 13. Use with the GNU Affero General Public License.
553 |
554 | Notwithstanding any other provision of this License, you have
555 | permission to link or combine any covered work with a work licensed
556 | under version 3 of the GNU Affero General Public License into a single
557 | combined work, and to convey the resulting work. The terms of this
558 | License will continue to apply to the part which is the covered work,
559 | but the special requirements of the GNU Affero General Public License,
560 | section 13, concerning interaction through a network will apply to the
561 | combination as such.
562 |
563 | 14. Revised Versions of this License.
564 |
565 | The Free Software Foundation may publish revised and/or new versions of
566 | the GNU General Public License from time to time. Such new versions will
567 | be similar in spirit to the present version, but may differ in detail to
568 | address new problems or concerns.
569 |
570 | Each version is given a distinguishing version number. If the
571 | Program specifies that a certain numbered version of the GNU General
572 | Public License "or any later version" applies to it, you have the
573 | option of following the terms and conditions either of that numbered
574 | version or of any later version published by the Free Software
575 | Foundation. If the Program does not specify a version number of the
576 | GNU General Public License, you may choose any version ever published
577 | by the Free Software Foundation.
578 |
579 | If the Program specifies that a proxy can decide which future
580 | versions of the GNU General Public License can be used, that proxy's
581 | public statement of acceptance of a version permanently authorizes you
582 | to choose that version for the Program.
583 |
584 | Later license versions may give you additional or different
585 | permissions. However, no additional obligations are imposed on any
586 | author or copyright holder as a result of your choosing to follow a
587 | later version.
588 |
589 | 15. Disclaimer of Warranty.
590 |
591 | THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
592 | APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
593 | HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
594 | OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
595 | THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
596 | PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
597 | IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
598 | ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
599 |
600 | 16. Limitation of Liability.
601 |
602 | IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
603 | WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
604 | THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
605 | GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
606 | USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
607 | DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
608 | PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
609 | EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
610 | SUCH DAMAGES.
611 |
612 | 17. Interpretation of Sections 15 and 16.
613 |
614 | If the disclaimer of warranty and limitation of liability provided
615 | above cannot be given local legal effect according to their terms,
616 | reviewing courts shall apply local law that most closely approximates
617 | an absolute waiver of all civil liability in connection with the
618 | Program, unless a warranty or assumption of liability accompanies a
619 | copy of the Program in return for a fee.
620 |
621 | END OF TERMS AND CONDITIONS
622 |
623 | How to Apply These Terms to Your New Programs
624 |
625 | If you develop a new program, and you want it to be of the greatest
626 | possible use to the public, the best way to achieve this is to make it
627 | free software which everyone can redistribute and change under these terms.
628 |
629 | To do so, attach the following notices to the program. It is safest
630 | to attach them to the start of each source file to most effectively
631 | state the exclusion of warranty; and each file should have at least
632 | the "copyright" line and a pointer to where the full notice is found.
633 |
634 |
635 | Copyright (C)
636 |
637 | This program is free software: you can redistribute it and/or modify
638 | it under the terms of the GNU General Public License as published by
639 | the Free Software Foundation, either version 3 of the License, or
640 | (at your option) any later version.
641 |
642 | This program is distributed in the hope that it will be useful,
643 | but WITHOUT ANY WARRANTY; without even the implied warranty of
644 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
645 | GNU General Public License for more details.
646 |
647 | You should have received a copy of the GNU General Public License
648 | along with this program. If not, see .
649 |
650 | Also add information on how to contact you by electronic and paper mail.
651 |
652 | If the program does terminal interaction, make it output a short
653 | notice like this when it starts in an interactive mode:
654 |
655 | Copyright (C)
656 | This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
657 | This is free software, and you are welcome to redistribute it
658 | under certain conditions; type `show c' for details.
659 |
660 | The hypothetical commands `show w' and `show c' should show the appropriate
661 | parts of the General Public License. Of course, your program's commands
662 | might be different; for a GUI interface, you would use an "about box".
663 |
664 | You should also get your employer (if you work as a programmer) or school,
665 | if any, to sign a "copyright disclaimer" for the program, if necessary.
666 | For more information on this, and how to apply and follow the GNU GPL, see
667 | .
668 |
669 | The GNU General Public License does not permit incorporating your program
670 | into proprietary programs. If your program is a subroutine library, you
671 | may consider it more useful to permit linking proprietary applications with
672 | the library. If this is what you want to do, use the GNU Lesser General
673 | Public License instead of this License. But first, please read
674 | .
675 |
--------------------------------------------------------------------------------