Merge pull request #4794 from hugovk/rm-3.5

This commit is contained in:
Hugo van Kemenade 2020-09-02 21:28:58 +03:00 committed by GitHub
commit 5ce2fac3d6
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
74 changed files with 327 additions and 371 deletions

View File

@ -3,7 +3,7 @@ repos:
rev: e66be67b9b6811913470f70c28b4d50f94d05b22 # frozen: 20.8b1 rev: e66be67b9b6811913470f70c28b4d50f94d05b22 # frozen: 20.8b1
hooks: hooks:
- id: black - id: black
args: ["--target-version", "py35"] args: ["--target-version", "py36"]
# Only .py files, until https://github.com/psf/black/issues/402 resolved # Only .py files, until https://github.com/psf/black/issues/402 resolved
files: \.py$ files: \.py$
types: [] types: []

View File

@ -28,15 +28,17 @@ def timer(func, label, *args):
func(*args) func(*args)
if time.time() - starttime > 10: if time.time() - starttime > 10:
print( print(
"%s: breaking at %s iterations, %.6f per iteration" "{}: breaking at {} iterations, {:.6f} per iteration".format(
% (label, x + 1, (time.time() - starttime) / (x + 1.0)) label, x + 1, (time.time() - starttime) / (x + 1.0)
)
) )
break break
if x == iterations - 1: if x == iterations - 1:
endtime = time.time() endtime = time.time()
print( print(
"%s: %.4f s %.6f per iteration" "{}: {:.4f} s {:.6f} per iteration".format(
% (label, endtime - starttime, (endtime - starttime) / (x + 1.0)) label, endtime - starttime, (endtime - starttime) / (x + 1.0)
)
) )

View File

@ -26,7 +26,7 @@ def _test_leak(min_iterations, max_iterations, fn, *args, **kwargs):
if i < min_iterations: if i < min_iterations:
mem_limit = mem + 1 mem_limit = mem + 1
continue continue
msg = "memory usage limit exceeded after %d iterations" % (i + 1) msg = f"memory usage limit exceeded after {i + 1} iterations"
assert mem <= mem_limit, msg assert mem <= mem_limit, msg

View File

@ -42,8 +42,8 @@ def test_dos_total_memory():
info = PngImagePlugin.PngInfo() info = PngImagePlugin.PngInfo()
for x in range(64): for x in range(64):
info.add_text("t%s" % x, compressed_data, zip=True) info.add_text(f"t{x}", compressed_data, zip=True)
info.add_itxt("i%s" % x, compressed_data, zip=True) info.add_itxt(f"i{x}", compressed_data, zip=True)
b = BytesIO() b = BytesIO()
im.save(b, "PNG", pnginfo=info) im.save(b, "PNG", pnginfo=info)

View File

@ -9,4 +9,4 @@ def pytest_report_header(config):
features.pilinfo(out=out, supported_formats=False) features.pilinfo(out=out, supported_formats=False)
return out.getvalue() return out.getvalue()
except Exception as e: except Exception as e:
return "pytest_report_header failed: %s" % e return f"pytest_report_header failed: {e}"

View File

@ -6,7 +6,7 @@ if __name__ == "__main__":
# create font data chunk for embedding # create font data chunk for embedding
font = "Tests/images/courB08" font = "Tests/images/courB08"
print(" f._load_pilfont_data(") print(" f._load_pilfont_data(")
print(" # %s" % os.path.basename(font)) print(f" # {os.path.basename(font)}")
print(" BytesIO(base64.decodestring(b'''") print(" BytesIO(base64.decodestring(b'''")
with open(font + ".pil", "rb") as fp: with open(font + ".pil", "rb") as fp:
print(base64.b64encode(fp.read()).decode()) print(base64.b64encode(fp.read()).decode())

View File

@ -68,37 +68,31 @@ def convert_to_comparable(a, b):
def assert_deep_equal(a, b, msg=None): def assert_deep_equal(a, b, msg=None):
try: try:
assert len(a) == len(b), msg or "got length {}, expected {}".format( assert len(a) == len(b), msg or f"got length {len(a)}, expected {len(b)}"
len(a), len(b)
)
except Exception: except Exception:
assert a == b, msg assert a == b, msg
def assert_image(im, mode, size, msg=None): def assert_image(im, mode, size, msg=None):
if mode is not None: if mode is not None:
assert im.mode == mode, msg or "got mode {!r}, expected {!r}".format( assert im.mode == mode, (
im.mode, mode msg or f"got mode {repr(im.mode)}, expected {repr(mode)}"
) )
if size is not None: if size is not None:
assert im.size == size, msg or "got size {!r}, expected {!r}".format( assert im.size == size, (
im.size, size msg or f"got size {repr(im.size)}, expected {repr(size)}"
) )
def assert_image_equal(a, b, msg=None): def assert_image_equal(a, b, msg=None):
assert a.mode == b.mode, msg or "got mode {!r}, expected {!r}".format( assert a.mode == b.mode, msg or f"got mode {repr(a.mode)}, expected {repr(b.mode)}"
a.mode, b.mode assert a.size == b.size, msg or f"got size {repr(a.size)}, expected {repr(b.size)}"
)
assert a.size == b.size, msg or "got size {!r}, expected {!r}".format(
a.size, b.size
)
if a.tobytes() != b.tobytes(): if a.tobytes() != b.tobytes():
if HAS_UPLOADER: if HAS_UPLOADER:
try: try:
url = test_image_results.upload(a, b) url = test_image_results.upload(a, b)
logger.error("Url for test images: %s" % url) logger.error(f"Url for test images: {url}")
except Exception: except Exception:
pass pass
@ -113,12 +107,8 @@ def assert_image_equal_tofile(a, filename, msg=None, mode=None):
def assert_image_similar(a, b, epsilon, msg=None): def assert_image_similar(a, b, epsilon, msg=None):
assert a.mode == b.mode, msg or "got mode {!r}, expected {!r}".format( assert a.mode == b.mode, msg or f"got mode {repr(a.mode)}, expected {repr(b.mode)}"
a.mode, b.mode assert a.size == b.size, msg or f"got size {repr(a.size)}, expected {repr(b.size)}"
)
assert a.size == b.size, msg or "got size {!r}, expected {!r}".format(
a.size, b.size
)
a, b = convert_to_comparable(a, b) a, b = convert_to_comparable(a, b)
@ -130,13 +120,14 @@ def assert_image_similar(a, b, epsilon, msg=None):
ave_diff = diff / (a.size[0] * a.size[1]) ave_diff = diff / (a.size[0] * a.size[1])
try: try:
assert epsilon >= ave_diff, ( assert epsilon >= ave_diff, (
msg or "" (msg or "")
) + " average pixel value difference %.4f > epsilon %.4f" % (ave_diff, epsilon) + f" average pixel value difference {ave_diff:.4f} > epsilon {epsilon:.4f}"
)
except Exception as e: except Exception as e:
if HAS_UPLOADER: if HAS_UPLOADER:
try: try:
url = test_image_results.upload(a, b) url = test_image_results.upload(a, b)
logger.error("Url for test images: %s" % url) logger.error(f"Url for test images: {url}")
except Exception: except Exception:
pass pass
raise e raise e
@ -167,7 +158,7 @@ def assert_tuple_approx_equal(actuals, targets, threshold, msg):
def skip_unless_feature(feature): def skip_unless_feature(feature):
reason = "%s not available" % feature reason = f"{feature} not available"
return pytest.mark.skipif(not features.check(feature), reason=reason) return pytest.mark.skipif(not features.check(feature), reason=reason)
@ -205,7 +196,7 @@ class PillowLeakTestCase:
for cycle in range(self.iterations): for cycle in range(self.iterations):
core() core()
mem = self._get_mem_usage() - start_mem mem = self._get_mem_usage() - start_mem
msg = "memory usage limit exceeded in iteration %d" % cycle msg = f"memory usage limit exceeded in iteration {cycle}"
assert mem < self.mem_limit, msg assert mem < self.mem_limit, msg

View File

@ -50,7 +50,7 @@ def test_questionable():
with Image.open(f) as im: with Image.open(f) as im:
im.load() im.load()
if os.path.basename(f) not in supported: if os.path.basename(f) not in supported:
print("Please add %s to the partially supported bmp specs." % f) print(f"Please add {f} to the partially supported bmp specs.")
except Exception: # as msg: except Exception: # as msg:
if os.path.basename(f) in supported: if os.path.basename(f) in supported:
raise raise
@ -85,7 +85,7 @@ def test_good():
if name in file_map: if name in file_map:
return os.path.join(base, "html", file_map[name]) return os.path.join(base, "html", file_map[name])
name = os.path.splitext(name)[0] name = os.path.splitext(name)[0]
return os.path.join(base, "html", "%s.png" % name) return os.path.join(base, "html", f"{name}.png")
for f in get_files("g"): for f in get_files("g"):
try: try:
@ -108,4 +108,4 @@ def test_good():
os.path.join(base, "g", "pal8rle.bmp"), os.path.join(base, "g", "pal8rle.bmp"),
os.path.join(base, "g", "pal4rle.bmp"), os.path.join(base, "g", "pal4rle.bmp"),
) )
assert f in unsupported, "Unsupported Image {}: {}".format(f, msg) assert f in unsupported, f"Unsupported Image {f}: {msg}"

View File

@ -312,7 +312,7 @@ def test_apng_sequence_errors():
] ]
for f in test_files: for f in test_files:
with pytest.raises(SyntaxError): with pytest.raises(SyntaxError):
with Image.open("Tests/images/apng/{0}".format(f)) as im: with Image.open(f"Tests/images/apng/{f}") as im:
im.seek(im.n_frames - 1) im.seek(im.n_frames - 1)
im.load() im.load()

View File

@ -171,18 +171,18 @@ class TestFileLibTiff(LibTiffTestCase):
assert ( assert (
c_float(val[0][0] / val[0][1]).value c_float(val[0][0] / val[0][1]).value
== c_float(value[0][0] / value[0][1]).value == c_float(value[0][0] / value[0][1]).value
), ("%s didn't roundtrip" % tag) ), f"{tag} didn't roundtrip"
else: else:
assert c_float(val).value == c_float(value).value, ( assert (
"%s didn't roundtrip" % tag c_float(val).value == c_float(value).value
) ), f"{tag} didn't roundtrip"
else: else:
assert val == value, "%s didn't roundtrip" % tag assert val == value, f"{tag} didn't roundtrip"
# https://github.com/python-pillow/Pillow/issues/1561 # https://github.com/python-pillow/Pillow/issues/1561
requested_fields = ["StripByteCounts", "RowsPerStrip", "StripOffsets"] requested_fields = ["StripByteCounts", "RowsPerStrip", "StripOffsets"]
for field in requested_fields: for field in requested_fields:
assert field in reloaded, "%s not in metadata" % field assert field in reloaded, f"{field} not in metadata"
def test_additional_metadata(self, tmp_path): def test_additional_metadata(self, tmp_path):
# these should not crash. Seriously dummy data, most of it doesn't make # these should not crash. Seriously dummy data, most of it doesn't make

View File

@ -46,7 +46,7 @@ def test_others():
with Image.open(path) as im: with Image.open(path) as im:
im.load() im.load()
assert isinstance(im, SunImagePlugin.SunImageFile) assert isinstance(im, SunImagePlugin.SunImageFile)
target_path = "%s.png" % os.path.splitext(path)[0] target_path = f"{os.path.splitext(path)[0]}.png"
# im.save(target_file) # im.save(target_file)
with Image.open(target_path) as target: with Image.open(target_path) as target:
assert_image_equal(im, target) assert_image_equal(im, target)

View File

@ -36,9 +36,7 @@ def test_sanity(tmp_path):
assert_image_equal(saved_im, original_im) assert_image_equal(saved_im, original_im)
png_paths = glob( png_paths = glob(os.path.join(_TGA_DIR_COMMON, f"*x*_{mode.lower()}.png"))
os.path.join(_TGA_DIR_COMMON, "*x*_{}.png".format(mode.lower()))
)
for png_path in png_paths: for png_path in png_paths:
with Image.open(png_path) as reference_im: with Image.open(png_path) as reference_im:

View File

@ -145,16 +145,16 @@ def test_write_metadata(tmp_path):
assert_deep_equal( assert_deep_equal(
original[tag], original[tag],
value, value,
"{} didn't roundtrip, {}, {}".format(tag, original[tag], value), f"{tag} didn't roundtrip, {original[tag]}, {value}",
) )
else: else:
assert original[tag] == value, "{} didn't roundtrip, {}, {}".format( assert (
tag, original[tag], value original[tag] == value
) ), f"{tag} didn't roundtrip, {original[tag]}, {value}"
for tag, value in original.items(): for tag, value in original.items():
if tag not in ignored: if tag not in ignored:
assert value == reloaded[tag], "%s didn't roundtrip" % tag assert value == reloaded[tag], f"{tag} didn't roundtrip"
def test_change_stripbytecounts_tag_type(tmp_path): def test_change_stripbytecounts_tag_type(tmp_path):

View File

@ -47,11 +47,11 @@ def save_font(request, tmp_path, encoding):
font.save(tempname) font.save(tempname)
with Image.open(tempname.replace(".pil", ".pbm")) as loaded: with Image.open(tempname.replace(".pil", ".pbm")) as loaded:
with Image.open("Tests/fonts/ter-x20b-%s.pbm" % encoding) as target: with Image.open(f"Tests/fonts/ter-x20b-{encoding}.pbm") as target:
assert_image_equal(loaded, target) assert_image_equal(loaded, target)
with open(tempname, "rb") as f_loaded: with open(tempname, "rb") as f_loaded:
with open("Tests/fonts/ter-x20b-%s.pil" % encoding, "rb") as f_target: with open(f"Tests/fonts/ter-x20b-{encoding}.pil", "rb") as f_target:
assert f_loaded.read() == f_target.read() assert f_loaded.read() == f_target.read()
return tempname return tempname

View File

