-
Notifications
You must be signed in to change notification settings - Fork 33
/
Copy pathimgcrush
executable file
·116 lines (85 loc) · 3.03 KB
/
imgcrush
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
#!/usr/bin/env python3
# Copyright 2018 The ChromiumOS Authors
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
"""Crush image files.
We try to shrink the image files by reencoding/stripping data losslessly.
We currently support png and jpg files.
"""
import logging
import multiprocessing
import os
from pathlib import Path
import sys
import libdot
# We pick a recent Chromium commit. It doesn't change much so shouldn't matter.
# Updating to the latest version shouldn't cause problems.
CHROMIUM_REF = "85.0.4148.0"
# Full path to Chromium's png crush script.
PNG_CRUSHER_URL = (
"https://chromium.googlesource.com/chromium/src/+/"
f"{CHROMIUM_REF}/tools/resources/optimize-png-files.sh"
)
# Our local cache of the script.
PNG_CRUSHER = libdot.BIN_DIR / f".png.crusher.{CHROMIUM_REF}"
# The tool used to crush jpeg images.
JPG_CRUSHER = "jpegoptim"
def update_png_crusher():
"""Update our local cache of Chromium's png optimizer script."""
if PNG_CRUSHER.exists():
return
for path in libdot.BIN_DIR.glob(".png.crusher.*"):
path.unlink()
libdot.fetch(f"{PNG_CRUSHER_URL}?format=TEXT", PNG_CRUSHER, b64=True)
PNG_CRUSHER.chmod(0o755)
def run(path, cmd):
"""Run the |cmd| for the |path| file."""
logging.info("Processing %s", path)
libdot.run([str(x) for x in cmd], check=False)
def process_file(pool, path):
"""Crush |path| as makes sense.
Jobs are thrown into the |pool|, but we don't currently bother checking
their return values. General life cycle management is handled by the pool.
"""
if path.suffix in (".png",):
update_png_crusher()
pool.apply_async(run, (path, [PNG_CRUSHER, path]))
elif path.suffix in (".jpg", ".jpeg"):
pool.apply_async(run, (path, [JPG_CRUSHER, path]))
else:
logging.debug("Skipping unknown file type %s", path.suffix)
def process_dir(pool, topdir):
"""Process all the paths under |topdir|."""
for root, dirs, files in os.walk(topdir):
root = Path(root)
# Not really needed, but makes things consistent.
dirs.sort()
files.sort()
for path in files:
process_file(pool, root / path)
def get_parser():
"""Get a command line parser."""
parser = libdot.ArgumentParser(description=__doc__)
parser.add_argument(
"paths",
nargs="+",
type=Path,
help="Image files or directories to crush.",
)
return parser
def main(argv):
"""The main func!"""
parser = get_parser()
opts = parser.parse_args(argv)
with multiprocessing.Pool() as pool:
# We walk the top set of args by hand to deref links.
for path in opts.paths:
if path.is_dir():
process_dir(pool, path)
elif path.is_file():
process_file(pool, path)
# Wait for the jobs to finish rather than let context manager terminate.
pool.close()
pool.join()
if __name__ == "__main__":
sys.exit(main(sys.argv[1:]))