Compare commits
No commits in common. "488fcabd69ca6f9e306a3ca30ccef600209115b1" and "14edc3974b351c4a921ff717bc302fe04e66024f" have entirely different histories.
488fcabd69
...
14edc3974b
7 changed files with 278 additions and 383 deletions
|
@ -1,28 +0,0 @@
|
||||||
{
|
|
||||||
"paddingVertical": "45px",
|
|
||||||
"paddingHorizontal": "45px",
|
|
||||||
"backgroundImage": null,
|
|
||||||
"backgroundImageSelection": null,
|
|
||||||
"backgroundMode": "color",
|
|
||||||
"backgroundColor": "rgba(31,129,109,1)",
|
|
||||||
"dropShadow": true,
|
|
||||||
"dropShadowOffsetY": "12px",
|
|
||||||
"dropShadowBlurRadius": "22px",
|
|
||||||
"theme": "nord",
|
|
||||||
"windowTheme": "none",
|
|
||||||
"language": "text",
|
|
||||||
"fontFamily": "Hack",
|
|
||||||
"fontSize": "14px",
|
|
||||||
"lineHeight": "150%",
|
|
||||||
"windowControls": true,
|
|
||||||
"widthAdjustment": true,
|
|
||||||
"lineNumbers": false,
|
|
||||||
"firstLineNumber": 1,
|
|
||||||
"exportSize": "4x",
|
|
||||||
"watermark": false,
|
|
||||||
"squaredImage": false,
|
|
||||||
"hiddenCharacters": false,
|
|
||||||
"name": "",
|
|
||||||
"width": 680,
|
|
||||||
"highlights": null
|
|
||||||
}
|
|
BIN
assets/demo.png
BIN
assets/demo.png
Binary file not shown.
Before Width: | Height: | Size: 226 KiB |
16
flake.lock
16
flake.lock
|
@ -16,21 +16,6 @@
|
||||||
"url": "https://gist.github.com/f0fd86b6c73063283afe550bc5d77594.git"
|
"url": "https://gist.github.com/f0fd86b6c73063283afe550bc5d77594.git"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"flake-utils": {
|
|
||||||
"locked": {
|
|
||||||
"lastModified": 1659877975,
|
|
||||||
"narHash": "sha256-zllb8aq3YO3h8B/U0/J1WBgAL8EX5yWf5pMj3G0NAmc=",
|
|
||||||
"owner": "numtide",
|
|
||||||
"repo": "flake-utils",
|
|
||||||
"rev": "c0e246b9b83f637f4681389ecabcb2681b4f3af0",
|
|
||||||
"type": "github"
|
|
||||||
},
|
|
||||||
"original": {
|
|
||||||
"owner": "numtide",
|
|
||||||
"repo": "flake-utils",
|
|
||||||
"type": "github"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"gourou-src": {
|
"gourou-src": {
|
||||||
"flake": false,
|
"flake": false,
|
||||||
"locked": {
|
"locked": {
|
||||||
|
@ -83,7 +68,6 @@
|
||||||
"root": {
|
"root": {
|
||||||
"inputs": {
|
"inputs": {
|
||||||
"base64-src": "base64-src",
|
"base64-src": "base64-src",
|
||||||
"flake-utils": "flake-utils",
|
|
||||||
"gourou-src": "gourou-src",
|
"gourou-src": "gourou-src",
|
||||||
"nixpkgs": "nixpkgs",
|
"nixpkgs": "nixpkgs",
|
||||||
"pugixml-src": "pugixml-src",
|
"pugixml-src": "pugixml-src",
|
||||||
|
|
106
flake.nix
106
flake.nix
|
@ -1,7 +1,6 @@
|
||||||
{
|
{
|
||||||
inputs = {
|
inputs = {
|
||||||
nixpkgs.url = "github:NixOS/nixpkgs/nixos-unstable";
|
nixpkgs.url = "github:NixOS/nixpkgs/nixos-unstable";
|
||||||
flake-utils.url = "github:numtide/flake-utils";
|
|
||||||
gourou-src = {
|
gourou-src = {
|
||||||
url = "git://soutade.fr/libgourou.git";
|
url = "git://soutade.fr/libgourou.git";
|
||||||
flake = false;
|
flake = false;
|
||||||
|
@ -20,76 +19,62 @@
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
outputs = flakes:
|
outputs = flakes: let
|
||||||
flakes.flake-utils.lib.eachSystem [ "x86_64-linux" "aarch64-linux" ]
|
version = "1.3.0";
|
||||||
(system:
|
self = flakes.self.packages.x86_64-linux;
|
||||||
let
|
nixpkgs = flakes.nixpkgs.legacyPackages.x86_64-linux.pkgsStatic;
|
||||||
version = "1.3.1";
|
nixpkgs-dyn = flakes.nixpkgs.legacyPackages.x86_64-linux;
|
||||||
self = flakes.self.packages.${system};
|
|
||||||
nixpkgs = flakes.nixpkgs.legacyPackages.${system}.pkgsStatic;
|
|
||||||
nixpkgs-dyn = flakes.nixpkgs.legacyPackages.${system};
|
|
||||||
nixpkgs-fmt = flakes.nixpkgs-fmt.defaultPackage.${system};
|
|
||||||
gourou-src = flakes.gourou-src;
|
gourou-src = flakes.gourou-src;
|
||||||
updfparser-src = flakes.updfparser-src;
|
updfparser-src = flakes.updfparser-src;
|
||||||
base64-src = flakes.base64-src;
|
base64-src = flakes.base64-src;
|
||||||
pugixml-src = flakes.pugixml-src;
|
pugixml-src = flakes.pugixml-src;
|
||||||
cxx = "${nixpkgs.stdenv.cc}/bin/${nixpkgs.stdenv.cc.targetPrefix}c++";
|
cxx = "${nixpkgs.stdenv.cc}/bin/x86_64-unknown-linux-musl-g++";
|
||||||
ar = "${nixpkgs.stdenv.cc.bintools.bintools_bin}/bin/${nixpkgs.stdenv.cc.targetPrefix}ar";
|
ar = "${nixpkgs.stdenv.cc.bintools.bintools_bin}/bin/x86_64-unknown-linux-musl-ar";
|
||||||
obj-flags = "-O2 -static";
|
obj-flags = "-O2 -static";
|
||||||
in
|
in rec {
|
||||||
rec {
|
packages.x86_64-linux.libzip-static = nixpkgs.libzip.overrideAttrs (prev: {
|
||||||
packages.libzip-static = nixpkgs.libzip.overrideAttrs (prev: {
|
cmakeFlags = (prev.cmakeFlags or []) ++ [
|
||||||
cmakeFlags = (prev.cmakeFlags or [ ]) ++ [
|
|
||||||
"-DBUILD_SHARED_LIBS=OFF"
|
"-DBUILD_SHARED_LIBS=OFF"
|
||||||
"-DBUILD_EXAMPLES=OFF"
|
"-DBUILD_EXAMPLES=OFF"
|
||||||
"-DBUILD_DOC=OFF"
|
"-DBUILD_DOC=OFF"
|
||||||
"-DBUILD_TOOLS=OFF"
|
"-DBUILD_TOOLS=OFF"
|
||||||
"-DBUILD_REGRESS=OFF"
|
"-DBUILD_REGRESS=OFF"
|
||||||
];
|
];
|
||||||
outputs = [ "out" ];
|
outputs = ["out"];
|
||||||
});
|
});
|
||||||
packages.base64 = derivation {
|
packages.x86_64-linux.base64 = derivation {
|
||||||
name = "updfparser";
|
name = "updfparser";
|
||||||
inherit system;
|
system = "x86_64-linux";
|
||||||
builder = "${nixpkgs.bash}/bin/bash";
|
builder = "${nixpkgs.bash}/bin/bash";
|
||||||
PATH = "${nixpkgs.coreutils}/bin";
|
PATH = "${nixpkgs.coreutils}/bin";
|
||||||
args = [
|
args = ["-c" ''
|
||||||
"-c"
|
|
||||||
''
|
|
||||||
mkdir -p $out/include/base64
|
mkdir -p $out/include/base64
|
||||||
cp ${base64-src}/Base64.h $out/include/base64/Base64.h
|
cp ${base64-src}/Base64.h $out/include/base64/Base64.h
|
||||||
''
|
''];
|
||||||
];
|
|
||||||
};
|
};
|
||||||
packages.updfparser = derivation {
|
packages.x86_64-linux.updfparser = derivation {
|
||||||
name = "updfparser";
|
name = "updfparser";
|
||||||
inherit system;
|
system = "x86_64-linux";
|
||||||
builder = "${nixpkgs.bash}/bin/bash";
|
builder = "${nixpkgs.bash}/bin/bash";
|
||||||
PATH = "${nixpkgs.coreutils}/bin";
|
PATH = "${nixpkgs.coreutils}/bin";
|
||||||
args = [
|
args = [ "-c" ''
|
||||||
"-c"
|
|
||||||
''
|
|
||||||
${cxx} \
|
${cxx} \
|
||||||
-c ${updfparser-src}/src/*.cpp \
|
-c ${updfparser-src}/src/*.cpp \
|
||||||
-I ${updfparser-src}/include \
|
-I ${updfparser-src}/include \
|
||||||
${obj-flags}
|
${obj-flags}
|
||||||
mkdir -p $out/lib
|
mkdir -p $out/lib
|
||||||
${ar} crs $out/lib/libupdfparser.a *.o
|
${ar} crs $out/lib/libupdfparser.a *.o
|
||||||
''
|
'' ];
|
||||||
];
|
|
||||||
};
|
};
|
||||||
packages.gourou = derivation {
|
packages.x86_64-linux.gourou = derivation {
|
||||||
name = "gourou";
|
name = "gourou";
|
||||||
inherit system;
|
system = "x86_64-linux";
|
||||||
builder = "${nixpkgs.bash}/bin/bash";
|
builder = "${nixpkgs.bash}/bin/bash";
|
||||||
PATH = "${nixpkgs.coreutils}/bin";
|
PATH = "${nixpkgs.coreutils}/bin";
|
||||||
args = [
|
args = [ "-c" ''
|
||||||
"-c"
|
|
||||||
''
|
|
||||||
shopt -s extglob
|
|
||||||
${cxx} \
|
${cxx} \
|
||||||
-c \
|
-c \
|
||||||
${gourou-src}/src/!(pugixml).cpp \
|
${gourou-src}/src/*.cpp \
|
||||||
${pugixml-src}/src/pugixml.cpp \
|
${pugixml-src}/src/pugixml.cpp \
|
||||||
-I ${self.base64}/include \
|
-I ${self.base64}/include \
|
||||||
-I ${gourou-src}/include \
|
-I ${gourou-src}/include \
|
||||||
|
@ -99,17 +84,14 @@
|
||||||
mkdir -p $out/lib $out/debug
|
mkdir -p $out/lib $out/debug
|
||||||
${ar} crs $out/lib/libgourou.a *.o
|
${ar} crs $out/lib/libgourou.a *.o
|
||||||
cp *.o $out/debug
|
cp *.o $out/debug
|
||||||
''
|
'' ];
|
||||||
];
|
|
||||||
};
|
};
|
||||||
packages.utils-common = derivation {
|
packages.x86_64-linux.utils-common = derivation {
|
||||||
name = "utils-common";
|
name = "utils-common";
|
||||||
inherit system;
|
system = "x86_64-linux";
|
||||||
builder = "${nixpkgs.bash}/bin/bash";
|
builder = "${nixpkgs.bash}/bin/bash";
|
||||||
PATH = "${nixpkgs.coreutils}/bin";
|
PATH = "${nixpkgs.coreutils}/bin";
|
||||||
args = [
|
args = [ "-c" ''
|
||||||
"-c"
|
|
||||||
''
|
|
||||||
${cxx} \
|
${cxx} \
|
||||||
-c ${gourou-src}/utils/drmprocessorclientimpl.cpp \
|
-c ${gourou-src}/utils/drmprocessorclientimpl.cpp \
|
||||||
${gourou-src}/utils/utils_common.cpp \
|
${gourou-src}/utils/utils_common.cpp \
|
||||||
|
@ -123,23 +105,19 @@
|
||||||
${obj-flags}
|
${obj-flags}
|
||||||
mkdir -p $out/lib
|
mkdir -p $out/lib
|
||||||
${ar} crs $out/lib/libutils-common.a *.o
|
${ar} crs $out/lib/libutils-common.a *.o
|
||||||
''
|
'' ];
|
||||||
];
|
|
||||||
};
|
};
|
||||||
packages.knock = derivation {
|
packages.x86_64-linux.knock = derivation {
|
||||||
name = "knock";
|
name = "knock";
|
||||||
inherit system;
|
system = "x86_64-linux";
|
||||||
builder = "${nixpkgs.bash}/bin/bash";
|
builder = "${nixpkgs.bash}/bin/bash";
|
||||||
PATH = "${nixpkgs.coreutils}/bin";
|
PATH = "${nixpkgs.coreutils}/bin";
|
||||||
args = [
|
args = [ "-c" ''
|
||||||
"-c"
|
|
||||||
''
|
|
||||||
mkdir -p $out/bin
|
mkdir -p $out/bin
|
||||||
${cxx} \
|
${cxx} \
|
||||||
-o $out/bin/knock \
|
-o $out/bin/knock \
|
||||||
${./src/knock.cpp} \
|
${./src/knock.cpp} \
|
||||||
-D KNOCK_VERSION='"${version}"' \
|
-D KNOCK_VERSION='"${version}"' \
|
||||||
--std=c++17 \
|
|
||||||
-Wl,--as-needed -static \
|
-Wl,--as-needed -static \
|
||||||
${self.utils-common}/lib/libutils-common.a \
|
${self.utils-common}/lib/libutils-common.a \
|
||||||
${self.gourou}/lib/libgourou.a \
|
${self.gourou}/lib/libgourou.a \
|
||||||
|
@ -164,19 +142,16 @@
|
||||||
-I ${nixpkgs.curl.dev}/include \
|
-I ${nixpkgs.curl.dev}/include \
|
||||||
-I ${nixpkgs.zlib.dev}/include \
|
-I ${nixpkgs.zlib.dev}/include \
|
||||||
-I ${self.libzip-static}/include
|
-I ${self.libzip-static}/include
|
||||||
''
|
'' ];
|
||||||
];
|
|
||||||
};
|
};
|
||||||
packages.default = self.knock;
|
packages.x86_64-linux.default = self.knock;
|
||||||
packages.tests = nixpkgs-dyn.stdenv.mkDerivation {
|
packages.x86_64-linux.tests = nixpkgs-dyn.stdenv.mkDerivation {
|
||||||
name = "tests";
|
name = "tests";
|
||||||
src = ./tests;
|
src = ./tests;
|
||||||
buildInputs = [
|
buildInputs = [ (nixpkgs-dyn.python3.withPackages(p: [
|
||||||
(nixpkgs-dyn.python3.withPackages (p: [
|
|
||||||
p.beautifulsoup4
|
p.beautifulsoup4
|
||||||
p.requests
|
p.requests
|
||||||
]))
|
])) ];
|
||||||
];
|
|
||||||
patchPhase = ''
|
patchPhase = ''
|
||||||
substituteInPlace tests.py --replace "./result/bin/knock" "${self.knock}/bin/knock"
|
substituteInPlace tests.py --replace "./result/bin/knock" "${self.knock}/bin/knock"
|
||||||
'';
|
'';
|
||||||
|
@ -186,12 +161,5 @@
|
||||||
chmod +x $out/bin/tests
|
chmod +x $out/bin/tests
|
||||||
'';
|
'';
|
||||||
};
|
};
|
||||||
packages.formatter = nixpkgs.writeShellScriptBin "formatter" ''
|
};
|
||||||
set -x
|
|
||||||
${nixpkgs-dyn.clang-tools}/bin/clang-format -i --verbose ./src/*.cpp
|
|
||||||
${nixpkgs-dyn.nixpkgs-fmt}/bin/nixpkgs-fmt .
|
|
||||||
${nixpkgs-dyn.black}/bin/black ./tests
|
|
||||||
'';
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
}
|
61
readme.md
61
readme.md
|
@ -2,32 +2,13 @@
|
||||||
|
|
||||||
Convert ACSM files to PDF/EPUBs with one command on Linux ([and MacOS very soon](https://github.com/BentonEdmondson/knock/issues/58)).
|
Convert ACSM files to PDF/EPUBs with one command on Linux ([and MacOS very soon](https://github.com/BentonEdmondson/knock/issues/58)).
|
||||||
|
|
||||||
![Demonstration of CLI usage](./assets/demo.png)
|
|
||||||
|
|
||||||
*This software does not utilize Adobe Digital Editions nor Wine. It is completely free and open-source software written natively for Linux.*
|
*This software does not utilize Adobe Digital Editions nor Wine. It is completely free and open-source software written natively for Linux.*
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
1. Run `uname -ms` and, based on the output, download the latest corresponding [release](https://github.com/BentonEdmondson/knock/releases).
|
* Download the latest [release](https://github.com/BentonEdmondson/knock/releases). Make sure it is the correct version for your architecture (run `uname -m` to check).
|
||||||
1. `cd` into the directory that knock is in (e.g. `cd ~/Downloads`).
|
* Rename the binary and make it executable.
|
||||||
1. Run `mv knock-version-arch-os knock` to rename the binary to `knock`.
|
* Run `knock /path/to/book.acsm` to perform the conversion.
|
||||||
1. Run `chmod +x knock` to make it executable.
|
|
||||||
1. Run `./knock ./path/to/book.acsm` to perform the conversion.
|
|
||||||
1. Run `mv knock ~/.local/bin` to allow it to be run from anywhere (might not work on some distributions).
|
|
||||||
|
|
||||||
### Nix and NixOS (with [flakes](https://nixos.wiki/wiki/Flakes) enabled)
|
|
||||||
|
|
||||||
If you are on a system with [Nix](https://github.com/NixOS/nix), you can use the following.
|
|
||||||
|
|
||||||
```
|
|
||||||
nix profile install github:BentonEdmondson/knock
|
|
||||||
```
|
|
||||||
|
|
||||||
If you are on [NixOS](https://github.com/NixOS/nixpkgs), you can add the flake to your system config.
|
|
||||||
|
|
||||||
## Dependencies
|
|
||||||
|
|
||||||
There are no userspace runtime dependencies.
|
|
||||||
|
|
||||||
## Verified Book Sources
|
## Verified Book Sources
|
||||||
|
|
||||||
|
@ -35,6 +16,23 @@ Knock should work on any ACSM file, but it has been specifically verified to wor
|
||||||
|
|
||||||
Before buying your ebook, check if it is available for free on [Project Gutenberg](https://gutenberg.org/).
|
Before buying your ebook, check if it is available for free on [Project Gutenberg](https://gutenberg.org/).
|
||||||
|
|
||||||
|
## The Name
|
||||||
|
|
||||||
|
The name comes from the [D&D 5e spell](https://roll20.net/compendium/dnd5e/Knock#content) for freeing locked items:
|
||||||
|
|
||||||
|
> ### Knock
|
||||||
|
> *2nd level transmutation*\
|
||||||
|
> **Casting Time**: 1 action\
|
||||||
|
> **Range**: 60 feet\
|
||||||
|
> **Components**: V\
|
||||||
|
> **Duration**: Instantaneous\
|
||||||
|
> **Classes**: Bard, Sorcerer, Wizard\
|
||||||
|
> Choose an object that you can see within range. The object can be a door, a box, a chest, a set of manacles, a padlock, or another object that contains a mundane or magical means that prevents access. A target that is held shut by a mundane lock or that is stuck or barred becomes unlocked, unstuck, or unbarred. If the object has multiple locks, only one of them is unlocked. If you choose a target that is held shut with arcane lock, that spell is suppressed for 10 minutes, during which time the target can be opened and shut normally. When you cast the spell, a loud knock, audible from as far away as 300 feet, emanates from the target object.
|
||||||
|
|
||||||
|
## Dependencies
|
||||||
|
|
||||||
|
There are no userspace runtime dependencies.
|
||||||
|
|
||||||
## Contributing
|
## Contributing
|
||||||
|
|
||||||
Install [Nix](https://github.com/NixOS/nix) if you don't have it. Enable [flakes](https://nixos.wiki/wiki/Flakes) if you haven't.
|
Install [Nix](https://github.com/NixOS/nix) if you don't have it. Enable [flakes](https://nixos.wiki/wiki/Flakes) if you haven't.
|
||||||
|
@ -59,25 +57,6 @@ nix run .#tests -- ./tests/workspace
|
||||||
|
|
||||||
Test books can be found [here](https://www.adobe.com/solutions/ebook/digital-editions/sample-ebook-library.html).
|
Test books can be found [here](https://www.adobe.com/solutions/ebook/digital-editions/sample-ebook-library.html).
|
||||||
|
|
||||||
### Formatting
|
|
||||||
|
|
||||||
```
|
|
||||||
nix run .#formatter
|
|
||||||
```
|
|
||||||
|
|
||||||
## The Name
|
|
||||||
|
|
||||||
The name comes from the [D&D 5e spell](https://roll20.net/compendium/dnd5e/Knock#content) for freeing locked items:
|
|
||||||
|
|
||||||
> ### Knock
|
|
||||||
> *2nd level transmutation*\
|
|
||||||
> **Casting Time**: 1 action\
|
|
||||||
> **Range**: 60 feet\
|
|
||||||
> **Components**: V\
|
|
||||||
> **Duration**: Instantaneous\
|
|
||||||
> **Classes**: Bard, Sorcerer, Wizard\
|
|
||||||
> Choose an object that you can see within range. The object can be a door, a box, a chest, a set of manacles, a padlock, or another object that contains a mundane or magical means that prevents access. A target that is held shut by a mundane lock or that is stuck or barred becomes unlocked, unstuck, or unbarred. If the object has multiple locks, only one of them is unlocked. If you choose a target that is held shut with arcane lock, that spell is suppressed for 10 minutes, during which time the target can be opened and shut normally. When you cast the spell, a loud knock, audible from as far away as 300 feet, emanates from the target object.
|
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
This software is licensed under GPLv3. The linked libraries have various licenses.
|
This software is licensed under GPLv3. The linked libraries have various licenses.
|
||||||
|
|
|
@ -1,24 +1,24 @@
|
||||||
#include "drmprocessorclientimpl.h"
|
|
||||||
#include "libgourou.h"
|
|
||||||
#include "libgourou_common.h"
|
|
||||||
#include <filesystem>
|
#include <filesystem>
|
||||||
|
#include "drmprocessorclientimpl.h"
|
||||||
|
#include "libgourou_common.h"
|
||||||
|
#include "libgourou.h"
|
||||||
|
|
||||||
#ifndef KNOCK_VERSION
|
#ifndef KNOCK_VERSION
|
||||||
#error KNOCK_VERSION must be defined
|
#error KNOCK_VERSION must be defined
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
std::string get_data_dir();
|
std::string get_data_dir();
|
||||||
void verify_absence(std::string file);
|
void verify_absence(std::string file);
|
||||||
void verify_presence(std::string file);
|
void verify_presence(std::string file);
|
||||||
|
|
||||||
int main(int argc, char **argv) try {
|
int main(int argc, char** argv) try {
|
||||||
|
|
||||||
if (argc == 1) {
|
if (argc == 1) {
|
||||||
std::cout << "info: knock version " << KNOCK_VERSION
|
std::cout
|
||||||
<< ", libgourou version " << LIBGOUROU_VERSION << "\n"
|
<< "info: knock version " << KNOCK_VERSION << ", libgourou version "
|
||||||
<< "usage: " << argv[0] << " [ACSM]\n"
|
<< LIBGOUROU_VERSION << "\n"
|
||||||
<< "result: converts file ACSM to a plain EPUB/PDF if present, "
|
<< "usage: " << argv[0] << " [ACSM]" << "\n"
|
||||||
"otherwise prints this"
|
<< "result: converts file ACSM to a plain EPUB/PDF if present, otherwise prints this"
|
||||||
<< std::endl;
|
<< std::endl;
|
||||||
return EXIT_SUCCESS;
|
return EXIT_SUCCESS;
|
||||||
}
|
}
|
||||||
|
@ -29,8 +29,7 @@ int main(int argc, char **argv) try {
|
||||||
|
|
||||||
const std::string acsm_file = argv[1];
|
const std::string acsm_file = argv[1];
|
||||||
verify_presence(acsm_file);
|
verify_presence(acsm_file);
|
||||||
const std::string acsm_stem =
|
const std::string acsm_stem = acsm_file.substr(0, acsm_file.find_last_of("."));
|
||||||
acsm_file.substr(0, acsm_file.find_last_of("."));
|
|
||||||
const std::string drm_file = acsm_stem + ".drm";
|
const std::string drm_file = acsm_stem + ".drm";
|
||||||
const std::string out_file = acsm_stem + ".out";
|
const std::string out_file = acsm_stem + ".out";
|
||||||
verify_absence(drm_file);
|
verify_absence(drm_file);
|
||||||
|
@ -38,17 +37,18 @@ int main(int argc, char **argv) try {
|
||||||
const std::string knock_data = get_data_dir();
|
const std::string knock_data = get_data_dir();
|
||||||
|
|
||||||
DRMProcessorClientImpl client;
|
DRMProcessorClientImpl client;
|
||||||
gourou::DRMProcessor *processor = gourou::DRMProcessor::createDRMProcessor(
|
gourou::DRMProcessor* processor = gourou::DRMProcessor::createDRMProcessor(
|
||||||
&client,
|
&client,
|
||||||
false, // don't "always generate a new device" (default)
|
false, // don't "always generate a new device" (default)
|
||||||
knock_data);
|
knock_data
|
||||||
|
);
|
||||||
|
|
||||||
std::cout << "anonymously signing in..." << std::endl;
|
std::cout << "anonymously signing in..." << std::endl;
|
||||||
processor->signIn("anonymous", "");
|
processor->signIn("anonymous", "");
|
||||||
processor->activateDevice();
|
processor->activateDevice();
|
||||||
|
|
||||||
std::cout << "downloading the file from Adobe..." << std::endl;
|
std::cout << "downloading the file from Adobe..." << std::endl;
|
||||||
gourou::FulfillmentItem *item = processor->fulfill(acsm_file);
|
gourou::FulfillmentItem* item = processor->fulfill(acsm_file);
|
||||||
gourou::DRMProcessor::ITEM_TYPE type = processor->download(item, drm_file);
|
gourou::DRMProcessor::ITEM_TYPE type = processor->download(item, drm_file);
|
||||||
|
|
||||||
std::cout << "removing DRM from the file..." << std::endl;
|
std::cout << "removing DRM from the file..." << std::endl;
|
||||||
|
@ -74,29 +74,28 @@ int main(int argc, char **argv) try {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (std::filesystem::exists(ext_file)) {
|
if (std::filesystem::exists(ext_file)) {
|
||||||
std::cerr << "warning: failed to update file extension; " + ext_file +
|
std::cerr
|
||||||
" already exists"
|
<< "warning: failed to update file extension; " + ext_file + " already exists"
|
||||||
<< std::endl;
|
<< std::endl;
|
||||||
ext_file = out_file;
|
ext_file = out_file;
|
||||||
} else {
|
} else {
|
||||||
std::filesystem::rename(out_file, ext_file);
|
std::filesystem::rename(out_file, ext_file);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::filesystem::remove(acsm_file);
|
|
||||||
std::cout << file_type + " file generated at " + ext_file << std::endl;
|
std::cout << file_type + " file generated at " + ext_file << std::endl;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
} catch (const gourou::Exception &e) {
|
} catch (const gourou::Exception& e) {
|
||||||
std::cerr << "error:\n" << e.what();
|
std::cerr << "error:\n" << e.what();
|
||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
} catch (const std::exception &e) {
|
} catch (const std::exception& e) {
|
||||||
std::cerr << "error: " << e.what() << std::endl;
|
std::cerr << "error: " << e.what() << std::endl;
|
||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string get_data_dir() {
|
std::string get_data_dir() {
|
||||||
char *xdg_data_home = std::getenv("XDG_DATA_HOME");
|
char* xdg_data_home = std::getenv("XDG_DATA_HOME");
|
||||||
std::string knock_data;
|
std::string knock_data;
|
||||||
if (xdg_data_home != nullptr) {
|
if (xdg_data_home != nullptr) {
|
||||||
knock_data = xdg_data_home;
|
knock_data = xdg_data_home;
|
||||||
|
@ -109,8 +108,7 @@ std::string get_data_dir() {
|
||||||
|
|
||||||
void verify_absence(std::string file) {
|
void verify_absence(std::string file) {
|
||||||
if (std::filesystem::exists(file)) {
|
if (std::filesystem::exists(file)) {
|
||||||
throw std::runtime_error("file " + file +
|
throw std::runtime_error("file " + file + " must be moved out of the way or deleted");
|
||||||
" must be moved out of the way or deleted");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -14,10 +14,9 @@ if not knock.exists():
|
||||||
if len(sys.argv) != 2:
|
if len(sys.argv) != 2:
|
||||||
print(
|
print(
|
||||||
"error: missing required argument: directory in which to perform the tests",
|
"error: missing required argument: directory in which to perform the tests",
|
||||||
file=sys.stderr,
|
file=sys.stderr
|
||||||
)
|
)
|
||||||
sys.exit()
|
sys.exit()
|
||||||
workspace = Path(sys.argv[1])
|
|
||||||
|
|
||||||
print("Testing " + str(knock))
|
print("Testing " + str(knock))
|
||||||
|
|
||||||
|
@ -27,13 +26,18 @@ if result.returncode != 0:
|
||||||
sys.exit()
|
sys.exit()
|
||||||
print("---")
|
print("---")
|
||||||
|
|
||||||
html = requests.get(
|
workspace = Path(sys.argv[1])
|
||||||
"https://www.adobe.com/solutions/ebook/digital-editions/sample-ebook-library.html"
|
if workspace.exists():
|
||||||
).text
|
shutil.rmtree(workspace)
|
||||||
soup = BeautifulSoup(html, "html.parser")
|
workspace.mkdir()
|
||||||
|
|
||||||
|
html = requests \
|
||||||
|
.get("https://www.adobe.com/solutions/ebook/digital-editions/sample-ebook-library.html") \
|
||||||
|
.text
|
||||||
|
soup = BeautifulSoup(html, 'html.parser')
|
||||||
|
|
||||||
links = []
|
links = []
|
||||||
for a_tag in soup.find_all("a"):
|
for a_tag in soup.find_all('a'):
|
||||||
if a_tag.string != "Download eBook":
|
if a_tag.string != "Download eBook":
|
||||||
continue
|
continue
|
||||||
if not urlparse(a_tag.get("href")).path.endswith(".acsm"):
|
if not urlparse(a_tag.get("href")).path.endswith(".acsm"):
|
||||||
|
@ -41,19 +45,9 @@ for a_tag in soup.find_all("a"):
|
||||||
|
|
||||||
links.append(a_tag.get("href"))
|
links.append(a_tag.get("href"))
|
||||||
|
|
||||||
if len(links) >= 10:
|
for i, link in enumerate(links):
|
||||||
break
|
|
||||||
|
|
||||||
for time in ["first", "second"]:
|
|
||||||
|
|
||||||
if workspace.exists():
|
|
||||||
shutil.rmtree(workspace)
|
|
||||||
workspace.mkdir()
|
|
||||||
|
|
||||||
for i, link in enumerate(links):
|
|
||||||
i = str(i)
|
i = str(i)
|
||||||
|
print("Testing URL #" + i + ":\n" + link)
|
||||||
print("Testing URL #" + i + " for the " + time + " time:\n" + link)
|
|
||||||
file = workspace.joinpath(i + ".acsm")
|
file = workspace.joinpath(i + ".acsm")
|
||||||
|
|
||||||
r = requests.get(link)
|
r = requests.get(link)
|
||||||
|
|
Loading…
Reference in a new issue