@ -128,16 +128,13 @@ class TestImageGetPixel(AccessTest):
im.putpixel((0, 0), c) im.putpixel((0, 0), c)
assert ( assert (
im.getpixel((0, 0)) == c im.getpixel((0, 0)) == c
), "put/getpixel roundtrip failed for mode {}, color {}".format(mode, c) ), f"put/getpixel roundtrip failed for mode {mode}, color {c}"
# check putpixel negative index # check putpixel negative index
im.putpixel((-1, -1), c) im.putpixel((-1, -1), c)
assert ( assert (
im.getpixel((-1, -1)) == c im.getpixel((-1, -1)) == c
), "put/getpixel roundtrip negative index failed for mode %s, color %s" % ( ), f"put/getpixel roundtrip negative index failed for mode {mode}, color {c}"
mode,
c,
)
# Check 0 # Check 0
im = Image.new(mode, (0, 0), None) im = Image.new(mode, (0, 0), None)
@ -155,11 +152,11 @@ class TestImageGetPixel(AccessTest):
im = Image.new(mode, (1, 1), c) im = Image.new(mode, (1, 1), c)
assert ( assert (
im.getpixel((0, 0)) == c im.getpixel((0, 0)) == c
), "initial color failed for mode {}, color {} ".format(mode, c) ), f"initial color failed for mode {mode}, color {c} "
# check initial color negative index # check initial color negative index
assert ( assert (
im.getpixel((-1, -1)) == c im.getpixel((-1, -1)) == c
), "initial color failed with negative index for mode %s, color %s " % (mode, c) ), f"initial color failed with negative index for mode {mode}, color {c} "
# Check 0 # Check 0
im = Image.new(mode, (0, 0), c) im = Image.new(mode, (0, 0), c)

View File

@ -162,8 +162,8 @@ def compare_reduce_with_reference(im, factor, average_diff=0.4, max_diff=1):
def assert_compare_images(a, b, max_average_diff, max_diff=255): def assert_compare_images(a, b, max_average_diff, max_diff=255):
assert a.mode == b.mode, "got mode %r, expected %r" % (a.mode, b.mode) assert a.mode == b.mode, f"got mode {repr(a.mode)}, expected {repr(b.mode)}"
assert a.size == b.size, "got size %r, expected %r" % (a.size, b.size) assert a.size == b.size, f"got size {repr(a.size)}, expected {repr(b.size)}"
a, b = convert_to_comparable(a, b) a, b = convert_to_comparable(a, b)
@ -176,16 +176,15 @@ def assert_compare_images(a, b, max_average_diff, max_diff=255):
a.size[0] * a.size[1] a.size[0] * a.size[1]
) )
msg = ( msg = (
"average pixel value difference {:.4f} > expected {:.4f} " f"average pixel value difference {average_diff:.4f} > "
"for '{}' band".format(average_diff, max_average_diff, band) f"expected {max_average_diff:.4f} for '{band}' band"
) )
assert max_average_diff >= average_diff, msg assert max_average_diff >= average_diff, msg
last_diff = [i for i, num in enumerate(ch_hist) if num > 0][-1] last_diff = [i for i, num in enumerate(ch_hist) if num > 0][-1]
assert ( assert max_diff >= last_diff, (
max_diff >= last_diff f"max pixel value difference {last_diff} > expected {max_diff} "
), "max pixel value difference {} > expected {} for '{}' band".format( f"for '{band}' band"
last_diff, max_diff, band
) )

View File

@ -82,15 +82,16 @@ class TestImagingCoreResampleAccuracy:
for y in range(case.size[1]): for y in range(case.size[1]):
for x in range(case.size[0]): for x in range(case.size[0]):
if c_px[x, y] != s_px[x, y]: if c_px[x, y] != s_px[x, y]:
message = "\nHave: \n{}\n\nExpected: \n{}".format( message = (
self.serialize_image(case), self.serialize_image(sample) f"\nHave: \n{self.serialize_image(case)}\n"
f"\nExpected: \n{self.serialize_image(sample)}"
) )
assert s_px[x, y] == c_px[x, y], message assert s_px[x, y] == c_px[x, y], message
def serialize_image(self, image): def serialize_image(self, image):
s_px = image.load() s_px = image.load()
return "\n".join( return "\n".join(
" ".join("{:02x}".format(s_px[x, y]) for x in range(image.size[0])) " ".join(f"{s_px[x, y]:02x}" for x in range(image.size[0]))
for y in range(image.size[1]) for y in range(image.size[1])
) )
@ -230,7 +231,7 @@ class TestCoreResampleConsistency:
for x in range(channel.size[0]): for x in range(channel.size[0]):
for y in range(channel.size[1]): for y in range(channel.size[1]):
if px[x, y] != color: if px[x, y] != color:
message = "{} != {} for pixel {}".format(px[x, y], color, (x, y)) message = f"{px[x, y]} != {color} for pixel {(x, y)}"
assert px[x, y] == color, message assert px[x, y] == color, message
def test_8u(self): def test_8u(self):
@ -269,10 +270,9 @@ class TestCoreResampleAlphaCorrect:
px = i.load() px = i.load()
for y in range(i.size[1]): for y in range(i.size[1]):
used_colors = {px[x, y][0] for x in range(i.size[0])} used_colors = {px[x, y][0] for x in range(i.size[0])}
assert 256 == len( assert 256 == len(used_colors), (
used_colors "All colors should be present in resized image. "
), "All colors should present in resized image. Only {} on {} line.".format( f"Only {len(used_colors)} on {y} line."
len(used_colors), y
) )
@pytest.mark.xfail(reason="Current implementation isn't precise enough") @pytest.mark.xfail(reason="Current implementation isn't precise enough")
@ -308,8 +308,9 @@ class TestCoreResampleAlphaCorrect:
for y in range(i.size[1]): for y in range(i.size[1]):
for x in range(i.size[0]): for x in range(i.size[0]):
if px[x, y][-1] != 0 and px[x, y][:-1] != clean_pixel: if px[x, y][-1] != 0 and px[x, y][:-1] != clean_pixel:
message = "pixel at ({}, {}) is differ:\n{}\n{}".format( message = (
x, y, px[x, y], clean_pixel f"pixel at ({x}, {y}) is different:\n"
f"{px[x, y]}\n{clean_pixel}"
) )
assert px[x, y][:3] == clean_pixel, message assert px[x, y][:3] == clean_pixel, message
@ -504,7 +505,7 @@ class TestCoreResampleBox:
]: ]:
res = im.resize(size, Image.LANCZOS, box) res = im.resize(size, Image.LANCZOS, box)
assert res.size == size assert res.size == size
assert_image_equal(res, im.crop(box), ">>> {} {}".format(size, box)) assert_image_equal(res, im.crop(box), f">>> {size} {box}")
def test_no_passthrough(self): def test_no_passthrough(self):
# When resize is required # When resize is required
@ -520,9 +521,7 @@ class TestCoreResampleBox:
assert res.size == size assert res.size == size
with pytest.raises(AssertionError, match=r"difference \d"): with pytest.raises(AssertionError, match=r"difference \d"):
# check that the difference at least that much # check that the difference at least that much
assert_image_similar( assert_image_similar(res, im.crop(box), 20, f">>> {size} {box}")
res, im.crop(box), 20, ">>> {} {}".format(size, box)
)
def test_skip_horizontal(self): def test_skip_horizontal(self):
# Can skip resize for one dimension # Can skip resize for one dimension
@ -542,7 +541,7 @@ class TestCoreResampleBox:
res, res,
im.crop(box).resize(size, flt), im.crop(box).resize(size, flt),
0.4, 0.4,
">>> {} {} {}".format(size, box, flt), f">>> {size} {box} {flt}",
) )
def test_skip_vertical(self): def test_skip_vertical(self):
@ -563,5 +562,5 @@ class TestCoreResampleBox:
res, res,
im.crop(box).resize(size, flt), im.crop(box).resize(size, flt),
0.4, 0.4,
">>> {} {} {}".format(size, box, flt), f">>> {size} {box} {flt}",
) )

View File

@ -182,7 +182,7 @@ def helper_chord(mode, bbox, start, end):
# Arrange # Arrange
im = Image.new(mode, (W, H)) im = Image.new(mode, (W, H))
draw = ImageDraw.Draw(im) draw = ImageDraw.Draw(im)
expected = "Tests/images/imagedraw_chord_{}.png".format(mode) expected = f"Tests/images/imagedraw_chord_{mode}.png"
# Act # Act
draw.chord(bbox, start, end, fill="red", outline="yellow") draw.chord(bbox, start, end, fill="red", outline="yellow")
@ -244,7 +244,7 @@ def helper_ellipse(mode, bbox):
# Arrange # Arrange
im = Image.new(mode, (W, H)) im = Image.new(mode, (W, H))
draw = ImageDraw.Draw(im) draw = ImageDraw.Draw(im)
expected = "Tests/images/imagedraw_ellipse_{}.png".format(mode) expected = f"Tests/images/imagedraw_ellipse_{mode}.png"
# Act # Act
draw.ellipse(bbox, fill="green", outline="blue") draw.ellipse(bbox, fill="green", outline="blue")
@ -514,7 +514,7 @@ def test_polygon_kite():
# Arrange # Arrange
im = Image.new(mode, (W, H)) im = Image.new(mode, (W, H))
draw = ImageDraw.Draw(im) draw = ImageDraw.Draw(im)
expected = "Tests/images/imagedraw_polygon_kite_{}.png".format(mode) expected = f"Tests/images/imagedraw_polygon_kite_{mode}.png"
# Act # Act
draw.polygon(KITE_POINTS, fill="blue", outline="yellow") draw.polygon(KITE_POINTS, fill="blue", outline="yellow")
@ -1091,7 +1091,5 @@ def test_same_color_outline():
draw_method(*args) draw_method(*args)
# Assert # Assert
expected = "Tests/images/imagedraw_outline_{}_{}.png".format( expected = f"Tests/images/imagedraw_outline_{operation}_{mode}.png"
operation, mode
)
assert_image_similar_tofile(im, expected, 1) assert_image_similar_tofile(im, expected, 1)

View File

@ -55,7 +55,7 @@ def helper_ellipse(mode, bbox):
draw = ImageDraw2.Draw(im) draw = ImageDraw2.Draw(im)
pen = ImageDraw2.Pen("blue", width=2) pen = ImageDraw2.Pen("blue", width=2)
brush = ImageDraw2.Brush("green") brush = ImageDraw2.Brush("green")
expected = "Tests/images/imagedraw_ellipse_{}.png".format(mode) expected = f"Tests/images/imagedraw_ellipse_{mode}.png"
# Act # Act
draw.ellipse(bbox, pen, brush) draw.ellipse(bbox, pen, brush)

View File

@ -35,7 +35,7 @@ def _check_alpha(im, original, op, amount):
assert_image_equal( assert_image_equal(
im.getchannel("A"), im.getchannel("A"),
original.getchannel("A"), original.getchannel("A"),
"Diff on {}: {}".format(op, amount), f"Diff on {op}: {amount}",
) )

View File

@ -3,7 +3,7 @@ from PIL import Image, ImageMath
def pixel(im): def pixel(im):
if hasattr(im, "im"): if hasattr(im, "im"):
return "{} {!r}".format(im.mode, im.getpixel((0, 0))) return "{} {}".format(im.mode, repr(im.getpixel((0, 0))))
else: else:
if isinstance(im, int): if isinstance(im, int):
return int(im) # hack to deal with booleans return int(im) # hack to deal with booleans

View File

@ -65,7 +65,7 @@ def create_lut():
for op in ("corner", "dilation4", "dilation8", "erosion4", "erosion8", "edge"): for op in ("corner", "dilation4", "dilation8", "erosion4", "erosion8", "edge"):
lb = ImageMorph.LutBuilder(op_name=op) lb = ImageMorph.LutBuilder(op_name=op)
lut = lb.build_lut() lut = lb.build_lut()
with open("Tests/images/%s.lut" % op, "wb") as f: with open(f"Tests/images/{op}.lut", "wb") as f:
f.write(lut) f.write(lut)
@ -76,7 +76,7 @@ def test_lut():
assert lb.get_lut() is None assert lb.get_lut() is None
lut = lb.build_lut() lut = lb.build_lut()
with open("Tests/images/%s.lut" % op, "rb") as f: with open(f"Tests/images/{op}.lut", "rb") as f:
assert lut == bytearray(f.read()) assert lut == bytearray(f.read())

View File

@ -27,7 +27,7 @@ def setup_module():
tk.Frame() tk.Frame()
# root = tk.Tk() # root = tk.Tk()
except tk.TclError as v: except tk.TclError as v:
pytest.skip("TCL Error: %s" % v) pytest.skip(f"TCL Error: {v}")
def test_kw(): def test_kw():

View File

@ -15,9 +15,9 @@ def verify(im1):
xy = x, y xy = x, y
p1 = pix1[xy] p1 = pix1[xy]
p2 = pix2[xy] p2 = pix2[xy]
assert p1 == p2, "got {!r} from mode {} at {}, expected {!r}".format( assert (
p1, im1.mode, xy, p2 p1 == p2
) ), f"got {repr(p1)} from mode {im1.mode} at {xy}, expected {repr(p2)}"
def test_basic(tmp_path): def test_basic(tmp_path):

View File

@ -43,7 +43,7 @@ def test_sanity(tmp_path):
continue continue
# Test saving the file # Test saving the file
tempfile = str(tmp_path / "temp_{}.png".format(mode)) tempfile = str(tmp_path / f"temp_{mode}.png")
data.save(tempfile) data.save(tempfile)
# Check that it actually worked. # Check that it actually worked.

View File

@ -16,5 +16,5 @@ class TestToQPixmap(PillowQPixmapTestCase):
assert not data.isNull() assert not data.isNull()
# Test saving the file # Test saving the file
tempfile = str(tmp_path / "temp_{}.png".format(mode)) tempfile = str(tmp_path / f"temp_{mode}.png")
data.save(tempfile) data.save(tempfile)

View File

@ -14,7 +14,6 @@ def test_is_path():
assert it_is assert it_is
@pytest.mark.skipif(not _util.py36, reason="os.path support for Paths added in 3.6")
def test_path_obj_is_path(): def test_path_obj_is_path():
# Arrange # Arrange
from pathlib import Path from pathlib import Path

View File

@ -212,10 +212,10 @@ class DdsImageFile(ImageFile.ImageFile):
def _open(self): def _open(self):
magic, header_size = struct.unpack("<II", self.fp.read(8)) magic, header_size = struct.unpack("<II", self.fp.read(8))
if header_size != 124: if header_size != 124:
raise OSError("Unsupported header size %r" % (header_size)) raise OSError(f"Unsupported header size {repr(header_size)}")
header_bytes = self.fp.read(header_size - 4) header_bytes = self.fp.read(header_size - 4)
if len(header_bytes) != 120: if len(header_bytes) != 120:
raise OSError("Incomplete header: %s bytes" % len(header_bytes)) raise OSError(f"Incomplete header: {len(header_bytes)} bytes")
header = BytesIO(header_bytes) header = BytesIO(header_bytes)
flags, height, width = struct.unpack("<3I", header.read(12)) flags, height, width = struct.unpack("<3I", header.read(12))
@ -235,7 +235,7 @@ class DdsImageFile(ImageFile.ImageFile):
elif fourcc == b"DXT5": elif fourcc == b"DXT5":
self.decoder = "DXT5" self.decoder = "DXT5"
else: else:
raise NotImplementedError("Unimplemented pixel format %r" % fourcc) raise NotImplementedError(f"Unimplemented pixel format {fourcc}")
self.tile = [(self.decoder, (0, 0) + self.size, 0, (self.mode, 0, 1))] self.tile = [(self.decoder, (0, 0) + self.size, 0, (self.mode, 0, 1))]

