better build process, better output, libgourou updated, better file dir structure

This commit is contained in:
benton 2021-09-11 22:25:19 -05:00
parent cf007616b7
commit 094fac4d9f
6 changed files with 169 additions and 114 deletions

18
flake.lock generated
View file

@ -7,11 +7,11 @@
]
},
"locked": {
"lastModified": 1627256311,
"narHash": "sha256-NEDsexz3lNmaLm6nSHURsmgNX592b3fcNR2fzEuk3U0=",
"lastModified": 1628616743,
"narHash": "sha256-XNka+o/55SYOtDUXXOTmDvX4Qiqbaz4COq+JZ8MPoe8=",
"owner": "BentonEdmondson",
"repo": "inept-epub",
"rev": "b992adc8be351c00673a032df10552d7bc577290",
"rev": "d6ba98a7159f61e7b77dc17c8c6bb62e193c8831",
"type": "github"
},
"original": {
@ -27,11 +27,11 @@
]
},
"locked": {
"lastModified": 1627508403,
"narHash": "sha256-JHOxPSJSaS+0NTUkNv6a9oI8g30pBuCDC+JwtZBaq3U=",
"lastModified": 1631412686,
"narHash": "sha256-u0v3qU+qckvZE/obCHjeyktcvH8Wyo4Zld9AxEI53bI=",
"owner": "BentonEdmondson",
"repo": "libgourou-utils",
"rev": "f16d3df200134209a1e2e530db056b3cd715425d",
"rev": "88d6dc2b549a28f886a15a7992890d60e576716d",
"type": "github"
},
"original": {
@ -42,11 +42,11 @@
},
"nixpkgs": {
"locked": {
"lastModified": 1627128856,
"narHash": "sha256-yw3lA8zyNFhj309lmxvNByEEymRT1rRy5oE+jEPnsP4=",
"lastModified": 1631206977,
"narHash": "sha256-o3Dct9aJ5ht5UaTUBzXrRcK1RZt2eG5/xSlWJuUCVZM=",
"owner": "NixOS",
"repo": "nixpkgs",
"rev": "dd14e5d78e90a2ccd6007e569820de9b4861a6c2",
"rev": "4f6d8095fd51954120a1d08ea5896fe42dc3923b",
"type": "github"
},
"original": {

View file

@ -14,28 +14,25 @@
libgourou-utils = flakes.libgourou-utils.defaultPackage.x86_64-linux;
inept-epub = flakes.inept-epub.defaultPackage.x86_64-linux;
in {
defaultPackage.x86_64-linux = nixpkgs.stdenv.mkDerivation {
defaultPackage.x86_64-linux = nixpkgs.python3Packages.buildPythonApplication {
pname = "knock";
version = "1.0.0-alpha";
src = self;
nativeBuildInputs = [ nixpkgs.makeWrapper ];
buildInputs = [
(nixpkgs.python3.withPackages
(python3Packages: [
python3Packages.python_magic
python3Packages.xdg
])
)
libgourou-utils inept-epub
propagatedBuildInputs = [
nixpkgs.python3Packages.python_magic
nixpkgs.python3Packages.xdg
nixpkgs.python3Packages.click
libgourou-utils
inept-epub
];
format = "other";
installPhase = ''
mkdir -p $out/bin
chmod +x knock
cp knock $out/bin
wrapProgram $out/bin/knock --prefix PATH : ${nixpkgs.lib.makeBinPath [libgourou-utils inept-epub]}
mkdir -p $out/bin $out/${nixpkgs.python3.sitePackages}
cp lib/*.py $out/${nixpkgs.python3.sitePackages}
cp src/knock.py $out/bin/knock
'';
meta = {

90
knock
View file

@ -1,90 +0,0 @@
#!/usr/bin/env python3
import os, sys, argparse, subprocess, magic, shutil
from pathlib import Path
from getpass import getpass
from xdg import xdg_config_home
parser = argparse.ArgumentParser(prog='knock', description='Convert an ACSM file to a DRM-free EPUB file')
parser.add_argument('acsm_file', metavar='file', type=str, help='the ACSM file to convert')
args = parser.parse_args()
# these are all Path objects: https://docs.python.org/3/library/pathlib.html
args.acsm_file = Path(args.acsm_file).expanduser().resolve()
args.drm_file = args.acsm_file.with_suffix('.drm')
args.epub_file = args.acsm_file.with_suffix('.epub')
args.adobe_dir = xdg_config_home() / 'knock'
if not args.acsm_file.exists():
sys.exit(f'ERROR: {str(args.acsm_file)} does not exist.')
if args.epub_file.exists():
sys.exit(f'ERROR: {str(args.epub_file)} already exists.')
if args.drm_file.exists():
sys.exit(f'ERROR: {str(args.drm_file)} must be moved or deleted.')
if not args.adobe_dir.exists():
print('This device is not registered with Adobe.')
email = input("Enter your Adobe account's email address: ")
password = getpass("Enter your Adobe account's password: ")
print('Registering this device with Adobe...')
result = subprocess.run([
'adept-register',
'-u', email,
'-O', str(args.adobe_dir)
], input=password.encode(), stdout=subprocess.PIPE, stderr=subprocess.PIPE)
if result.stdout.decode().strip().startswith('Exception code : 0x1003'):
shutil.rmtree(str(args.adobe_dir))
sys.exit('ERROR: Incorrect password')
if result.stdout.decode().strip().startswith('Exception code : 0x500a'):
shutil.rmtree(str(args.adobe_dir))
sys.exit('ERROR: No internet access')
if result.returncode != 0 or not args.adobe_dir.exists() or result.stdout.decode().strip().startswith('Exception code : '):
if args.adobe_dir.exists():
shutil.rmtree(str(args.adobe_dir))
print('ERROR: ', file=sys.stderr)
sys.exit(result)
print('Downloading the EPUB file from Adobe...')
result = subprocess.run([
'adept-download',
'-d', str(args.adobe_dir.joinpath('device.xml')),
'-a', str(args.adobe_dir.joinpath('activation.xml')),
'-k', str(args.adobe_dir.joinpath('devicesalt')),
'-o', str(args.drm_file),
'-f', str(args.acsm_file)
], stdout=subprocess.PIPE, stderr=subprocess.PIPE)
if result.stdout.decode().strip().startswith('Exception code : 0x500a'):
sys.exit('ERROR: No internet access')
if result.returncode != 0 or not args.drm_file.exists():
print('ERROR: ', file=sys.stderr)
sys.exit(result)
drm_file_type = magic.from_file(str(args.drm_file), mime=True)
if drm_file_type != 'application/epub+zip':
sys.exit(f'Received a file of type:\n{drm_file_type}\nKnock only supports EPUB files.')
print('Decrypting the file...')
result = subprocess.run([
'inept-epub',
str(args.adobe_dir.joinpath('activation.xml')),
str(args.drm_file),
str(args.epub_file)
], stdout=subprocess.PIPE, stderr=subprocess.PIPE)
if result.returncode != 0 or not args.epub_file.exists():
print('ERROR: ', file=sys.stderr)
sys.exit(result)
args.drm_file.unlink()
print(f'\nDRM-free EPUB file created:\n{str(args.epub_file)}')

72
lib/handle_acsm.py Normal file
View file

@ -0,0 +1,72 @@
from xdg import xdg_config_home
import click, sys, shutil, subprocess
from run import run
def handle_acsm(acsm_path):
drm_path = acsm_path.with_suffix('.drm')
adobe_dir = xdg_config_home() / 'knock' / 'acsm'
if drm_path.exists():
click.echo(f"Error: {drm_path} must be moved out of the way or deleted.", err=True)
sys.exit(1)
adobe_dir.mkdir(parents=True, exist_ok=True)
if (
not (adobe_dir / 'device.xml').exists()
or not (adobe_dir / 'activation.xml').exists()
or not (adobe_dir / 'devicesalt').exists()
):
shutil.rmtree(str(adobe_dir))
click.echo('This device is not registered with Adobe.')
email = click.prompt("Enter your Adobe account's email address")
password = click.prompt("Enter your Adobe account's password", hide_input=True)
click.echo('Registering this device with Adobe...')
run(
[
'adept-register',
'-u', email,
'-O', str(adobe_dir)
],
stdin=password+'\n',
cleanser=lambda:shutil.rmtree(str(adobe_dir))
)
click.echo('Downloading the EPUB file from Adobe...')
run([
'adept-download',
'-d', str(adobe_dir.joinpath('device.xml')),
'-a', str(adobe_dir.joinpath('activation.xml')),
'-k', str(adobe_dir.joinpath('devicesalt')),
'-o', str(drm_path),
'-f', str(acsm_path)
])
drm_file_type = magic.from_file(str(args.drm_file), mime=True)
if drm_file_type == 'application/epub+zip':
decryption_command = 'inept-epub'
elif drm_file_type == 'application/pdf':
decryption_command = 'inept-pdf'
else:
click.echo(f'Error: Received file of media type {drm_file_type}.', err=True)
click.echo('Only the following ACSM conversions are currently supported:', err=True)
click.echo(' * ACSM -> EPUB', err=True)
click.echo(' * ACSM -> PDF', err=True)
click.echo('Please open a feature request at:', err=True)
click.echo(f' https://github.com/BentonEdmondson/knock/issues/new?title=Support%20{drm_file_type}%20Files&labels=enhancement', err=True)
sys.exit(1)
click.echo('Decrypting the file...')
run([
decryption_command,
str(args.adobe_dir.joinpath('activation.xml')),
str(args.drm_file),
str(args.epub_file)
])
args.drm_file.unlink()
click.secho(f'DRM-free EPUB file created:\n{str(args.epub_file)}', color='green')

33
lib/run.py Normal file
View file

@ -0,0 +1,33 @@
import click, subprocess, sys
# run a command and display output in a styled terminal
# cleanser is called if the command returns a >0 exit code
def run(command: [str], stdin: str = '', cleanser = lambda: None) -> int:
# newline and set styles
click.secho('', fg='white', bg='black', bold=True, reset=False)
# show command
click.echo('knock> ' + ' '.join(command))
# remove bold
click.secho('', fg='white', bg='black', bold=False, reset=False)
result = subprocess.run(
command,
stderr=subprocess.STDOUT,
input=stdin.encode(),
check=False # don't throw Python error if returncode isn't 0
)
# show returncode in bold, then reset styles
click.secho(f'\nknock[{result.returncode}]>', bold=True)
# newline
click.echo('')
if result.returncode > 0:
cleanser()
click.echo(f'Error: Command returned error code {result.returncode}.', err=True)
sys.exit(1)
return result.returncode

43
src/knock.py Executable file
View file

@ -0,0 +1,43 @@
#!/usr/bin/env python3
import subprocess, magic, shutil, click
from pathlib import Path
from getpass import getpass
from handle_acsm import handle_acsm
from xdg import xdg_config_home
__version__ = "1.0.0-alpha"
@click.version_option()
@click.command()
@click.argument(
"file",
type=click.Path(
exists=True,
file_okay=True,
dir_okay=False,
readable=True,
resolve_path=True
)
)
def main(file):
file = Path(file)
# make the config dir if it doesn't exist
(xdg_config_home() / 'knock').mkdir(parents=True, exist_ok=True)
file_type = file.suffix[1:].upper()
if file_type == 'ACSM':
click.echo('Received an ACSM (Adobe) file...')
handle_acsm(file)
else:
click.echo(f'Error: Files of type {file.suffix[1:].upper()} are not supported.\n', err=True)
click.echo('Only the following file types are currently supported:', err=True)
click.echo(' * ACSM (Adobe)\n')
click.echo('Please open a feature request at:', err=True)
click.echo(f' https://github.com/BentonEdmondson/knock/issues/new?title=Support%20{file_type}%20Files&labels=enhancement', err=True)
sys.exit(1)
if __name__ == "__main__":
main()