@@ -8,48 +8,48 @@ id: bar
8
8
9
9
We do not discourage authors to release software on Python 2. While this guide
10
10
is mostly written with the assumption that software are going to stop Python 2
11
- support, it does perfectly apply to a package that wish to not support Python 3,
11
+ support, it does perfectly apply to a package that wishes to not support Python 3,
12
12
or is stopping support for any minor version.
13
13
14
14
15
- This page gather information and links to resources allowing to release a
16
- library that stop supporting an older version of Python without causing too
15
+ This page gathers information and links to resources allowing a library
16
+ to stop supporting an older version of Python without causing too
17
17
much disruption for users who haven't upgraded to this new version.
18
18
19
- Whether you are a user, or a developer, being aware of the issue listed here, at
20
- least the main points should ease lots of the pain.
19
+ Whether you are a user or a developer, being aware of the issue listed here -- at
20
+ least the main points -- should ease lots of the pain.
21
21
22
22
# Too long, did not read:
23
23
24
24
- Help and encourage users to install ** pip 9.0+**
25
25
- Help and encourage users to install ** setuptools 24.3+**
26
- - As maintainer use ` setup(..., python_requires='>=3.4') ` new option.
27
- - do use ` pip install [-e] . ` and do ** not** invoke ` setup.py ` directly.
28
- - ** Fail** early at ** install time** if on Python 2.
29
- - We are giving a talk at PyCon 2017 (likely recorded), add link here .
26
+ - As maintainer, use the new ` setup(..., python_requires='>=3.4') ` option.
27
+ - Use ` pip install [-e] . ` and do ** not** invoke ` setup.py ` directly.
28
+ - ** Fail** early at ** install time** if user is on Python 2.
29
+ - We are giving a talk at PyCon 2017 (likely recorded; link to follow) .
30
30
31
31
## The problem
32
32
33
- Up until December 2016 it was hard to publish a new major version of library
34
- that changed requirements in Python version and mark it as such so that user
33
+ Up until December 2016 it was hard to publish a new major version of a library
34
+ that changed requirements in Python version and mark it as such so that a user's
35
35
system will not try to upgrade said library.
36
36
37
37
With the recent changes in Python packaging this is now possible.
38
38
39
- As an example let's look at the example of the ` fictitious ` library.
39
+ As an example let's look at a non-existent ` fictitious ` library.
40
40
41
41
- ` fictitious ` 1.1, 1.2, 1.3, 1.4 are compatible Python 2.7 and 3.3+
42
42
- ` fictitious ` 2.0 has been released and is python 3.4+ only.
43
43
44
44
As a Python 2.7 user, if I don't pay attention, or if the library is not
45
- correctly tagged, if I issue the following :
45
+ correctly tagged, there can be issues when you try to update the library :
46
46
47
47
$ python -c 'import fictitious; print(fictitious.__version__)'
48
48
1.3.2
49
49
$ pip install fictitious --upgrade
50
50
51
- Either my system will install 2.0, which will not work, on the worst case
52
- scenario, or fail to install, in which case I will not get the critical 1.4
51
+ Either my system will install 2.0, which will not work -- the worst case
52
+ scenario -- or fail to install, in which case I will not get the critical 1.4
53
53
upgrade.
54
54
55
55
## As a user
@@ -60,17 +60,17 @@ If you are already a Python 3 user, you should not encounter a lot of
60
60
disruption. Please still check that the libraries you use follow best practices
61
61
not to break for Python 2 users. Python is a community regardless of which
62
62
python version you have to (or decided to) run, making sure that everything
63
- works make the community strong.
63
+ works makes the community strong.
64
64
65
65
Make sure you have Pip ≥ 9.0, this is especially important if you have Python
66
- 2 installations. Having pip 9.0+ is not a guaranty to flawless upgrade. But pip
67
- 9.0+ does have a number of safety check not available on previous versions.
66
+ 2 installations. Having pip 9.0+ is not a guarantee of a flawless upgrade. But pip
67
+ 9.0+ does have a number of safety check not available in previous versions.
68
68
69
69
Having a version of pip < 9.0 can lead your system to try to upgrade to
70
70
non-compatible versions of Python packages even if these are marked as
71
71
non-compatible.
72
72
73
- Help as many other _ users_ as possible to install pip ≥ 9.0, for the
73
+ Help as many other _ users_ as possible to install pip ≥ 9.0. For the
74
74
transition, it is the slowest part of the ecosystem to update, and is the only
75
75
piece that requires action of all Python users.
76
76
@@ -92,16 +92,16 @@ All good.
92
92
93
93
## Setuptools
94
94
95
- If you are on a system for which no wheel is available, pip will try to
95
+ If you are on a system for which no wheel is available, pip will try to
96
96
install a source distribution (aka ` sdist ` ).
97
97
98
- Installing an ` sdist ` will require setuptools make sure you have setuptools
99
- ≥ 24.2.0 or building Python 3 only libraries is likely to fail. In particular
98
+ Installing an ` sdist ` will require setuptools, so make sure you have setuptools
99
+ ≥ 24.2.0 or building Python 3- only libraries will likely fail. In particular
100
100
if library authors have taken time to mark their library as Python 3 only, the
101
101
` python_requires ` argument to ` setup() ` may not be recognized and installation
102
102
will fail.
103
103
104
- Use the following to check setuptools version :
104
+ Use the following to check your setuptools version :
105
105
106
106
$ python -c 'import setuptools; print(setuptools.__version__)'
107
107
24.2.0
@@ -114,9 +114,9 @@ date system:
114
114
## Local package index
115
115
116
116
If you are using a custom local package index, for example if you are working
117
- at a company with private packages, make sure it implement correctly
118
- [ pep-503] ( https://www.python.org/dev/peps/pep-0503/ ) and let pip knows about
119
- the ` python_requires ` field. This _ mostly_ mean that the html you are exposing
117
+ at a company with private packages, make sure it correctly implements
118
+ [ pep-503] ( https://www.python.org/dev/peps/pep-0503/ ) to let pip know about
119
+ the ` python_requires ` field. This _ mostly_ mean that the HTML you are exposing
120
120
should get a ` data-python-requires ` data attribute with the (html escaped)
121
121
version specifier.
122
122
@@ -129,20 +129,20 @@ insure they support this new functionality.
129
129
# Preparing your library
130
130
131
131
132
- As a library author one of the most important factor in a smooth transition is
132
+ As a library author one of the most important factors in a smooth transition is
133
133
planning and communication, letting your user base know in advance that the
134
134
transition is happening and what step to take is critical for a transition.
135
135
136
136
For your library code here the steps you need to take to ensure that
137
- installation will fail in the least number of case :
137
+ installation will fail in the least number of cases :
138
138
139
- You need to release your new packages version with
139
+ You need to release your package's new version with
140
140
[ setuptools] ( https://pypi.python.org/pypi/setuptools ) version 24.2.0 or above.
141
- You can also use one of the alternate package manager that can set the
141
+ You can also use one of the alternate package managers that can set the
142
142
[ Requires-Python] ( https://www.python.org/dev/peps/pep-0345/#requires-python )
143
- metadata field. Without this, pip 9.0 ** will try** to install non-compatible
143
+ metadata field. Without this, pip 9.0 ** will try** to install a non-compatible
144
144
version of your software on Python 2. This version of setuptools is recent
145
- (July 20, 2016) and this possible thank to the [ work of Xavier
145
+ (July 20, 2016) and this is all possible thanks to the [ work of Xavier
146
146
Fernandez] ( https://github.com/pypa/setuptools/pull/631 )
147
147
148
148
Add the following to your ` setup.py `
@@ -172,19 +172,20 @@ they will get the right version of your library.
172
172
It is recommended ** not** to invoke ` setup.py ` directly either with ` install ` or
173
173
` develop ` subcommands. These may not correctly resolve dependencies, and can
174
174
install incompatible versions of dependencies. Please recommend and use `pip
175
- install .` and ` pip install -e .` for regular and developer install .
175
+ install .` and ` pip install -e .` for regular and developer installs, respectively .
176
176
177
- Check in scripts, and documentation that the correct installation command is
177
+ Check in scripts and documentation that the correct installation command is
178
178
used.
179
179
180
180
# Recommended Mitigations
181
181
182
182
These are not mandatory but should make the transition seamless by warning your
183
- user early enough _ and_ providing useful error messages.
183
+ users early enough _ and_ providing useful error messages.
184
184
185
185
## Runtime warning on master
186
186
187
- Add a warning at _ runtime_ early on master (before switching to Python 3 only)
187
+ Add a warning at _ runtime_ that triggers early on master
188
+ (before switching to Python 3 only)
188
189
189
190
```
190
191
import warnings
@@ -196,19 +197,20 @@ if sys.version_info < (3,):
196
197
UserWarning)
197
198
```
198
199
199
- Your Python 2 user have a chance to upgrade, or get off master, (for example on
200
- the LTS branch).
200
+ Your Python 2 users will have a chance to upgrade, or get off master,
201
+ (for example on the LTS branch).
201
202
202
203
## Fail early at import time
203
204
204
- Add an error early at import at runtime with a clear error message, leave the
205
- early import compatible Python 2 for users to not be welcomed with a useless
206
- ` SyntaxError ` . Don't hesitate to use multi-line strings in error messages.
205
+ Add an error early through import at runtime with a clear error message, leave the
206
+ early import compatible Python 2 as users will not feel welcomed with a useless
207
+ ` SyntaxError ` due to their Python 2 usage. Don't hesitate to use multi-line strings
208
+ in error messages.
207
209
208
- Error at import time _ will_ happen on system with old version of pip and
210
+ Error at import time _ will_ happen on systems with old version of pip and
209
211
setuptools. Keep in mind that saying the package is Python 3 only is not a lot
210
- more helpful than a Syntax error . The most reasonable reason would be out of
211
- data pip and setuptools:
212
+ more helpful than a ` SyntaxError ` . The most reasonable reason would be
213
+ out-of-date pip and setuptools:
212
214
213
215
214
216
```
@@ -220,7 +222,7 @@ if sys.version_info < (3,):
220
222
221
223
Unfortunately Frobulator 6.0 and above are not compatible with Python 2
222
224
anymore, and you still ended up with this version installed on your system.
223
- That's a bummer. Sorry about that. It should not have happened. Make sure you
225
+ That's a bummer; sorry about that. It should not have happened. Make sure you
224
226
have pip ≥ 9.0 to avoid this kind of issues, as well as setuptools ≥ 24.2:
225
227
226
228
$ pip install pip setuptools --upgrade
@@ -247,14 +249,15 @@ https://i.am.an/url
247
249
## Watch out for beta releases
248
250
249
251
250
- Make sure your version number match pep 440 or you will get surprises during
251
- beta in particular as the ` sdist ` and ` wheel ` will appear as being different
252
- versions, in particular sdist (during beta/rc/post) can appear with a greater
253
- version number than wheels. Pip thus try to install the sdist instead of the
254
- wheel, which have more chance of failing, in particular with pre 24.2 versions
255
- of setuptools.
252
+ Make sure your version number matches
253
+ [ PEP 440] ( https://www.python.org/dev/peps/pep-0440/ ) or you will get surprises
254
+ during beta, in particular as the ` sdist ` and ` wheel ` will appear as being
255
+ different versions (the sdist (during beta/rc/post) can appear with
256
+ a greater version number than wheels). Pip thus will try to install the sdist
257
+ instead of the wheel, which has more chance of failing, in particular with
258
+ pre-24.2 versions of setuptools.
256
259
257
- The regular expression to check for validity of pep440 can be find below:
260
+ The regular expression to check for validity of pep440 can be found below:
258
261
259
262
^
260
263
([1-9]\\d*!)?
@@ -268,11 +271,11 @@ The regular expression to check for validity of pep440 can be find below:
268
271
## fail early in setup.py
269
272
270
273
Leave ` setup.py ` python 2 compatible and fail early. If you detect Python 2
271
- raise a clear error message and ask user to make sure they have pip > 9.0 (or
274
+ raise a clear error message and ask the user to make sure they have pip > 9.0 (or
272
275
migrate to Python 3). You can (try to) conditionally import pip and check for
273
276
its version but this might not be the same pip. Failing early is important to
274
277
make sure the Python installation does not install an incompatible version.
275
- Otherwise user code can fail at runtime arbitrary later in the future, which can
278
+ Otherwise user code can fail at runtime arbitrarily later in the future, which can
276
279
be a difficult to debug and fix. Get inspiration from the message of failure at
277
280
runtime, and adapt for installation time.
278
281
@@ -281,7 +284,7 @@ runtime, and adapt for installation time.
281
284
If you control dependant packages, Make sure to include conditional dependencies
282
285
depending on the version of Python.
283
286
284
- # Non recommended mitigations
287
+ # Non- recommended mitigations
285
288
286
289
This is a collection of "mitigation" or "solutions" you will find on the web
287
290
and that you will hear about. This is an attempt to acknowledge them, and
@@ -290,65 +293,66 @@ implement them.
290
293
291
294
### Use a meta-package.
292
295
293
- It is possible to release a meta-package that has _ virtually_ no code and rely
294
- on conditional dependency to install its actual core code on the user system.
296
+ It is possible to release a meta-package that has _ virtually_ no code and relies
297
+ on a conditional dependency to install its actual core code on the user system.
295
298
For example, Frob-6.0 could be a meta-package which depends on
296
299
Frob-real-py2 on Python < 3.0, and Frob-real-py3 on Python ≥ 3.4. While
297
300
this approach is _ doable_ this can make imports confusing.
298
301
299
302
## Depend on setuptools
300
303
301
- You can mark your library as dependent on setuptools greater than 24.3 this
304
+ You can mark your library as dependent on setuptools greater than 24.3 as this
302
305
will insure that during the next upgrade (when the packages drop python 2
303
306
support) will have the right version of setuptools.
304
307
305
308
Of course regardless of all the care you will take for your library to no break
306
- and to install only on python 2, you will likely have cases where it still end
307
- up being installed on incompatible versions of Python. Simply because users
308
- upgrades rarely and only an old version of pip or setuptools is enough to make
309
- the all update process broken.
309
+ and to install only on python 2, you will likely have cases where it will still
310
+ end up being installed on incompatible versions of Python. Simply because users
311
+ upgrade rarely and only an old version of pip or setuptools is enough to make
312
+ the update process broken.
310
313
311
314
Plus setuptools is rarely an actual dependency of your project but a
312
315
requirement to build wheels.
313
316
314
317
315
- ### Multiple Sdist.
318
+ ### Multiple sdist files
316
319
317
320
Pip (used to) support a "feature" where a sdist ending in ` -pyX.Y.tar.gz ` would
318
321
only be seen as compatible on Python X.Y, thus it used to be possible to
319
322
publish multiple sdist of a package targeting various python version.
320
323
321
- Though it is not possible anymore to upload multiple sdist on PyPI. This
322
- solution is thus not possible .
324
+ It is not possible anymore to upload multiple sdist files on PyPI, so this
325
+ solution is no longer tenable .
323
326
324
327
### Wheel only ?
325
328
326
- Releasing a package only using wheel for a given python version is doable, but
329
+ Releasing a package only using wheels for a given python version is doable, but
327
330
this will break downstream packages that may require the original source to
328
- reproduce the build.
331
+ reproduce their build.
329
332
330
- # Why all that ?
333
+ # Why all * this * ?! ?
331
334
332
- You might wonder why all this, it's 2016 already, so how come all these
333
- issues ? Python 3 has been out for 8+ years now !
335
+ You might wonder why all this, it's 2016 already, so how come this is now an
336
+ issue ? Python 3 has been out for 8+ years now !
334
337
335
- Well there are many reasons to this, first of all, this issue mostly affect
338
+ Well there are many reasons for this. First of all, this issue mostly affects
336
339
libraries that are currently python 2 and Python 3 compatible at the same time.
337
340
Many libraries have transitioned from Python 2-only to Python 2 + 3. And the
338
341
issue of transitioning to Python 3 only is relatively recent. Technically it
339
- can also apply to libraries that are only stopping support for 2.6, or even are
340
- already Python 3 only, but are starting to stop support for earlier versions of
341
- Python. For example a library releasing a Python 3.4+ only version.
342
+ can also apply to libraries that are only stopping support for 2.6, or are even
343
+ already Python 3 only, but are starting to stop supporting earlier versions of
344
+ Python (for example a library releasing a Python 3.4+ only version) .
342
345
343
- Python 3.3 was release at the end of 2012, and was the first version to
346
+ Python 3.3 was released at the end of 2012, and was the first version to
344
347
support (again) ` u ` as a prefix for Unicode string. It was one of the first
345
- minor version of Python 3 that saw a majority of single-source project working
346
- both on Python 2 and Python 3. These are the Project that will likely be
348
+ minor versions of Python 3 that saw a majority of single-source projects working
349
+ both on Python 2 and Python 3. These are the projects that will likely be
347
350
affected by this issue.
348
351
349
- The introduction of Python 3 was chaotic, there are still strong argument both
350
- in Python 2 and Python 3 camps. In the one suffering the most from this are
351
- users. Starting with the fact that inevitably some libraries will stop support
352
- for Python 2 and release Python 3 only library. And that inevitably some system
353
- will will not be upgraded to Python 3 how can we _ ensure_ that users get the
354
- _ least_ breakage as possible ? And what are the best practices to follow.
352
+ The introduction of Python 3 was chaotic; there are still strong arguments in both
353
+ the Python 2 and Python 3 camps. Regardless of what side you take, the ones suffering
354
+ the most from this are users (starting with the fact that inevitably some libraries
355
+ will stop supporting for Python 2 and release Python 3 only library). Inevitably, some
356
+ systems and people will will not be upgraded to Python 3, so this document hopefully
357
+ helps to _ ensure_ that users get the _ least_ breakage as possible and what are the best
358
+ practices are to follow.
0 commit comments