View File

@ -124,7 +124,7 @@ Identify Image Files
for infile in sys.argv[1:]: for infile in sys.argv[1:]:
try: try:
with Image.open(infile) as im: with Image.open(infile) as im:
print(infile, im.format, "%dx%d" % im.size, im.mode) print(infile, im.format, f"{im.size}x{im.mode}")
except OSError: except OSError:
pass pass

View File

@ -18,7 +18,7 @@ Example: Vary the sharpness of an image
for i in range(8): for i in range(8):
factor = i / 4.0 factor = i / 4.0
enhancer.enhance(factor).show("Sharpness %f" % factor) enhancer.enhance(factor).show(f"Sharpness {factor:f}")
Also see the :file:`enhancer.py` demo program in the :file:`Scripts/` Also see the :file:`enhancer.py` demo program in the :file:`Scripts/`
directory. directory.

View File

@ -17,7 +17,7 @@ Extracting frames from an animation
with Image.open("animation.fli") as im: with Image.open("animation.fli") as im:
index = 1 index = 1
for frame in ImageSequence.Iterator(im): for frame in ImageSequence.Iterator(im):
frame.save("frame%d.png" % index) frame.save(f"frame{index}.png")
index += 1 index += 1
The :py:class:`~PIL.ImageSequence.Iterator` class The :py:class:`~PIL.ImageSequence.Iterator` class

View File

@ -21,7 +21,7 @@ from setuptools.command.build_ext import build_ext
def get_version(): def get_version():
version_file = "src/PIL/_version.py" version_file = "src/PIL/_version.py"
with open(version_file, "r") as f: with open(version_file) as f:
exec(compile(f.read(), version_file, "exec")) exec(compile(f.read(), version_file, "exec"))
return locals()["__version__"] return locals()["__version__"]
@ -39,10 +39,10 @@ ZLIB_ROOT = None
if sys.platform == "win32" and sys.version_info >= (3, 9): if sys.platform == "win32" and sys.version_info >= (3, 9):
warnings.warn( warnings.warn(
"Pillow {} does not support Python {}.{} and does not provide prebuilt " f"Pillow {PILLOW_VERSION} does not support Python "
"Windows binaries. We do not recommend building from source on Windows.".format( f"{sys.version_info.major}.{sys.version_info.minor} and does not provide "
PILLOW_VERSION, sys.version_info.major, sys.version_info.minor "prebuilt Windows binaries. We do not recommend building from source on "
), "Windows.",
RuntimeWarning, RuntimeWarning,
) )
@ -174,7 +174,7 @@ def _find_library_dirs_ldconfig():
# Assuming GLIBC's ldconfig (with option -p) # Assuming GLIBC's ldconfig (with option -p)
# Alpine Linux uses musl that can't print cache # Alpine Linux uses musl that can't print cache
args = ["/sbin/ldconfig", "-p"] args = ["/sbin/ldconfig", "-p"]
expr = r".*\(%s.*\) => (.*)" % abi_type expr = fr".*\({abi_type}.*\) => (.*)"
env = dict(os.environ) env = dict(os.environ)
env["LC_ALL"] = "C" env["LC_ALL"] = "C"
env["LANG"] = "C" env["LANG"] = "C"
@ -302,8 +302,8 @@ class pil_build_ext(build_ext):
user_options = ( user_options = (
build_ext.user_options build_ext.user_options
+ [("disable-%s" % x, None, "Disable support for %s" % x) for x in feature] + [(f"disable-{x}", None, f"Disable support for {x}") for x in feature]
+ [("enable-%s" % x, None, "Enable support for %s" % x) for x in feature] + [(f"enable-{x}", None, f"Enable support for {x}") for x in feature]
+ [ + [
("disable-platform-guessing", None, "Disable platform guessing on Linux"), ("disable-platform-guessing", None, "Disable platform guessing on Linux"),
("debug", None, "Debug logging"), ("debug", None, "Debug logging"),
@ -316,8 +316,8 @@ class pil_build_ext(build_ext):
self.add_imaging_libs = "" self.add_imaging_libs = ""
build_ext.initialize_options(self) build_ext.initialize_options(self)
for x in self.feature: for x in self.feature:
setattr(self, "disable_%s" % x, None) setattr(self, f"disable_{x}", None)
setattr(self, "enable_%s" % x, None) setattr(self, f"enable_{x}", None)
def finalize_options(self): def finalize_options(self):
build_ext.finalize_options(self) build_ext.finalize_options(self)
@ -334,15 +334,15 @@ class pil_build_ext(build_ext):
except TypeError: except TypeError:
self.parallel = None self.parallel = None
for x in self.feature: for x in self.feature:
if getattr(self, "disable_%s" % x): if getattr(self, f"disable_{x}"):
setattr(self.feature, x, False) setattr(self.feature, x, False)
self.feature.required.discard(x) self.feature.required.discard(x)
_dbg("Disabling %s", x) _dbg("Disabling %s", x)
if getattr(self, "enable_%s" % x): if getattr(self, f"enable_{x}"):
raise ValueError( raise ValueError(
"Conflicting options: --enable-{} and --disable-{}".format(x, x) f"Conflicting options: --enable-{x} and --disable-{x}"
) )
if getattr(self, "enable_%s" % x): if getattr(self, f"enable_{x}"):
_dbg("Requiring %s", x) _dbg("Requiring %s", x)
self.feature.required.add(x) self.feature.required.add(x)
@ -393,12 +393,12 @@ class pil_build_ext(build_ext):
if root is None and pkg_config: if root is None and pkg_config:
if isinstance(lib_name, tuple): if isinstance(lib_name, tuple):
for lib_name2 in lib_name: for lib_name2 in lib_name:
_dbg("Looking for `%s` using pkg-config." % lib_name2) _dbg(f"Looking for `{lib_name2}` using pkg-config.")
root = pkg_config(lib_name2) root = pkg_config(lib_name2)
if root: if root:
break break
else: else:
_dbg("Looking for `%s` using pkg-config." % lib_name) _dbg(f"Looking for `{lib_name}` using pkg-config.")
root = pkg_config(lib_name) root = pkg_config(lib_name)
if isinstance(root, tuple): if isinstance(root, tuple):
@ -740,9 +740,9 @@ class pil_build_ext(build_ext):
and sys.version_info < (3, 9) and sys.version_info < (3, 9)
and not (PLATFORM_PYPY or PLATFORM_MINGW) and not (PLATFORM_PYPY or PLATFORM_MINGW)
): ):
defs.append(("PILLOW_VERSION", '"\\"%s\\""' % PILLOW_VERSION)) defs.append(("PILLOW_VERSION", f'"\\"{PILLOW_VERSION}\\""'))
else: else:
defs.append(("PILLOW_VERSION", '"%s"' % PILLOW_VERSION)) defs.append(("PILLOW_VERSION", f'"{PILLOW_VERSION}"'))
self._update_extension("PIL._imaging", libs, defs) self._update_extension("PIL._imaging", libs, defs)
@ -792,11 +792,11 @@ class pil_build_ext(build_ext):
print("-" * 68) print("-" * 68)
print("PIL SETUP SUMMARY") print("PIL SETUP SUMMARY")
print("-" * 68) print("-" * 68)
print("version Pillow %s" % PILLOW_VERSION) print(f"version Pillow {PILLOW_VERSION}")
v = sys.version.split("[") v = sys.version.split("[")
print("platform {} {}".format(sys.platform, v[0].strip())) print(f"platform {sys.platform} {v[0].strip()}")
for v in v[1:]: for v in v[1:]:
print(" [%s" % v.strip()) print(f" [{v.strip()}")
print("-" * 68) print("-" * 68)
options = [ options = [
@ -817,10 +817,10 @@ class pil_build_ext(build_ext):
if option[0]: if option[0]:
version = "" version = ""
if len(option) >= 3 and option[2]: if len(option) >= 3 and option[2]:
version = " (%s)" % option[2] version = f" ({option[2]})"
print("--- {} support available{}".format(option[1], version)) print(f"--- {option[1]} support available{version}")
else: else:
print("*** %s support not available" % option[1]) print(f"*** {option[1]} support not available")
all = 0 all = 0
print("-" * 68) print("-" * 68)
@ -903,28 +903,23 @@ try:
zip_safe=not (debug_build() or PLATFORM_MINGW), zip_safe=not (debug_build() or PLATFORM_MINGW),
) )
except RequiredDependencyException as err: except RequiredDependencyException as err:
msg = """ msg = f"""
The headers or library files could not be found for %s, The headers or library files could not be found for {str(err)},
a required dependency when compiling Pillow from source. a required dependency when compiling Pillow from source.
Please see the install instructions at: Please see the install instructions at:
https://pillow.readthedocs.io/en/latest/installation.html https://pillow.readthedocs.io/en/latest/installation.html
""" % ( """
str(err)
)
sys.stderr.write(msg) sys.stderr.write(msg)
raise RequiredDependencyException(msg) raise RequiredDependencyException(msg)
except DependencyException as err: except DependencyException as err:
msg = """ msg = f"""
The headers or library files could not be found for %s, The headers or library files could not be found for {str(err)},
which was requested by the option flag --enable-%s which was requested by the option flag --enable-{str(err)}
""" % ( """
str(err),
str(err),
)
sys.stderr.write(msg) sys.stderr.write(msg)
raise DependencyException(msg) raise DependencyException(msg)

View File

@ -250,7 +250,7 @@ class BlpImageFile(ImageFile.ImageFile):
decoder = "BLP2" decoder = "BLP2"
self.mode = "RGBA" if self._blp_alpha_depth else "RGB" self.mode = "RGBA" if self._blp_alpha_depth else "RGB"
else: else:
raise BLPFormatError("Bad BLP magic %r" % (self.magic)) raise BLPFormatError(f"Bad BLP magic {repr(self.magic)}")
self.tile = [(decoder, (0, 0) + self.size, 0, (self.mode, 0, 1))] self.tile = [(decoder, (0, 0) + self.size, 0, (self.mode, 0, 1))]
@ -336,11 +336,11 @@ class BLP1Decoder(_BLPBaseDecoder):
self.set_as_raw(bytes(data)) self.set_as_raw(bytes(data))
else: else:
raise BLPFormatError( raise BLPFormatError(
"Unsupported BLP encoding %r" % (self._blp_encoding) f"Unsupported BLP encoding {repr(self._blp_encoding)}"
) )
else: else:
raise BLPFormatError( raise BLPFormatError(
"Unsupported BLP compression %r" % (self._blp_encoding) f"Unsupported BLP compression {repr(self._blp_encoding)}"
) )
def _decode_jpeg_stream(self): def _decode_jpeg_stream(self):
@ -400,13 +400,15 @@ class BLP2Decoder(_BLPBaseDecoder):
data += d data += d
else: else:
raise BLPFormatError( raise BLPFormatError(
"Unsupported alpha encoding %r" % (self._blp_alpha_encoding) f"Unsupported alpha encoding {repr(self._blp_alpha_encoding)}"
) )
else: else:
raise BLPFormatError("Unknown BLP encoding %r" % (self._blp_encoding)) raise BLPFormatError(f"Unknown BLP encoding {repr(self._blp_encoding)}")
else: else:
raise BLPFormatError("Unknown BLP compression %r" % (self._blp_compression)) raise BLPFormatError(
f"Unknown BLP compression {repr(self._blp_compression)}"
)
self.set_as_raw(bytes(data)) self.set_as_raw(bytes(data))

View File

@ -149,7 +149,7 @@ class BmpImageFile(ImageFile.ImageFile):
file_info["a_mask"], file_info["a_mask"],
) )
else: else:
raise OSError("Unsupported BMP header type (%d)" % file_info["header_size"]) raise OSError(f"Unsupported BMP header type ({file_info['header_size']})")
# ------------------ Special case : header is reported 40, which # ------------------ Special case : header is reported 40, which
# ---------------------- is shorter than real size for bpp >= 16 # ---------------------- is shorter than real size for bpp >= 16
@ -169,7 +169,7 @@ class BmpImageFile(ImageFile.ImageFile):
# ---------------------- Check bit depth for unusual unsupported values # ---------------------- Check bit depth for unusual unsupported values
self.mode, raw_mode = BIT2MODE.get(file_info["bits"], (None, None)) self.mode, raw_mode = BIT2MODE.get(file_info["bits"], (None, None))
if self.mode is None: if self.mode is None:
raise OSError("Unsupported BMP pixel depth (%d)" % file_info["bits"]) raise OSError(f"Unsupported BMP pixel depth ({file_info['bits']})")
# ---------------- Process BMP with Bitfields compression (not palette) # ---------------- Process BMP with Bitfields compression (not palette)
if file_info["compression"] == self.BITFIELDS: if file_info["compression"] == self.BITFIELDS:
@ -214,14 +214,14 @@ class BmpImageFile(ImageFile.ImageFile):
if file_info["bits"] == 32 and header == 22: # 32-bit .cur offset if file_info["bits"] == 32 and header == 22: # 32-bit .cur offset
raw_mode, self.mode = "BGRA", "RGBA" raw_mode, self.mode = "BGRA", "RGBA"
else: else:
raise OSError("Unsupported BMP compression (%d)" % file_info["compression"]) raise OSError(f"Unsupported BMP compression ({file_info['compression']})")
# --------------- Once the header is processed, process the palette/LUT # --------------- Once the header is processed, process the palette/LUT
if self.mode == "P": # Paletted for 1, 4 and 8 bit images if self.mode == "P": # Paletted for 1, 4 and 8 bit images
# ---------------------------------------------------- 1-bit images # ---------------------------------------------------- 1-bit images
if not (0 < file_info["colors"] <= 65536): if not (0 < file_info["colors"] <= 65536):
raise OSError("Unsupported BMP Palette size (%d)" % file_info["colors"]) raise OSError(f"Unsupported BMP Palette size ({file_info['colors']})")
else: else:
padding = file_info["palette_padding"] padding = file_info["palette_padding"]
palette = read(padding * file_info["colors"]) palette = read(padding * file_info["colors"])
@ -310,7 +310,7 @@ def _save(im, fp, filename, bitmap_header=True):
try: try:
rawmode, bits, colors = SAVE[im.mode] rawmode, bits, colors = SAVE[im.mode]
except KeyError as e: except KeyError as e:
raise OSError("cannot write mode %s as BMP" % im.mode) from e raise OSError(f"cannot write mode {im.mode} as BMP") from e
info = im.encoderinfo info = im.encoderinfo

