Skip to content

Commit ddf44d8

Browse files
author
Release Manager
committed
Trac #31306: sage.repl: Replace use of SAGE_EXTCODE by importlib.resources
We eliminate direct reading of files from the package directories of sagelib by using `importlib.resources` (available since python 3.7) - `git grep 'SAGE_EXTCODE' src/sage` - `git grep '__file__' src/sage` This will help make sagelib `zip_safe` (https://setuptools.readthedocs.i o/en/latest/userguide/miscellaneous.html#setting-the-zip-safe-flag) In this ticket, we take care of `sage.repl`, moving data needed by its doctests from `SAGE_EXTCODE` (= `src/sage/ext_data`) into the package directory. (Thus, we avoid a dependency on support for resources in namespace packages brought by Python 3.10(?) or the backport package `importlib-resources`, see python/importlib_resources#196) Follow-up tickets will deal with other parts of the library. In the end, the directory `src/sage/ext_data` will be eliminated. References: - https://docs.python.org/3/library/importlib.html#module- importlib.resources - https://importlib-resources.readthedocs.io/en/latest/migration.html - https://importlib-resources.readthedocs.io/en/latest/using.html URL: https://trac.sagemath.org/31306 Reported by: mkoeppe Ticket author(s): Matthias Koeppe Reviewer(s): Dima Pasechnik
2 parents d04fbee + d925cd2 commit ddf44d8

23 files changed

+27
-54
lines changed

src/sage/repl/display/formatter.py

+3-3
Original file line numberDiff line numberDiff line change
@@ -142,16 +142,16 @@ def format(self, obj, include=None, exclude=None):
142142
TESTS::
143143
144144
sage: import os
145-
sage: from sage.env import SAGE_EXTCODE
146-
sage: example_png = os.path.join(SAGE_EXTCODE, 'doctest', 'rich_output', 'example.png')
145+
sage: import importlib.resources
147146
sage: from sage.repl.rich_output.backend_ipython import BackendIPython
148147
sage: backend = BackendIPython()
149148
sage: shell = get_test_shell()
150149
sage: backend.install(shell=shell)
151150
sage: shell.run_cell('get_ipython().display_formatter')
152151
<sage.repl.display.formatter.SageDisplayFormatter object at 0x...>
153152
sage: shell.run_cell('from IPython.display import Image')
154-
sage: shell.run_cell('ipython_image = Image("{0}")'.format(example_png))
153+
sage: with importlib.resources.path(sage.repl.rich_output, 'example.png') as example_png:
154+
....: shell.run_cell('ipython_image = Image("{0}")'.format(example_png))
155155
sage: shell.run_cell('ipython_image')
156156
<IPython.core.display.Image object>
157157
sage: shell.run_cell('get_ipython().display_formatter.format(ipython_image)')

src/sage/repl/rich_output/output_graphics.py

+7-24
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@
1818

1919
import os
2020
import base64
21+
import importlib.resources
2122

2223
from sage.cpython.string import bytes_to_str
2324
from sage.repl.rich_output.output_basic import OutputBase
@@ -74,10 +75,7 @@ def example(cls):
7475
sage: OutputImagePng.example().png.get().startswith(b'\x89PNG')
7576
True
7677
"""
77-
from sage.env import SAGE_EXTCODE
78-
filename = os.path.join(SAGE_EXTCODE, 'doctest', 'rich_output', 'example.png')
79-
with open(filename, 'rb') as f:
80-
return cls(f.read())
78+
return cls(importlib.resources.read_binary(__package__, 'example.png'))
8179

8280

8381
class OutputImageGif(OutputBase):
@@ -125,10 +123,7 @@ def example(cls):
125123
sage: OutputImageGif.example().gif.get().startswith(b'GIF89a')
126124
True
127125
"""
128-
from sage.env import SAGE_EXTCODE
129-
filename = os.path.join(SAGE_EXTCODE, 'doctest', 'rich_output', 'example.gif')
130-
with open(filename, 'rb') as f:
131-
return cls(f.read())
126+
return cls(importlib.resources.read_binary(__package__, 'example.gif'))
132127

133128
def html_fragment(self):
134129
"""
@@ -195,10 +190,7 @@ def example(cls):
195190
sage: OutputImageJpg.example().jpg.get().startswith(b'\xff\xd8\xff\xe0\x00\x10JFIF')
196191
True
197192
"""
198-
from sage.env import SAGE_EXTCODE
199-
filename = os.path.join(SAGE_EXTCODE, 'doctest', 'rich_output', 'example.jpg')
200-
with open(filename, 'rb') as f:
201-
return cls(f.read())
193+
return cls(importlib.resources.read_binary(__package__, 'example.jpg'))
202194

203195

204196
class OutputImageSvg(OutputBase):
@@ -246,10 +238,7 @@ def example(cls):
246238
sage: b'</svg>' in OutputImageSvg.example().svg.get()
247239
True
248240
"""
249-
from sage.env import SAGE_EXTCODE
250-
filename = os.path.join(SAGE_EXTCODE, 'doctest', 'rich_output', 'example.svg')
251-
with open(filename, 'rb') as f:
252-
return cls(f.read())
241+
return cls(importlib.resources.read_binary(__package__, 'example.svg'))
253242

254243

255244
class OutputImagePdf(OutputBase):
@@ -297,10 +286,7 @@ def example(cls):
297286
sage: OutputImagePdf.example().pdf.get().startswith(b'%PDF-1.4')
298287
True
299288
"""
300-
from sage.env import SAGE_EXTCODE
301-
filename = os.path.join(SAGE_EXTCODE, 'doctest', 'rich_output', 'example.pdf')
302-
with open(filename, 'rb') as f:
303-
return cls(f.read())
289+
return cls(importlib.resources.read_binary(__package__, 'example.pdf'))
304290

305291

306292
class OutputImageDvi(OutputBase):
@@ -348,7 +334,4 @@ def example(cls):
348334
sage: b'TeX output' in OutputImageDvi.example().dvi.get()
349335
True
350336
"""
351-
from sage.env import SAGE_EXTCODE
352-
filename = os.path.join(SAGE_EXTCODE, 'doctest', 'rich_output', 'example.dvi')
353-
with open(filename, 'rb') as f:
354-
return cls(f.read())
337+
return cls(importlib.resources.read_binary(__package__, 'example.dvi'))

src/sage/repl/rich_output/output_graphics3d.py

+10-20
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@
1616

1717

1818
import os
19+
import importlib.resources
1920

2021
from sage.cpython.string import bytes_to_str, FS_ENCODING
2122
from sage.repl.rich_output.output_basic import OutputBase
@@ -110,15 +111,8 @@ def example(cls):
110111
sage: rich_output.preview_png.get().startswith(b'\x89PNG')
111112
True
112113
"""
113-
from sage.env import SAGE_EXTCODE
114-
example_png_filename = os.path.join(
115-
SAGE_EXTCODE, 'doctest', 'rich_output', 'example.png')
116-
with open(example_png_filename, 'rb') as f:
117-
example_png = f.read()
118-
scene_zip_filename = os.path.join(
119-
SAGE_EXTCODE, 'doctest', 'rich_output', 'example_jmol.spt.zip')
120-
with open(scene_zip_filename, 'rb') as f:
121-
scene_zip = f.read()
114+
example_png = importlib.resources.read_binary(__package__, 'example.png')
115+
scene_zip = importlib.resources.read_binary(__package__, 'example_jmol.spt.zip')
122116
return cls(scene_zip, example_png)
123117

124118

@@ -163,10 +157,8 @@ def example(cls):
163157
sage: rich_output.canvas3d.get_str()
164158
'[{"vertices":[{"x":1,"y":1,"z":1},...{"x":1,"y":-1,"z":-1}],"faces":[[0,1,2,3]],"color":"008000"}]'
165159
"""
166-
from sage.env import SAGE_EXTCODE
167-
filename = os.path.join(
168-
SAGE_EXTCODE, 'doctest', 'rich_output', 'example.canvas3d')
169-
return cls(OutputBuffer.from_file(filename))
160+
with importlib.resources.path(__package__, 'example.canvas3d') as filename:
161+
return cls(OutputBuffer.from_file(filename))
170162

171163

172164
class OutputSceneThreejs(OutputBase):
@@ -358,10 +350,8 @@ def example(cls):
358350
sage: rich_output.mtl.get_str()
359351
'newmtl texture177\nKa 0.2 0.2 0.5\nKd 0.4 0.4 1.0\nKs 0.0 0.0 0.0\nillum 1\nNs 1\nd 1\n'
360352
"""
361-
from sage.env import SAGE_EXTCODE
362-
with_path = lambda x: os.path.join(
363-
SAGE_EXTCODE, 'doctest', 'rich_output', 'example_wavefront', x)
364-
return cls(
365-
OutputBuffer.from_file(with_path('scene.obj')),
366-
OutputBuffer.from_file(with_path('scene.mtl')),
367-
)
353+
with importlib.resources.path(__package__, 'example_wavefront_scene.obj') as filename:
354+
scene_obj = OutputBuffer.from_file(filename)
355+
with importlib.resources.path(__package__, 'example_wavefront_scene.mtl') as filename:
356+
scene_mtl = OutputBuffer.from_file(filename)
357+
return cls(scene_obj, scene_mtl)

src/sage/repl/rich_output/output_video.py

+4-5
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@
1616

1717

1818
import os
19+
import importlib.resources
1920

2021
from sage.repl.rich_output.output_basic import OutputBase
2122
from sage.repl.rich_output.buffer import OutputBuffer
@@ -70,11 +71,9 @@ def example(cls):
7071
sage: OutputVideoOgg.example().mimetype
7172
'video/ogg'
7273
"""
73-
from sage.env import SAGE_EXTCODE
74-
filename = os.path.join(SAGE_EXTCODE, 'doctest', 'rich_output',
75-
'example' + cls.ext)
76-
return cls(OutputBuffer.from_file(filename),
77-
{'controls': True, 'loop': False})
74+
with importlib.resources.path(__package__, 'example' + cls.ext) as filename:
75+
return cls(OutputBuffer.from_file(filename),
76+
{'controls': True, 'loop': False})
7877

7978
def html_fragment(self, url, link_attrs=''):
8079
r"""

src/setup.cfg.m4

+3-2
Original file line numberDiff line numberDiff line change
@@ -133,6 +133,9 @@ sage.interfaces =
133133
sage.doctest =
134134
tests/*
135135

136+
sage.repl.rich_output =
137+
example*
138+
136139
sage =
137140
ext_data/*
138141
ext_data/kenzo/*
@@ -141,8 +144,6 @@ sage =
141144
ext_data/images/*
142145
ext_data/doctest/*
143146
ext_data/doctest/invalid/*
144-
ext_data/doctest/rich_output/*
145-
ext_data/doctest/rich_output/example_wavefront/*
146147
ext_data/gap/*
147148
ext_data/gap/joyner/*
148149
ext_data/mwrank/*

0 commit comments

Comments
 (0)