View File

@ -106,10 +106,10 @@ class DdsImageFile(ImageFile.ImageFile):
def _open(self): def _open(self):
magic, header_size = struct.unpack("<II", self.fp.read(8)) magic, header_size = struct.unpack("<II", self.fp.read(8))
if header_size != 124: if header_size != 124:
raise OSError("Unsupported header size %r" % (header_size)) raise OSError(f"Unsupported header size {repr(header_size)}")
header_bytes = self.fp.read(header_size - 4) header_bytes = self.fp.read(header_size - 4)
if len(header_bytes) != 120: if len(header_bytes) != 120:
raise OSError("Incomplete header: %s bytes" % len(header_bytes)) raise OSError(f"Incomplete header: {len(header_bytes)} bytes")
header = BytesIO(header_bytes) header = BytesIO(header_bytes)
flags, height, width = struct.unpack("<3I", header.read(12)) flags, height, width = struct.unpack("<3I", header.read(12))
@ -159,10 +159,10 @@ class DdsImageFile(ImageFile.ImageFile):
n = 7 n = 7
else: else:
raise NotImplementedError( raise NotImplementedError(
"Unimplemented DXGI format %d" % (dxgi_format) f"Unimplemented DXGI format {dxgi_format}"
) )
else: else:
raise NotImplementedError("Unimplemented pixel format %r" % (fourcc)) raise NotImplementedError(f"Unimplemented pixel format {repr(fourcc)}")
self.tile = [("bcn", (0, 0) + self.size, data_start, (n))] self.tile = [("bcn", (0, 0) + self.size, data_start, (n))]

View File

@ -118,10 +118,10 @@ def Ghostscript(tile, size, fp, scale=1):
"-dNOPAUSE", # don't pause between pages "-dNOPAUSE", # don't pause between pages
"-dSAFER", # safe mode "-dSAFER", # safe mode
"-sDEVICE=ppmraw", # ppm driver "-sDEVICE=ppmraw", # ppm driver
"-sOutputFile=%s" % outfile, # output file f"-sOutputFile={outfile}", # output file
# adjust for image origin # adjust for image origin
"-c", "-c",
"%d %d translate" % (-bbox[0], -bbox[1]), f"{-bbox[0]} {-bbox[1]} translate",
"-f", "-f",
infile, # input file infile, # input file
# showpage (see https://bugs.ghostscript.com/show_bug.cgi?id=698272) # showpage (see https://bugs.ghostscript.com/show_bug.cgi?id=698272)
@ -386,10 +386,10 @@ def _save(im, fp, filename, eps=1):
# image header # image header
fp.write("gsave\n") fp.write("gsave\n")
fp.write("10 dict begin\n") fp.write("10 dict begin\n")
fp.write("/buf %d string def\n" % (im.size[0] * operator[1])) fp.write(f"/buf {im.size[0] * operator[1]} string def\n")
fp.write("%d %d scale\n" % im.size) fp.write("%d %d scale\n" % im.size)
fp.write("%d %d 8\n" % im.size) # <= bits fp.write("%d %d 8\n" % im.size) # <= bits
fp.write("[%d 0 0 -%d 0 %d]\n" % (im.size[0], im.size[1], im.size[1])) fp.write(f"[{im.size[0]} 0 0 -{im.size[1]} 0 {im.size[1]}]\n")
fp.write("{ currentfile buf readhexstring pop } bind\n") fp.write("{ currentfile buf readhexstring pop } bind\n")
fp.write(operator[2] + "\n") fp.write(operator[2] + "\n")
if hasattr(fp, "flush"): if hasattr(fp, "flush"):

View File

@ -133,7 +133,7 @@ class FliImageFile(ImageFile.ImageFile):
self.load() self.load()
if frame != self.__frame + 1: if frame != self.__frame + 1:
raise ValueError("cannot seek to frame %d" % frame) raise ValueError(f"cannot seek to frame {frame}")
self.__frame = frame self.__frame = frame
# move to next frame # move to next frame

View File

@ -101,7 +101,7 @@ class FontFile:
# font metrics # font metrics
with open(os.path.splitext(filename)[0] + ".pil", "wb") as fp: with open(os.path.splitext(filename)[0] + ".pil", "wb") as fp:
fp.write(b"PILfont\n") fp.write(b"PILfont\n")
fp.write((";;;;;;%d;\n" % self.ysize).encode("ascii")) # HACK!!! fp.write(f";;;;;;{self.ysize};\n".encode("ascii")) # HACK!!!
fp.write(b"DATA\n") fp.write(b"DATA\n")
for id in range(256): for id in range(256):
m = self.metrics[id] m = self.metrics[id]

View File

@ -73,7 +73,7 @@ class FpxImageFile(ImageFile.ImageFile):
# get the Image Contents Property Set # get the Image Contents Property Set
prop = self.ole.getproperties( prop = self.ole.getproperties(
["Data Object Store %06d" % index, "\005Image Contents"] [f"Data Object Store {index:06d}", "\005Image Contents"]
) )
# size (highest resolution) # size (highest resolution)
@ -121,8 +121,8 @@ class FpxImageFile(ImageFile.ImageFile):
# setup tile descriptors for a given subimage # setup tile descriptors for a given subimage
stream = [ stream = [
"Data Object Store %06d" % index, f"Data Object Store {index:06d}",
"Resolution %04d" % subimage, f"Resolution {subimage:04d}",
"Subimage 0000 Header", "Subimage 0000 Header",
] ]

View File

@ -89,7 +89,7 @@ class FtexImageFile(ImageFile.ImageFile):
elif format == FORMAT_UNCOMPRESSED: elif format == FORMAT_UNCOMPRESSED:
self.tile = [("raw", (0, 0) + self.size, 0, ("RGB", 0, 1))] self.tile = [("raw", (0, 0) + self.size, 0, ("RGB", 0, 1))]
else: else:
raise ValueError("Invalid texture compression format: %r" % (format)) raise ValueError(f"Invalid texture compression format: {repr(format)}")
self.fp.close() self.fp.close()
self.fp = BytesIO(data) self.fp = BytesIO(data)

View File

@ -47,7 +47,7 @@ class GbrImageFile(ImageFile.ImageFile):
if header_size < 20: if header_size < 20:
raise SyntaxError("not a GIMP brush") raise SyntaxError("not a GIMP brush")
if version not in (1, 2): if version not in (1, 2):
raise SyntaxError("Unsupported GIMP brush version: %s" % version) raise SyntaxError(f"Unsupported GIMP brush version: {version}")
width = i32(self.fp.read(4)) width = i32(self.fp.read(4))
height = i32(self.fp.read(4)) height = i32(self.fp.read(4))
@ -55,7 +55,7 @@ class GbrImageFile(ImageFile.ImageFile):
if width <= 0 or height <= 0: if width <= 0 or height <= 0:
raise SyntaxError("not a GIMP brush") raise SyntaxError("not a GIMP brush")
if color_depth not in (1, 4): if color_depth not in (1, 4):
raise SyntaxError("Unsupported GIMP brush color depth: %s" % color_depth) raise SyntaxError(f"Unsupported GIMP brush color depth: {color_depth}")
if version == 1: if version == 1:
comment_length = header_size - 20 comment_length = header_size - 20

View File

@ -154,7 +154,7 @@ class GifImageFile(ImageFile.ImageFile):
self.load() self.load()
if frame != self.__frame + 1: if frame != self.__frame + 1:
raise ValueError("cannot seek to frame %d" % frame) raise ValueError(f"cannot seek to frame {frame}")
self.__frame = frame self.__frame = frame
self.tile = [] self.tile = []

View File

@ -83,7 +83,7 @@ def read_32(fobj, start_length, size):
if bytesleft <= 0: if bytesleft <= 0:
break break
if bytesleft != 0: if bytesleft != 0:
raise SyntaxError("Error reading channel [%r left]" % bytesleft) raise SyntaxError(f"Error reading channel [{repr(bytesleft)} left]")
band = Image.frombuffer("L", pixel_size, b"".join(data), "raw", "L", 0, 1) band = Image.frombuffer("L", pixel_size, b"".join(data), "raw", "L", 0, 1)
im.im.putband(band.im, band_ix) im.im.putband(band.im, band_ix)
return {"RGB": im} return {"RGB": im}
@ -321,7 +321,7 @@ def _save(im, fp, filename):
last_w = None last_w = None
second_path = None second_path = None
for w in [16, 32, 128, 256, 512]: for w in [16, 32, 128, 256, 512]:
prefix = "icon_{}x{}".format(w, w) prefix = f"icon_{w}x{w}"
first_path = os.path.join(iconset, prefix + ".png") first_path = os.path.join(iconset, prefix + ".png")
if last_w == w: if last_w == w:

View File

@ -86,16 +86,16 @@ OPEN = {
# ifunc95 extensions # ifunc95 extensions
for i in ["8", "8S", "16", "16S", "32", "32F"]: for i in ["8", "8S", "16", "16S", "32", "32F"]:
OPEN["L %s image" % i] = ("F", "F;%s" % i) OPEN[f"L {i} image"] = ("F", f"F;{i}")
OPEN["L*%s image" % i] = ("F", "F;%s" % i) OPEN[f"L*{i} image"] = ("F", f"F;{i}")
for i in ["16", "16L", "16B"]: for i in ["16", "16L", "16B"]:
OPEN["L %s image" % i] = ("I;%s" % i, "I;%s" % i) OPEN[f"L {i} image"] = (f"I;{i}", f"I;{i}")
OPEN["L*%s image" % i] = ("I;%s" % i, "I;%s" % i) OPEN[f"L*{i} image"] = (f"I;{i}", f"I;{i}")
for i in ["32S"]: for i in ["32S"]:
OPEN["L %s image" % i] = ("I", "I;%s" % i) OPEN[f"L {i} image"] = ("I", f"I;{i}")
OPEN["L*%s image" % i] = ("I", "I;%s" % i) OPEN[f"L*{i} image"] = ("I", f"I;{i}")
for i in range(2, 33): for i in range(2, 33):
OPEN["L*%s image" % i] = ("F", "F;%s" % i) OPEN[f"L*{i} image"] = ("F", f"F;{i}")
# -------------------------------------------------------------------- # --------------------------------------------------------------------
@ -342,11 +342,11 @@ def _save(im, fp, filename):
try: try:
image_type, rawmode = SAVE[im.mode] image_type, rawmode = SAVE[im.mode]
except KeyError as e: except KeyError as e:
raise ValueError("Cannot save %s images as IM" % im.mode) from e raise ValueError(f"Cannot save {im.mode} images as IM") from e
frames = im.encoderinfo.get("frames", 1) frames = im.encoderinfo.get("frames", 1)
fp.write(("Image type: %s image\r\n" % image_type).encode("ascii")) fp.write(f"Image type: {image_type} image\r\n".encode("ascii"))
if filename: if filename:
# Each line must be 100 characters or less, # Each line must be 100 characters or less,
# or: SyntaxError("not an IM file") # or: SyntaxError("not an IM file")
@ -355,9 +355,9 @@ def _save(im, fp, filename):
name, ext = os.path.splitext(os.path.basename(filename)) name, ext = os.path.splitext(os.path.basename(filename))
name = "".join([name[: 92 - len(ext)], ext]) name = "".join([name[: 92 - len(ext)], ext])
fp.write(("Name: %s\r\n" % name).encode("ascii")) fp.write(f"Name: {name}\r\n".encode("ascii"))
fp.write(("Image size (x*y): %d*%d\r\n" % im.size).encode("ascii")) fp.write(("Image size (x*y): %d*%d\r\n" % im.size).encode("ascii"))
fp.write(("File size (no of images): %d\r\n" % frames).encode("ascii")) fp.write(f"File size (no of images): {frames}\r\n".encode("ascii"))
if im.mode in ["P", "PA"]: if im.mode in ["P", "PA"]:
fp.write(b"Lut: 1\r\n") fp.write(b"Lut: 1\r\n")
fp.write(b"\000" * (511 - fp.tell()) + b"\032") fp.write(b"\000" * (511 - fp.tell()) + b"\032")

View File

@ -59,7 +59,7 @@ if sys.version_info >= (3, 7):
if name == "PILLOW_VERSION": if name == "PILLOW_VERSION":
_raise_version_warning() _raise_version_warning()
return __version__ return __version__
raise AttributeError("module '{}' has no attribute '{}'".format(__name__, name)) raise AttributeError(f"module '{__name__}' has no attribute '{name}'")
else: else:
@ -96,8 +96,8 @@ try:
if __version__ != getattr(core, "PILLOW_VERSION", None): if __version__ != getattr(core, "PILLOW_VERSION", None):
raise ImportError( raise ImportError(
"The _imaging extension was built for another version of Pillow or PIL:\n" "The _imaging extension was built for another version of Pillow or PIL:\n"
"Core version: %s\n" f"Core version: {getattr(core, 'PILLOW_VERSION', None)}\n"
"Pillow version: %s" % (getattr(core, "PILLOW_VERSION", None), __version__) f"Pillow version: {__version__}"
) )
except ImportError as v: except ImportError as v:
@ -403,7 +403,7 @@ def init():
for plugin in _plugins: for plugin in _plugins:
try: try:
logger.debug("Importing %s", plugin) logger.debug("Importing %s", plugin)
__import__("PIL.%s" % plugin, globals(), locals(), []) __import__(f"PIL.{plugin}", globals(), locals(), [])
except ImportError as e: except ImportError as e:
logger.debug("Image: failed to import %s: %s", plugin, e) logger.debug("Image: failed to import %s: %s", plugin, e)
@ -435,7 +435,7 @@ def _getdecoder(mode, decoder_name, args, extra=()):
# get decoder # get decoder
decoder = getattr(core, decoder_name + "_decoder") decoder = getattr(core, decoder_name + "_decoder")
except AttributeError as e: except AttributeError as e:
raise OSError("decoder %s not available" % decoder_name) from e raise OSError(f"decoder {decoder_name} not available") from e
return decoder(mode, *args + extra) return decoder(mode, *args + extra)
@ -458,7 +458,7 @@ def _getencoder(mode, encoder_name, args, extra=()):
# get encoder # get encoder
encoder = getattr(core, encoder_name + "_encoder") encoder = getattr(core, encoder_name + "_encoder")
except AttributeError as e: except AttributeError as e:
raise OSError("encoder %s not available" % encoder_name) from e raise OSError(f"encoder {encoder_name} not available") from e
return encoder(mode, *args + extra) return encoder(mode, *args + extra)
@ -743,7 +743,7 @@ class Image:
if s: if s:
break break
if s < 0: if s < 0:
raise RuntimeError("encoder error %d in tobytes" % s) raise RuntimeError(f"encoder error {s} in tobytes")
return b"".join(data) return b"".join(data)
@ -764,9 +764,9 @@ class Image:
data = self.tobytes("xbm") data = self.tobytes("xbm")
return b"".join( return b"".join(
[ [
("#define %s_width %d\n" % (name, self.size[0])).encode("ascii"), f"#define {name}_width {self.size[0]}\n".encode("ascii"),
("#define %s_height %d\n" % (name, self.size[1])).encode("ascii"), f"#define {name}_height {self.size[1]}\n".encode("ascii"),
("static char %s_bits[] = {\n" % name).encode("ascii"), f"static char {name}_bits[] = {{\n".encode("ascii"),
data, data,
b"};", b"};",
] ]
@ -1862,7 +1862,7 @@ class Image:
""" """
if resample not in (NEAREST, BILINEAR, BICUBIC, LANCZOS, BOX, HAMMING): if resample not in (NEAREST, BILINEAR, BICUBIC, LANCZOS, BOX, HAMMING):
message = "Unknown resampling filter ({}).".format(resample) message = f"Unknown resampling filter ({resample})."
filters = [ filters = [
"{} ({})".format(filter[1], filter[0]) "{} ({})".format(filter[1], filter[0])
@ -2130,7 +2130,7 @@ class Image:
try: try:
format = EXTENSION[ext] format = EXTENSION[ext]
except KeyError as e: except KeyError as e:
raise ValueError("unknown file extension: {}".format(ext)) from e raise ValueError(f"unknown file extension: {ext}") from e
if format.upper() not in SAVE: if format.upper() not in SAVE:
init() init()
@ -2242,7 +2242,7 @@ class Image:
try: try:
channel = self.getbands().index(channel) channel = self.getbands().index(channel)
except ValueError as e: except ValueError as e:
raise ValueError('The image has no channel "{}"'.format(channel)) from e raise ValueError(f'The image has no channel "{channel}"') from e
return self._new(self.im.getband(channel)) return self._new(self.im.getband(channel))
@ -2463,9 +2463,9 @@ class Image:
BOX: "Image.BOX", BOX: "Image.BOX",
HAMMING: "Image.HAMMING", HAMMING: "Image.HAMMING",
LANCZOS: "Image.LANCZOS/Image.ANTIALIAS", LANCZOS: "Image.LANCZOS/Image.ANTIALIAS",
}[resample] + " ({}) cannot be used.".format(resample) }[resample] + f" ({resample}) cannot be used."
else: else:
message = "Unknown resampling filter ({}).".format(resample) message = f"Unknown resampling filter ({resample})."
filters = [ filters = [
"{} ({})".format(filter[1], filter[0]) "{} ({})".format(filter[1], filter[0])
@ -2759,7 +2759,7 @@ def fromarray(obj, mode=None):
else: else:
ndmax = 4 ndmax = 4
if ndim > ndmax: if ndim > ndmax:
raise ValueError("Too many dimensions: %d > %d." % (ndim, ndmax)) raise ValueError(f"Too many dimensions: {ndim} > {ndmax}.")
size = 1 if ndim == 1 else shape[1], shape[0] size = 1 if ndim == 1 else shape[1], shape[0]
if strides is not None: if strides is not None:
@ -2825,14 +2825,14 @@ def _decompression_bomb_check(size):
if pixels > 2 * MAX_IMAGE_PIXELS: if pixels > 2 * MAX_IMAGE_PIXELS:
raise DecompressionBombError( raise DecompressionBombError(
"Image size (%d pixels) exceeds limit of %d pixels, " f"Image size ({pixels} pixels) exceeds limit of {2 * MAX_IMAGE_PIXELS} "
"could be decompression bomb DOS attack." % (pixels, 2 * MAX_IMAGE_PIXELS) "pixels, could be decompression bomb DOS attack."
) )
if pixels > MAX_IMAGE_PIXELS: if pixels > MAX_IMAGE_PIXELS:
warnings.warn( warnings.warn(
"Image size (%d pixels) exceeds limit of %d pixels, " f"Image size ({pixels} pixels) exceeds limit of {MAX_IMAGE_PIXELS} pixels, "
"could be decompression bomb DOS attack." % (pixels, MAX_IMAGE_PIXELS), "could be decompression bomb DOS attack.",
DecompressionBombWarning, DecompressionBombWarning,
) )
@ -2861,7 +2861,7 @@ def open(fp, mode="r"):
""" """
if mode != "r": if mode != "r":
raise ValueError("bad mode %r" % mode) raise ValueError(f"bad mode {repr(mode)}")
elif isinstance(fp, io.StringIO): elif isinstance(fp, io.StringIO):
raise ValueError( raise ValueError(
"StringIO cannot be used to open an image. " "StringIO cannot be used to open an image. "
@ -3242,13 +3242,13 @@ def _apply_env_variables(env=None):
try: try:
var = int(var) * units var = int(var) * units
except ValueError: except ValueError:
warnings.warn("{} is not int".format(var_name)) warnings.warn(f"{var_name} is not int")
continue continue
try: try:
setter(var) setter(var)
except ValueError as e: except ValueError as e:
warnings.warn("{}: {}".format(var_name, e)) warnings.warn(f"{var_name}: {e}")
_apply_env_variables() _apply_env_variables()
@ -3371,8 +3371,8 @@ class Exif(MutableMapping):
if len(data) != size: if len(data) != size:
warnings.warn( warnings.warn(
"Possibly corrupt EXIF MakerNote data. " "Possibly corrupt EXIF MakerNote data. "
"Expecting to read %d bytes but only got %d." f"Expecting to read {size} bytes but only got "
" Skipping tag %s" % (size, len(data), ifd_tag) f"{len(data)}. Skipping tag {ifd_tag}"
) )
continue continue

View File

@ -680,8 +680,7 @@ def createProfile(colorSpace, colorTemp=-1):
if colorSpace not in ["LAB", "XYZ", "sRGB"]: if colorSpace not in ["LAB", "XYZ", "sRGB"]:
raise PyCMSError( raise PyCMSError(
"Color space not supported for on-the-fly profile creation (%s)" f"Color space not supported for on-the-fly profile creation ({colorSpace})"
% colorSpace
) )
if colorSpace == "LAB": if colorSpace == "LAB":
@ -689,7 +688,7 @@ def createProfile(colorSpace, colorTemp=-1):
colorTemp = float(colorTemp) colorTemp = float(colorTemp)
except (TypeError, ValueError) as e: except (TypeError, ValueError) as e:
raise PyCMSError( raise PyCMSError(
'Color temperature must be numeric, "%s" not valid' % colorTemp f'Color temperature must be numeric, "{colorTemp}" not valid'
) from e ) from e
try: try:
@ -734,7 +733,7 @@ def getProfileName(profile):
return (profile.profile.profile_description or "") + "\n" return (profile.profile.profile_description or "") + "\n"
if not manufacturer or len(model) > 30: if not manufacturer or len(model) > 30:
return model + "\n" return model + "\n"
return "{} - {}\n".format(model, manufacturer) return f"{model} - {manufacturer}\n"
except (AttributeError, OSError, TypeError, ValueError) as v: except (AttributeError, OSError, TypeError, ValueError) as v:
raise PyCMSError(v) from v raise PyCMSError(v) from v

View File

@ -113,7 +113,7 @@ def getrgb(color):
m = re.match(r"rgba\(\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)\s*\)$", color) m = re.match(r"rgba\(\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)\s*\)$", color)
if m: if m:
return (int(m.group(1)), int(m.group(2)), int(m.group(3)), int(m.group(4))) return (int(m.group(1)), int(m.group(2)), int(m.group(3)), int(m.group(4)))
raise ValueError("unknown color specifier: %r" % color) raise ValueError(f"unknown color specifier: {repr(color)}")
def getcolor(color, mode): def getcolor(color, mode):

View File

@ -276,7 +276,7 @@ class ImageDraw:
stroke_width=0, stroke_width=0,
stroke_fill=None, stroke_fill=None,
*args, *args,
**kwargs **kwargs,
): ):
if self._multiline_check(text): if self._multiline_check(text):
return self.multiline_text( return self.multiline_text(

View File

@ -63,7 +63,7 @@ def raise_oserror(error):
except AttributeError: except AttributeError:
message = ERRORS.get(error) message = ERRORS.get(error)
if not message: if not message:
message = "decoder error %d" % error message = f"decoder error {error}"
raise OSError(message + " when reading image file") raise OSError(message + " when reading image file")
@ -203,7 +203,7 @@ class ImageFile(Image.Image):
# use mmap, if possible # use mmap, if possible
import mmap import mmap
with open(self.filename, "r") as fp: with open(self.filename) as fp:
self.map = mmap.mmap( self.map = mmap.mmap(
fp.fileno(), 0, access=mmap.ACCESS_READ fp.fileno(), 0, access=mmap.ACCESS_READ
) )
@ -258,7 +258,7 @@ class ImageFile(Image.Image):
else: else:
raise OSError( raise OSError(
"image file is truncated " "image file is truncated "
"(%d bytes not processed)" % len(b) f"({len(b)} bytes not processed)"
) )
b = b + s b = b + s
@ -334,7 +334,7 @@ class StubImageFile(ImageFile):
def load(self): def load(self):
loader = self._load() loader = self._load()
if loader is None: if loader is None:
raise OSError("cannot find loader for this %s file" % self.format) raise OSError(f"cannot find loader for this {self.format} file")
image = loader.load(self) image = loader.load(self)
assert image is not None assert image is not None
# become the other object (!) # become the other object (!)
@ -526,7 +526,7 @@ def _save(im, fp, tile, bufsize=0):
if s: if s:
break break
if s < 0: if s < 0:
raise OSError("encoder error %d when writing image file" % s) from exc raise OSError(f"encoder error {s} when writing image file") from exc
e.cleanup() e.cleanup()
else: else:
# slight speedup: compress to real file object # slight speedup: compress to real file object
@ -541,7 +541,7 @@ def _save(im, fp, tile, bufsize=0):
else: else:
s = e.encode_to_file(fh, bufsize) s = e.encode_to_file(fh, bufsize)
if s < 0: if s < 0:
raise OSError("encoder error %d when writing image file" % s) raise OSError(f"encoder error {s} when writing image file")
e.cleanup() e.cleanup()
if hasattr(fp, "flush"): if hasattr(fp, "flush"):
fp.flush() fp.flush()

View File

@ -401,9 +401,8 @@ class Color3DLUT(MultibandFilter):
raise ValueError( raise ValueError(
"The table should have either channels * size**3 float items " "The table should have either channels * size**3 float items "
"or size**3 items of channels-sized tuples with floats. " "or size**3 items of channels-sized tuples with floats. "
"Table should be: {}x{}x{}x{}. Actual length: {}".format( f"Table should be: {channels}x{size[0]}x{size[1]}x{size[2]}. "
channels, size[0], size[1], size[2], len(table) f"Actual length: {len(table)}"
)
) )
self.table = table self.table = table
@ -513,12 +512,12 @@ class Color3DLUT(MultibandFilter):
def __repr__(self): def __repr__(self):
r = [ r = [
"{} from {}".format(self.__class__.__name__, self.table.__class__.__name__), f"{self.__class__.__name__} from {self.table.__class__.__name__}",
"size={:d}x{:d}x{:d}".format(*self.size), "size={:d}x{:d}x{:d}".format(*self.size),
"channels={:d}".format(self.channels), f"channels={self.channels:d}",
] ]
if self.mode: if self.mode:
r.append("target_mode={}".format(self.mode)) r.append(f"target_mode={self.mode}")
return "<{}>".format(" ".join(r)) return "<{}>".format(" ".join(r))
def filter(self, image): def filter(self, image):

View File

@ -415,7 +415,7 @@ class FreeTypeFont:
language=None, language=None,
stroke_width=0, stroke_width=0,
*args, *args,
**kwargs **kwargs,
): ):
""" """
Create a bitmap for the text. Create a bitmap for the text.

View File

@ -41,7 +41,7 @@ class _Operand:
elif im1.im.mode in ("I", "F"): elif im1.im.mode in ("I", "F"):
return im1.im return im1.im
else: else:
raise ValueError("unsupported mode: %s" % im1.im.mode) raise ValueError(f"unsupported mode: {im1.im.mode}")
else: else:
# argument was a constant # argument was a constant
if _isconstant(im1) and self.im.mode in ("1", "L", "I"): if _isconstant(im1) and self.im.mode in ("1", "L", "I"):
@ -58,7 +58,7 @@ class _Operand:
try: try:
op = getattr(_imagingmath, op + "_" + im1.mode) op = getattr(_imagingmath, op + "_" + im1.mode)
except AttributeError as e: except AttributeError as e:
raise TypeError("bad operand type for '%s'" % op) from e raise TypeError(f"bad operand type for '{op}'") from e
_imagingmath.unop(op, out.im.id, im1.im.id) _imagingmath.unop(op, out.im.id, im1.im.id)
else: else:
# binary operation # binary operation
@ -86,7 +86,7 @@ class _Operand:
try: try:
op = getattr(_imagingmath, op + "_" + im1.mode) op = getattr(_imagingmath, op + "_" + im1.mode)
except AttributeError as e: except AttributeError as e:
raise TypeError("bad operand type for '%s'" % op) from e raise TypeError(f"bad operand type for '{op}'") from e
_imagingmath.binop(op, out.im.id, im1.im.id, im2.im.id) _imagingmath.binop(op, out.im.id, im1.im.id, im2.im.id)
return _Operand(out) return _Operand(out)

View File

@ -111,7 +111,7 @@ class ImagePalette:
self.dirty = 1 self.dirty = 1
return index return index
else: else:
raise ValueError("unknown color specifier: %r" % color) raise ValueError(f"unknown color specifier: {repr(color)}")
def save(self, fp): def save(self, fp):
"""Save palette to text file. """Save palette to text file.
@ -123,12 +123,12 @@ class ImagePalette:
if isinstance(fp, str): if isinstance(fp, str):
fp = open(fp, "w") fp = open(fp, "w")
fp.write("# Palette\n") fp.write("# Palette\n")
fp.write("# Mode: %s\n" % self.mode) fp.write(f"# Mode: {self.mode}\n")
for i in range(256): for i in range(256):
fp.write("%d" % i) fp.write(f"{i}")
for j in range(i * len(self.mode), (i + 1) * len(self.mode)): for j in range(i * len(self.mode), (i + 1) * len(self.mode)):
try: try:
fp.write(" %d" % self.palette[j]) fp.write(f" {self.palette[j]}")
except IndexError: except IndexError:
fp.write(" 0") fp.write(" 0")
fp.write("\n") fp.write("\n")

View File

@ -145,7 +145,7 @@ def _toqclass_helper(im):
data = im.tobytes("raw", "BGRA") data = im.tobytes("raw", "BGRA")
format = QImage.Format_ARGB32 format = QImage.Format_ARGB32
else: else:
raise ValueError("unsupported image mode %r" % im.mode) raise ValueError(f"unsupported image mode {repr(im.mode)}")
__data = data or align8to32(im.tobytes(), im.size[0], im.mode) __data = data or align8to32(im.tobytes(), im.size[0], im.mode)
return {"data": __data, "im": im, "format": format, "colortable": colortable} return {"data": __data, "im": im, "format": format, "colortable": colortable}

View File

@ -123,9 +123,9 @@ class WindowsViewer(Viewer):
def get_command(self, file, **options): def get_command(self, file, **options):
return ( return (
'start "Pillow" /WAIT "%s" ' f'start "Pillow" /WAIT "{file}" '
"&& ping -n 2 127.0.0.1 >NUL " "&& ping -n 2 127.0.0.1 >NUL "
'&& del /f "%s"' % (file, file) f'&& del /f "{file}"'
) )
@ -143,9 +143,7 @@ class MacViewer(Viewer):
# on darwin open returns immediately resulting in the temp # on darwin open returns immediately resulting in the temp
# file removal while app is opening # file removal while app is opening
command = "open -a Preview.app" command = "open -a Preview.app"
command = "({} {}; sleep 20; rm -f {})&".format( command = f"({command} {quote(file)}; sleep 20; rm -f {quote(file)})&"
command, quote(file), quote(file)
)
return command return command
def show_file(self, file, **options): def show_file(self, file, **options):
@ -153,7 +151,7 @@ class MacViewer(Viewer):
fd, path = tempfile.mkstemp() fd, path = tempfile.mkstemp()
with os.fdopen(fd, "w") as f: with os.fdopen(fd, "w") as f:
f.write(file) f.write(file)
with open(path, "r") as f: with open(path) as f:
subprocess.Popen( subprocess.Popen(
["im=$(cat); open -a Preview.app $im; sleep 20; rm -f $im"], ["im=$(cat); open -a Preview.app $im; sleep 20; rm -f $im"],
shell=True, shell=True,
@ -173,14 +171,14 @@ class UnixViewer(Viewer):
def get_command(self, file, **options): def get_command(self, file, **options):
command = self.get_command_ex(file, **options)[0] command = self.get_command_ex(file, **options)[0]
return "({} {}; rm -f {})&".format(command, quote(file), quote(file)) return f"({command} {quote(file)}; rm -f {quote(file)})&"
def show_file(self, file, **options): def show_file(self, file, **options):
"""Display given file""" """Display given file"""
fd, path = tempfile.mkstemp() fd, path = tempfile.mkstemp()
with os.fdopen(fd, "w") as f: with os.fdopen(fd, "w") as f:
f.write(file) f.write(file)
with open(path, "r") as f: with open(path) as f:
command = self.get_command_ex(file, **options)[0] command = self.get_command_ex(file, **options)[0]
subprocess.Popen( subprocess.Popen(
["im=$(cat);" + command + " $im; rm -f $im"], shell=True, stdin=f ["im=$(cat);" + command + " $im; rm -f $im"], shell=True, stdin=f
@ -216,7 +214,7 @@ class XVViewer(UnixViewer):
# imagemagick's display command instead. # imagemagick's display command instead.
command = executable = "xv" command = executable = "xv"
if title: if title:
command += " -name %s" % quote(title) command += f" -name {quote(title)}"
return command, executable return command, executable

View File

@ -41,7 +41,7 @@ def _pilbitmap_check():
if _pilbitmap_ok is None: if _pilbitmap_ok is None:
try: try:
im = Image.new("1", (1, 1)) im = Image.new("1", (1, 1))
tkinter.BitmapImage(data="PIL:%d" % im.im.id) tkinter.BitmapImage(data=f"PIL:{im.im.id}")
_pilbitmap_ok = 1 _pilbitmap_ok = 1
except tkinter.TclError: except tkinter.TclError:
_pilbitmap_ok = 0 _pilbitmap_ok = 0
@ -229,7 +229,7 @@ class BitmapImage:
if _pilbitmap_check(): if _pilbitmap_check():
# fast way (requires the pilbitmap booster patch) # fast way (requires the pilbitmap booster patch)
image.load() image.load()
kw["data"] = "PIL:%d" % image.im.id kw["data"] = f"PIL:{image.im.id}"
self.__im = image # must keep a reference self.__im = image # must keep a reference
else: else:
# slow but safe way # slow but safe way

View File

@ -198,7 +198,7 @@ def SOF(self, marker):
self.bits = i8(s[0]) self.bits = i8(s[0])
if self.bits != 8: if self.bits != 8:
raise SyntaxError("cannot handle %d-bit layers" % self.bits) raise SyntaxError(f"cannot handle {self.bits}-bit layers")
self.layers = i8(s[5]) self.layers = i8(s[5])
if self.layers == 1: if self.layers == 1:
@ -208,7 +208,7 @@ def SOF(self, marker):
elif self.layers == 4: elif self.layers == 4:
self.mode = "CMYK" self.mode = "CMYK"
else: else:
raise SyntaxError("cannot handle %d-layer images" % self.layers) raise SyntaxError(f"cannot handle {self.layers}-layer images")
if marker in [0xFFC2, 0xFFC6, 0xFFCA, 0xFFCE]: if marker in [0xFFC2, 0xFFC6, 0xFFCA, 0xFFCE]:
self.info["progressive"] = self.info["progression"] = 1 self.info["progressive"] = self.info["progression"] = 1
@ -521,7 +521,7 @@ def _getmp(self):
rawmpentries = mp[0xB002] rawmpentries = mp[0xB002]
for entrynum in range(0, quant): for entrynum in range(0, quant):
unpackedentry = struct.unpack_from( unpackedentry = struct.unpack_from(
"{}LLLHH".format(endianness), rawmpentries, entrynum * 16 f"{endianness}LLLHH", rawmpentries, entrynum * 16
) )
labels = ("Attribute", "Size", "DataOffset", "EntryNo1", "EntryNo2") labels = ("Attribute", "Size", "DataOffset", "EntryNo1", "EntryNo2")
mpentry = dict(zip(labels, unpackedentry)) mpentry = dict(zip(labels, unpackedentry))
@ -616,7 +616,7 @@ def _save(im, fp, filename):
try: try:
rawmode = RAWMODE[im.mode] rawmode = RAWMODE[im.mode]
except KeyError as e: except KeyError as e:
raise OSError("cannot write mode %s as JPEG" % im.mode) from e raise OSError(f"cannot write mode {im.mode} as JPEG") from e
info = im.encoderinfo info = im.encoderinfo

View File

@ -116,7 +116,7 @@ class MspDecoder(ImageFile.PyDecoder):
try: try:
self.fd.seek(32) self.fd.seek(32)
rowmap = struct.unpack_from( rowmap = struct.unpack_from(
"<%dH" % (self.state.ysize), self.fd.read(self.state.ysize * 2) f"<{self.state.ysize}H", self.fd.read(self.state.ysize * 2)
) )
except struct.error as e: except struct.error as e:
raise OSError("Truncated MSP file in row map") from e raise OSError("Truncated MSP file in row map") from e
@ -145,7 +145,7 @@ class MspDecoder(ImageFile.PyDecoder):
idx += runcount idx += runcount
except struct.error as e: except struct.error as e:
raise OSError("Corrupted MSP file in row %d" % x) from e raise OSError(f"Corrupted MSP file in row {x}") from e
self.set_as_raw(img.getvalue(), ("1", 0, 1)) self.set_as_raw(img.getvalue(), ("1", 0, 1))
@ -162,7 +162,7 @@ Image.register_decoder("MSP", MspDecoder)
def _save(im, fp, filename): def _save(im, fp, filename):
if im.mode != "1": if im.mode != "1":
raise OSError("cannot write mode %s as MSP" % im.mode) raise OSError(f"cannot write mode {im.mode} as MSP")
# create MSP header # create MSP header
header = [0] * 16 header = [0] * 16

View File

@ -71,10 +71,10 @@ class PSDraw:
""" """
if font not in self.isofont: if font not in self.isofont:
# reencode font # reencode font
self._fp_write("/PSDraw-{} ISOLatin1Encoding /{} E\n".format(font, font)) self._fp_write(f"/PSDraw-{font} ISOLatin1Encoding /{font} E\n")
self.isofont[font] = 1 self.isofont[font] = 1
# rough # rough
self._fp_write("/F0 %d /PSDraw-%s F\n" % (size, font)) self._fp_write(f"/F0 {size} /PSDraw-{font} F\n")
def line(self, xy0, xy1): def line(self, xy0, xy1):
""" """
@ -82,8 +82,7 @@ class PSDraw:
PostScript point coordinates (72 points per inch, (0, 0) is the lower PostScript point coordinates (72 points per inch, (0, 0) is the lower
left corner of the page). left corner of the page).
""" """
xy = xy0 + xy1 self._fp_write("%d %d %d %d Vl\n" % (*xy0, *xy1))
self._fp_write("%d %d %d %d Vl\n" % xy)
def rectangle(self, box): def rectangle(self, box):
""" """
@ -107,8 +106,7 @@ class PSDraw:
""" """
text = "\\(".join(text.split("(")) text = "\\(".join(text.split("("))
text = "\\)".join(text.split(")")) text = "\\)".join(text.split(")"))
xy = xy + (text,) self._fp_write(f"{xy[0]} {xy[1]} M ({text}) S\n")
self._fp_write("%d %d M (%s) S\n" % xy)
def image(self, box, im, dpi=None): def image(self, box, im, dpi=None):
"""Draw a PIL image, centered in the given box.""" """Draw a PIL image, centered in the given box."""
@ -132,12 +130,12 @@ class PSDraw:
y = ymax y = ymax
dx = (xmax - x) / 2 + box[0] dx = (xmax - x) / 2 + box[0]
dy = (ymax - y) / 2 + box[1] dy = (ymax - y) / 2 + box[1]
self._fp_write("gsave\n{:f} {:f} translate\n".format(dx, dy)) self._fp_write(f"gsave\n{dx:f} {dy:f} translate\n")
if (x, y) != im.size: if (x, y) != im.size:
# EpsImagePlugin._save prints the image at (0,0,xsize,ysize) # EpsImagePlugin._save prints the image at (0,0,xsize,ysize)
sx = x / im.size[0] sx = x / im.size[0]
sy = y / im.size[1] sy = y / im.size[1]
self._fp_write("{:f} {:f} scale\n".format(sx, sy)) self._fp_write(f"{sx:f} {sy:f} scale\n")
EpsImagePlugin._save(im, self.fp, None, 0) EpsImagePlugin._save(im, self.fp, None, 0)
self._fp_write("\ngrestore\n") self._fp_write("\ngrestore\n")

View File

@ -138,7 +138,7 @@ def _save(im, fp, filename):
bpp = im.info["bpp"] bpp = im.info["bpp"]
im = im.point(lambda x, maxval=(1 << bpp) - 1: maxval - (x & maxval)) im = im.point(lambda x, maxval=(1 << bpp) - 1: maxval - (x & maxval))
else: else:
raise OSError("cannot write mode %s as Palm" % im.mode) raise OSError(f"cannot write mode {im.mode} as Palm")
# we ignore the palette here # we ignore the palette here
im.mode = "P" im.mode = "P"
@ -154,7 +154,7 @@ def _save(im, fp, filename):
else: else:
raise OSError("cannot write mode %s as Palm" % im.mode) raise OSError(f"cannot write mode {im.mode} as Palm")
# #
# make sure image data is available # make sure image data is available

View File

@ -135,7 +135,7 @@ def _save(im, fp, filename):
try: try:
version, bits, planes, rawmode = SAVE[im.mode] version, bits, planes, rawmode = SAVE[im.mode]
except KeyError as e: except KeyError as e:
raise ValueError("Cannot save %s images as PCX" % im.mode) from e raise ValueError(f"Cannot save {im.mode} images as PCX") from e
# bytes per plane # bytes per plane
stride = (im.size[0] * bits + 7) // 8 stride = (im.size[0] * bits + 7) // 8

View File

@ -77,7 +77,7 @@ def _save(im, fp, filename, save_all=False):
existing_pdf.start_writing() existing_pdf.start_writing()
existing_pdf.write_header() existing_pdf.write_header()
existing_pdf.write_comment("created by Pillow {} PDF driver".format(__version__)) existing_pdf.write_comment(f"created by Pillow {__version__} PDF driver")
# #
# pages # pages
@ -130,7 +130,7 @@ def _save(im, fp, filename, save_all=False):
bits = 1 bits = 1
elif im.mode == "L": elif im.mode == "L":
filter = "DCTDecode" filter = "DCTDecode"
# params = "<< /Predictor 15 /Columns %d >>" % (width-2) # params = f"<< /Predictor 15 /Columns {width-2} >>"
colorspace = PdfParser.PdfName("DeviceGray") colorspace = PdfParser.PdfName("DeviceGray")
procset = "ImageB" # grayscale procset = "ImageB" # grayscale
elif im.mode == "P": elif im.mode == "P":
@ -153,7 +153,7 @@ def _save(im, fp, filename, save_all=False):
procset = "ImageC" # color images procset = "ImageC" # color images
decode = [1, 0, 1, 0, 1, 0, 1, 0] decode = [1, 0, 1, 0, 1, 0, 1, 0]
else: else:
raise ValueError("cannot save mode %s" % im.mode) raise ValueError(f"cannot save mode {im.mode}")
# #
# image # image
@ -175,7 +175,7 @@ def _save(im, fp, filename, save_all=False):
elif filter == "RunLengthDecode": elif filter == "RunLengthDecode":
ImageFile._save(im, op, [("packbits", (0, 0) + im.size, 0, im.mode)]) ImageFile._save(im, op, [("packbits", (0, 0) + im.size, 0, im.mode)])
else: else:
raise ValueError("unsupported PDF filter (%s)" % filter) raise ValueError(f"unsupported PDF filter ({filter})")
# #
# Get image characteristics # Get image characteristics

View File

@ -183,8 +183,8 @@ class XrefTable:
this_deleted_object_id = deleted_keys.pop(0) this_deleted_object_id = deleted_keys.pop(0)
check_format_condition( check_format_condition(
object_id == this_deleted_object_id, object_id == this_deleted_object_id,
"expected the next deleted object ID to be %s, instead found %s" f"expected the next deleted object ID to be {object_id}, "
% (object_id, this_deleted_object_id), f"instead found {this_deleted_object_id}",
) )
try: try:
next_in_linked_list = deleted_keys[0] next_in_linked_list = deleted_keys[0]
@ -218,7 +218,7 @@ class PdfName:
return hash(self.name) return hash(self.name)
def __repr__(self): def __repr__(self):
return "PdfName(%s)" % repr(self.name) return f"PdfName({repr(self.name)})"
@classmethod @classmethod
def from_pdf_stream(cls, data): def from_pdf_stream(cls, data):
@ -315,7 +315,7 @@ class PdfStream:
return zlib.decompress(self.buf, bufsize=int(expected_length)) return zlib.decompress(self.buf, bufsize=int(expected_length))
else: else:
raise NotImplementedError( raise NotImplementedError(
"stream filter %s unknown/unsupported" % repr(self.dictionary.Filter) f"stream filter {repr(self.dictionary.Filter)} unknown/unsupported"
) )
@ -423,7 +423,7 @@ class PdfParser:
self.f.write(b"%PDF-1.4\n") self.f.write(b"%PDF-1.4\n")
def write_comment(self, s): def write_comment(self, s):
self.f.write(("% {}\n".format(s)).encode("utf-8")) self.f.write(f"% {s}\n".encode("utf-8"))
def write_catalog(self): def write_catalog(self):
self.del_root() self.del_root()
@ -966,9 +966,8 @@ class PdfParser:
offset, generation = self.xref_table[ref[0]] offset, generation = self.xref_table[ref[0]]
check_format_condition( check_format_condition(
generation == ref[1], generation == ref[1],
"expected to find generation %s for object ID %s in xref table, " f"expected to find generation {ref[1]} for object ID {ref[0]} in xref "
"instead found generation %s at offset %s" f"table, instead found generation {generation} at offset {offset}",
% (ref[1], ref[0], generation, offset),
) )
value = self.get_value( value = self.get_value(
self.buf, self.buf,

View File

@ -164,7 +164,7 @@ class ChunkStream:
if not is_cid(cid): if not is_cid(cid):
if not ImageFile.LOAD_TRUNCATED_IMAGES: if not ImageFile.LOAD_TRUNCATED_IMAGES:
raise SyntaxError("broken PNG file (chunk %s)" % repr(cid)) raise SyntaxError(f"broken PNG file (chunk {repr(cid)})")
return cid, pos, length return cid, pos, length
@ -201,10 +201,12 @@ class ChunkStream:
crc1 = _crc32(data, _crc32(cid)) crc1 = _crc32(data, _crc32(cid))
crc2 = i32(self.fp.read(4)) crc2 = i32(self.fp.read(4))
if crc1 != crc2: if crc1 != crc2:
raise SyntaxError("broken PNG file (bad header checksum in %r)" % cid) raise SyntaxError(
f"broken PNG file (bad header checksum in {repr(cid)})"
)
except struct.error as e: except struct.error as e:
raise SyntaxError( raise SyntaxError(
"broken PNG file (incomplete checksum in %r)" % cid f"broken PNG file (incomplete checksum in {repr(cid)})"
) from e ) from e
def crc_skip(self, cid, data): def crc_skip(self, cid, data):
@ -356,8 +358,8 @@ class PngStream(ChunkStream):
self.text_memory += chunklen self.text_memory += chunklen
if self.text_memory > MAX_TEXT_MEMORY: if self.text_memory > MAX_TEXT_MEMORY:
raise ValueError( raise ValueError(
"Too much memory used in text chunks: %s>MAX_TEXT_MEMORY" "Too much memory used in text chunks: "
% self.text_memory f"{self.text_memory}>MAX_TEXT_MEMORY"
) )
def save_rewind(self): def save_rewind(self):
@ -386,9 +388,7 @@ class PngStream(ChunkStream):
logger.debug("Compression method %s", i8(s[i])) logger.debug("Compression method %s", i8(s[i]))
comp_method = i8(s[i]) comp_method = i8(s[i])
if comp_method != 0: if comp_method != 0:
raise SyntaxError( raise SyntaxError(f"Unknown compression method {comp_method} in iCCP chunk")
"Unknown compression method %s in iCCP chunk" % comp_method
)
try: try:
icc_profile = _safe_zlib_decompress(s[i + 2 :]) icc_profile = _safe_zlib_decompress(s[i + 2 :])
except ValueError: except ValueError:
@ -536,9 +536,7 @@ class PngStream(ChunkStream):
else: else:
comp_method = 0 comp_method = 0
if comp_method != 0: if comp_method != 0:
raise SyntaxError( raise SyntaxError(f"Unknown compression method {comp_method} in zTXt chunk")
"Unknown compression method %s in zTXt chunk" % comp_method
)
try: try:
v = _safe_zlib_decompress(v[1:]) v = _safe_zlib_decompress(v[1:])
except ValueError: except ValueError:
@ -800,7 +798,7 @@ class PngImageFile(ImageFile.ImageFile):
return return
else: else:
if frame != self.__frame + 1: if frame != self.__frame + 1:
raise ValueError("cannot seek to frame %d" % frame) raise ValueError(f"cannot seek to frame {frame}")
# ensure previous frame was loaded # ensure previous frame was loaded
self.load() self.load()
@ -1197,7 +1195,7 @@ def _save(im, fp, filename, chunk=putchunk, save_all=False):
else: else:
bits = 8 bits = 8
if bits != 8: if bits != 8:
mode = "%s;%d" % (mode, bits) mode = f"{mode};{bits}"
# encoder options # encoder options
im.encoderconfig = ( im.encoderconfig = (
@ -1211,7 +1209,7 @@ def _save(im, fp, filename, chunk=putchunk, save_all=False):
try: try:
rawmode, mode = _OUTMODES[mode] rawmode, mode = _OUTMODES[mode]
except KeyError as e: except KeyError as e:
raise OSError("cannot write mode %s as PNG" % mode) from e raise OSError(f"cannot write mode {mode} as PNG") from e
# #
# write minimal PNG file # write minimal PNG file

View File

@ -104,7 +104,7 @@ class PpmImageFile(ImageFile.ImageFile):
# maxgrey # maxgrey
if s > 255: if s > 255:
if not mode == "L": if not mode == "L":
raise ValueError("Too many colors for band: %s" % s) raise ValueError(f"Too many colors for band: {s}")
if s < 2 ** 16: if s < 2 ** 16:
self.mode = "I" self.mode = "I"
rawmode = "I;16B" rawmode = "I;16B"
@ -135,7 +135,7 @@ def _save(im, fp, filename):
elif im.mode == "RGBA": elif im.mode == "RGBA":
rawmode, head = "RGB", b"P6" rawmode, head = "RGB", b"P6"
else: else:
raise OSError("cannot write mode %s as PPM" % im.mode) raise OSError(f"cannot write mode {im.mode} as PPM")
fp.write(head + ("\n%d %d\n" % im.size).encode("ascii")) fp.write(head + ("\n%d %d\n" % im.size).encode("ascii"))
if head == b"P6": if head == b"P6":
fp.write(b"255\n") fp.write(b"255\n")

View File

@ -160,9 +160,7 @@ def _save(im, fp, filename):
# assert we've got the right number of bands. # assert we've got the right number of bands.
if len(im.getbands()) != z: if len(im.getbands()) != z:
raise ValueError( raise ValueError(
"incorrect number of bands in SGI write: {} vs {}".format( f"incorrect number of bands in SGI write: {z} vs {len(im.getbands())}"
z, len(im.getbands())
)
) )
# Minimum Byte value # Minimum Byte value

View File

@ -213,7 +213,7 @@ def loadImageSeries(filelist=None):
imglist = [] imglist = []
for img in filelist: for img in filelist:
if not os.path.exists(img): if not os.path.exists(img):
print("unable to find %s" % img) print(f"unable to find {img}")
continue continue
try: try:
with Image.open(img) as im: with Image.open(img) as im:
@ -318,7 +318,7 @@ if __name__ == "__main__":
# perform some image operation # perform some image operation
im = im.transpose(Image.FLIP_LEFT_RIGHT) im = im.transpose(Image.FLIP_LEFT_RIGHT)
print( print(
"saving a flipped version of %s as %s " f"saving a flipped version of {os.path.basename(filename)} "
% (os.path.basename(filename), outfile) f"as {outfile} "
) )
im.save(outfile, SpiderImageFile.format) im.save(outfile, SpiderImageFile.format)

View File

@ -171,7 +171,7 @@ def _save(im, fp, filename):
try: try:
rawmode, bits, colormaptype, imagetype = SAVE[im.mode] rawmode, bits, colormaptype, imagetype = SAVE[im.mode]
except KeyError as e: except KeyError as e:
raise OSError("cannot write mode %s as TGA" % im.mode) from e raise OSError(f"cannot write mode {im.mode} as TGA") from e
if "rle" in im.encoderinfo: if "rle" in im.encoderinfo:
rle = im.encoderinfo["rle"] rle = im.encoderinfo["rle"]

View File

@ -466,7 +466,7 @@ class ImageFileDirectory_v2(MutableMapping):
:param prefix: Override the endianness of the file. :param prefix: Override the endianness of the file.
""" """
if ifh[:4] not in PREFIXES: if ifh[:4] not in PREFIXES:
raise SyntaxError("not a TIFF file (header %r not valid)" % ifh) raise SyntaxError(f"not a TIFF file (header {repr(ifh)} not valid)")
self._prefix = prefix if prefix is not None else ifh[:2] self._prefix = prefix if prefix is not None else ifh[:2]
if self._prefix == MM: if self._prefix == MM:
self._endian = ">" self._endian = ">"
@ -596,8 +596,8 @@ class ImageFileDirectory_v2(MutableMapping):
except ValueError: except ValueError:
# We've got a builtin tag with 1 expected entry # We've got a builtin tag with 1 expected entry
warnings.warn( warnings.warn(
"Metadata Warning, tag %s had too many entries: %s, expected 1" f"Metadata Warning, tag {tag} had too many entries: "
% (tag, len(values)) f"{len(values)}, expected 1"
) )
dest[tag] = values[0] dest[tag] = values[0]
@ -732,7 +732,7 @@ class ImageFileDirectory_v2(MutableMapping):
if len(ret) != size: if len(ret) != size:
raise OSError( raise OSError(
"Corrupt EXIF data. " "Corrupt EXIF data. "
+ "Expecting to read %d bytes but only got %d. " % (size, len(ret)) f"Expecting to read {size} bytes but only got {len(ret)}. "
) )
return ret return ret
@ -747,18 +747,18 @@ class ImageFileDirectory_v2(MutableMapping):
tagname = TiffTags.lookup(tag).name tagname = TiffTags.lookup(tag).name
typname = TYPES.get(typ, "unknown") typname = TYPES.get(typ, "unknown")
msg = "tag: %s (%d) - type: %s (%d)" % (tagname, tag, typname, typ) msg = f"tag: {tagname} ({tag}) - type: {typname} ({typ})"
try: try:
unit_size, handler = self._load_dispatch[typ] unit_size, handler = self._load_dispatch[typ]
except KeyError: except KeyError:
logger.debug(msg + " - unsupported type {}".format(typ)) logger.debug(msg + f" - unsupported type {typ}")
continue # ignore unsupported type continue # ignore unsupported type
size = count * unit_size size = count * unit_size
if size > 4: if size > 4:
here = fp.tell() here = fp.tell()
(offset,) = self._unpack("L", data) (offset,) = self._unpack("L", data)
msg += " Tag Location: {} - Data Location: {}".format(here, offset) msg += f" Tag Location: {here} - Data Location: {offset}"
fp.seek(offset) fp.seek(offset)
data = ImageFile._safe_read(fp, size) data = ImageFile._safe_read(fp, size)
fp.seek(here) fp.seek(here)
@ -768,8 +768,8 @@ class ImageFileDirectory_v2(MutableMapping):
if len(data) != size: if len(data) != size:
warnings.warn( warnings.warn(
"Possibly corrupt EXIF data. " "Possibly corrupt EXIF data. "
"Expecting to read %d bytes but only got %d." f"Expecting to read {size} bytes but only got {len(data)}."
" Skipping tag %s" % (size, len(data), tag) f" Skipping tag {tag}"
) )
logger.debug(msg) logger.debug(msg)
continue continue
@ -805,7 +805,7 @@ class ImageFileDirectory_v2(MutableMapping):
if tag == STRIPOFFSETS: if tag == STRIPOFFSETS:
stripoffsets = len(entries) stripoffsets = len(entries)
typ = self.tagtype.get(tag) typ = self.tagtype.get(tag)
logger.debug("Tag {}, Type: {}, Value: {}".format(tag, typ, value)) logger.debug(f"Tag {tag}, Type: {typ}, Value: {value}")
is_ifd = typ == TiffTags.LONG and isinstance(value, dict) is_ifd = typ == TiffTags.LONG and isinstance(value, dict)
if is_ifd: if is_ifd:
if self._endian == "<": if self._endian == "<":
@ -822,7 +822,7 @@ class ImageFileDirectory_v2(MutableMapping):
tagname = TiffTags.lookup(tag).name tagname = TiffTags.lookup(tag).name
typname = "ifd" if is_ifd else TYPES.get(typ, "unknown") typname = "ifd" if is_ifd else TYPES.get(typ, "unknown")
msg = "save: %s (%d) - type: %s (%d)" % (tagname, tag, typname, typ) msg = f"save: {tagname} ({tag}) - type: {typname} ({typ})"
msg += " - value: " + ( msg += " - value: " + (
"<table: %d bytes>" % len(data) if len(data) >= 16 else str(values) "<table: %d bytes>" % len(data) if len(data) >= 16 else str(values)
) )
@ -852,9 +852,7 @@ class ImageFileDirectory_v2(MutableMapping):
# pass 2: write entries to file # pass 2: write entries to file
for tag, typ, count, value, data in entries: for tag, typ, count, value, data in entries:
logger.debug( logger.debug(f"{tag} {typ} {count} {repr(value)} {repr(data)}")
"{} {} {} {} {}".format(tag, typ, count, repr(value), repr(data))
)
result += self._pack("HHL4s", tag, typ, count, value) result += self._pack("HHL4s", tag, typ, count, value)
# -- overwrite here for multi-page -- # -- overwrite here for multi-page --
@ -1023,8 +1021,8 @@ class TiffImageFile(ImageFile.ImageFile):
self._n_frames = None self._n_frames = None
logger.debug("*** TiffImageFile._open ***") logger.debug("*** TiffImageFile._open ***")
logger.debug("- __first: {}".format(self.__first)) logger.debug(f"- __first: {self.__first}")
logger.debug("- ifh: {!r}".format(ifh)) # Use !r to avoid str(bytes) logger.debug(f"- ifh: {repr(ifh)}") # Use repr to avoid str(bytes)
# and load the first frame # and load the first frame
self._seek(0) self._seek(0)
@ -1056,8 +1054,8 @@ class TiffImageFile(ImageFile.ImageFile):
if not self.__next: if not self.__next:
raise EOFError("no more images in TIFF file") raise EOFError("no more images in TIFF file")
logger.debug( logger.debug(
"Seeking to frame %s, on frame %s, __next %s, location: %s" f"Seeking to frame {frame}, on frame {self.__frame}, "
% (frame, self.__frame, self.__next, self.fp.tell()) f"__next {self.__next}, location: {self.fp.tell()}"
) )
# reset buffered io handle in case fp # reset buffered io handle in case fp
# was passed to libtiff, invalidating the buffer # was passed to libtiff, invalidating the buffer
@ -1214,18 +1212,18 @@ class TiffImageFile(ImageFile.ImageFile):
fillorder = self.tag_v2.get(FILLORDER, 1) fillorder = self.tag_v2.get(FILLORDER, 1)
logger.debug("*** Summary ***") logger.debug("*** Summary ***")
logger.debug("- compression: {}".format(self._compression)) logger.debug(f"- compression: {self._compression}")
logger.debug("- photometric_interpretation: {}".format(photo)) logger.debug(f"- photometric_interpretation: {photo}")
logger.debug("- planar_configuration: {}".format(self._planar_configuration)) logger.debug(f"- planar_configuration: {self._planar_configuration}")
logger.debug("- fill_order: {}".format(fillorder)) logger.debug(f"- fill_order: {fillorder}")
logger.debug("- YCbCr subsampling: {}".format(self.tag.get(530))) logger.debug(f"- YCbCr subsampling: {self.tag.get(530)}")
# size # size
xsize = int(self.tag_v2.get(IMAGEWIDTH)) xsize = int(self.tag_v2.get(IMAGEWIDTH))
ysize = int(self.tag_v2.get(IMAGELENGTH)) ysize = int(self.tag_v2.get(IMAGELENGTH))
self._size = xsize, ysize self._size = xsize, ysize
logger.debug("- size: {}".format(self.size)) logger.debug(f"- size: {self.size}")
sampleFormat = self.tag_v2.get(SAMPLEFORMAT, (1,)) sampleFormat = self.tag_v2.get(SAMPLEFORMAT, (1,))
if len(sampleFormat) > 1 and max(sampleFormat) == min(sampleFormat) == 1: if len(sampleFormat) > 1 and max(sampleFormat) == min(sampleFormat) == 1:
@ -1259,15 +1257,15 @@ class TiffImageFile(ImageFile.ImageFile):
bps_tuple, bps_tuple,
extra_tuple, extra_tuple,
) )
logger.debug("format key: {}".format(key)) logger.debug(f"format key: {key}")
try: try:
self.mode, rawmode = OPEN_INFO[key] self.mode, rawmode = OPEN_INFO[key]
except KeyError as e: except KeyError as e:
logger.debug("- unsupported format") logger.debug("- unsupported format")
raise SyntaxError("unknown pixel mode") from e raise SyntaxError("unknown pixel mode") from e
logger.debug("- raw mode: {}".format(rawmode)) logger.debug(f"- raw mode: {rawmode}")
logger.debug("- pil mode: {}".format(self.mode)) logger.debug(f"- pil mode: {self.mode}")
self.info["compression"] = self._compression self.info["compression"] = self._compression
@ -1308,7 +1306,7 @@ class TiffImageFile(ImageFile.ImageFile):
if fillorder == 2: if fillorder == 2:
# Replace fillorder with fillorder=1 # Replace fillorder with fillorder=1
key = key[:3] + (1,) + key[4:] key = key[:3] + (1,) + key[4:]
logger.debug("format key: {}".format(key)) logger.debug(f"format key: {key}")
# this should always work, since all the # this should always work, since all the
# fillorder==2 modes have a corresponding # fillorder==2 modes have a corresponding
# fillorder=1 mode # fillorder=1 mode
@ -1432,7 +1430,7 @@ def _save(im, fp, filename):
try: try:
rawmode, prefix, photo, format, bits, extra = SAVE_INFO[im.mode] rawmode, prefix, photo, format, bits, extra = SAVE_INFO[im.mode]
except KeyError as e: except KeyError as e:
raise OSError("cannot write mode %s as TIFF" % im.mode) from e raise OSError(f"cannot write mode {im.mode} as TIFF") from e
ifd = ImageFileDirectory_v2(prefix=prefix) ifd = ImageFileDirectory_v2(prefix=prefix)
@ -1629,7 +1627,7 @@ def _save(im, fp, filename):
if s: if s:
break break
if s < 0: if s < 0:
raise OSError("encoder error %d when writing image file" % s) raise OSError(f"encoder error {s} when writing image file")
else: else:
offset = ifd.save(fp) offset = ifd.save(fp)
@ -1795,29 +1793,29 @@ class AppendingTiffWriter:
self.f.seek(-2, os.SEEK_CUR) self.f.seek(-2, os.SEEK_CUR)
bytesWritten = self.f.write(struct.pack(self.longFmt, value)) bytesWritten = self.f.write(struct.pack(self.longFmt, value))
if bytesWritten is not None and bytesWritten != 4: if bytesWritten is not None and bytesWritten != 4:
raise RuntimeError("wrote only %u bytes but wanted 4" % bytesWritten) raise RuntimeError(f"wrote only {bytesWritten} bytes but wanted 4")
def rewriteLastShort(self, value): def rewriteLastShort(self, value):
self.f.seek(-2, os.SEEK_CUR) self.f.seek(-2, os.SEEK_CUR)
bytesWritten = self.f.write(struct.pack(self.shortFmt, value)) bytesWritten = self.f.write(struct.pack(self.shortFmt, value))
if bytesWritten is not None and bytesWritten != 2: if bytesWritten is not None and bytesWritten != 2:
raise RuntimeError("wrote only %u bytes but wanted 2" % bytesWritten) raise RuntimeError(f"wrote only {bytesWritten} bytes but wanted 2")
def rewriteLastLong(self, value): def rewriteLastLong(self, value):
self.f.seek(-4, os.SEEK_CUR) self.f.seek(-4, os.SEEK_CUR)
bytesWritten = self.f.write(struct.pack(self.longFmt, value)) bytesWritten = self.f.write(struct.pack(self.longFmt, value))
if bytesWritten is not None and bytesWritten != 4: if bytesWritten is not None and bytesWritten != 4:
raise RuntimeError("wrote only %u bytes but wanted 4" % bytesWritten) raise RuntimeError(f"wrote only {bytesWritten} bytes but wanted 4")
def writeShort(self, value): def writeShort(self, value):
bytesWritten = self.f.write(struct.pack(self.shortFmt, value)) bytesWritten = self.f.write(struct.pack(self.shortFmt, value))
if bytesWritten is not None and bytesWritten != 2: if bytesWritten is not None and bytesWritten != 2:
raise RuntimeError("wrote only %u bytes but wanted 2" % bytesWritten) raise RuntimeError(f"wrote only {bytesWritten} bytes but wanted 2")
def writeLong(self, value): def writeLong(self, value):
bytesWritten = self.f.write(struct.pack(self.longFmt, value)) bytesWritten = self.f.write(struct.pack(self.longFmt, value))
if bytesWritten is not None and bytesWritten != 4: if bytesWritten is not None and bytesWritten != 4:
raise RuntimeError("wrote only %u bytes but wanted 4" % bytesWritten) raise RuntimeError(f"wrote only {bytesWritten} bytes but wanted 4")
def close(self): def close(self):
self.finalize() self.finalize()

View File

@ -69,15 +69,15 @@ class XbmImageFile(ImageFile.ImageFile):
def _save(im, fp, filename): def _save(im, fp, filename):
if im.mode != "1": if im.mode != "1":
raise OSError("cannot write mode %s as XBM" % im.mode) raise OSError(f"cannot write mode {im.mode} as XBM")
fp.write(("#define im_width %d\n" % im.size[0]).encode("ascii")) fp.write(f"#define im_width {im.size[0]}\n".encode("ascii"))
fp.write(("#define im_height %d\n" % im.size[1]).encode("ascii")) fp.write(f"#define im_height {im.size[1]}\n".encode("ascii"))
hotspot = im.encoderinfo.get("hotspot") hotspot = im.encoderinfo.get("hotspot")
if hotspot: if hotspot:
fp.write(("#define im_x_hot %d\n" % hotspot[0]).encode("ascii")) fp.write(f"#define im_x_hot {hotspot[0]}\n".encode("ascii"))
fp.write(("#define im_y_hot %d\n" % hotspot[1]).encode("ascii")) fp.write(f"#define im_y_hot {hotspot[1]}\n".encode("ascii"))
fp.write(b"static char im_bits[] = {\n") fp.write(b"static char im_bits[] = {\n")

View File

@ -39,7 +39,7 @@ if sys.version_info >= (3, 7):
if name == "PILLOW_VERSION": if name == "PILLOW_VERSION":
_raise_version_warning() _raise_version_warning()
return __version__ return __version__
raise AttributeError("module '{}' has no attribute '{}'".format(__name__, name)) raise AttributeError(f"module '{__name__}' has no attribute '{name}'")
else: else:

View File

@ -1,20 +1,9 @@
import os import os
import sys from pathlib import Path
py36 = sys.version_info[0:2] >= (3, 6)
if py36: def isPath(f):
from pathlib import Path return isinstance(f, (bytes, str, Path))
def isPath(f):
return isinstance(f, (bytes, str, Path))
else:
def isPath(f):
return isinstance(f, (bytes, str))
# Checks if an object is a string, and that it points to a directory. # Checks if an object is a string, and that it points to a directory.

View File

@ -25,7 +25,7 @@ def check_module(feature):
:raises ValueError: If the module is not defined in this version of Pillow. :raises ValueError: If the module is not defined in this version of Pillow.
""" """
if not (feature in modules): if not (feature in modules):
raise ValueError("Unknown module %s" % feature) raise ValueError(f"Unknown module {feature}")
module, ver = modules[feature] module, ver = modules[feature]
@ -78,7 +78,7 @@ def check_codec(feature):
:raises ValueError: If the codec is not defined in this version of Pillow. :raises ValueError: If the codec is not defined in this version of Pillow.
""" """
if feature not in codecs: if feature not in codecs:
raise ValueError("Unknown codec %s" % feature) raise ValueError(f"Unknown codec {feature}")
codec, lib = codecs[feature] codec, lib = codecs[feature]
@ -133,7 +133,7 @@ def check_feature(feature):
:raises ValueError: If the feature is not defined in this version of Pillow. :raises ValueError: If the feature is not defined in this version of Pillow.
""" """
if feature not in features: if feature not in features:
raise ValueError("Unknown feature %s" % feature) raise ValueError(f"Unknown feature {feature}")
module, flag, ver = features[feature] module, flag, ver = features[feature]
@ -182,7 +182,7 @@ def check(feature):
return check_codec(feature) return check_codec(feature)
if feature in features: if feature in features:
return check_feature(feature) return check_feature(feature)
warnings.warn("Unknown feature '%s'." % feature, stacklevel=2) warnings.warn(f"Unknown feature '{feature}'.", stacklevel=2)
return False return False
@ -230,18 +230,18 @@ def pilinfo(out=None, supported_formats=True):
Image.init() Image.init()
print("-" * 68, file=out) print("-" * 68, file=out)
print("Pillow {}".format(PIL.__version__), file=out) print(f"Pillow {PIL.__version__}", file=out)
py_version = sys.version.splitlines() py_version = sys.version.splitlines()
print("Python {}".format(py_version[0].strip()), file=out) print(f"Python {py_version[0].strip()}", file=out)
for py_version in py_version[1:]: for py_version in py_version[1:]:
print(" {}".format(py_version.strip()), file=out) print(f" {py_version.strip()}", file=out)
print("-" * 68, file=out) print("-" * 68, file=out)
print( print(
"Python modules loaded from {}".format(os.path.dirname(Image.__file__)), f"Python modules loaded from {os.path.dirname(Image.__file__)}",
file=out, file=out,
) )
print( print(
"Binary modules loaded from {}".format(os.path.dirname(Image.core.__file__)), f"Binary modules loaded from {os.path.dirname(Image.core.__file__)}",
file=out, file=out,
) )
print("-" * 68, file=out) print("-" * 68, file=out)
@ -283,9 +283,9 @@ def pilinfo(out=None, supported_formats=True):
extensions[i].append(ext) extensions[i].append(ext)
for i in sorted(Image.ID): for i in sorted(Image.ID):
line = "{}".format(i) line = f"{i}"
if i in Image.MIME: if i in Image.MIME:
line = "{} {}".format(line, Image.MIME[i]) line = f"{line} {Image.MIME[i]}"
print(line, file=out) print(line, file=out)
if i in extensions: if i in extensions:

View File

@ -417,7 +417,7 @@ def build_dep(name):
for patch_file, patch_list in dep.get("patch", {}).items(): for patch_file, patch_list in dep.get("patch", {}).items():
patch_file = os.path.join(sources_dir, dir, patch_file.format(**prefs)) patch_file = os.path.join(sources_dir, dir, patch_file.format(**prefs))
with open(patch_file, "r") as f: with open(patch_file) as f:
text = f.read() text = f.read()
for patch_from, patch_to in patch_list.items(): for patch_from, patch_to in patch_list.items():
patch_from = patch_from.format(**prefs) patch_from = patch_from.format(**prefs)