Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

bpo-33201: Modernize "Extension types" doc #6337

Merged
merged 7 commits into from
Apr 7, 2018

Conversation

pitrou
Copy link
Member

@pitrou pitrou commented Apr 1, 2018

  • Use C99 idioms for struct initialization
  • Streamline and clarify explanations
  • Fix some outdated code
  • Split the tutorial part from the other topics, for clarity

https://bugs.python.org/issue33201

@pitrou pitrou force-pushed the modernize-extending branch 2 times, most recently from 9a897d0 to 8302432 Compare April 1, 2018 20:17
@pitrou
Copy link
Member Author

pitrou commented Apr 1, 2018

@ncoghlan

@pitrou pitrou force-pushed the modernize-extending branch from 8302432 to e2fa664 Compare April 1, 2018 21:00
.. note::
The explicit cast to ``destructor`` above is needed because we defined
``Noddy_dealloc`` to take a ``NoddyObject *`` argument, but the ``tp_dealloc``
function pointer expects to receive a ``PyObject *`` argument. Otherwise,
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why not define Noddy_dealloc as accepting a PyObject pointer, and then convert the parameter to a NoddyObject pointer inside the function? That should avoid the compiler warning, and avoid Python calling the function with the wrong prototype, which I understand is undefined in standard C and Posix.

If you just want to explain the cast without rewriting the code, perhaps admit that the function prototype is wrong and the cast hides a related compiler warning.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's an idiom that is safely used in many parts of CPython. I don't think it's "undefined" in C to do that (a pointer is a pointer, regardless of what it points to), unlike perhaps C++. Do you have a reference?

The alternative needs either many casts or using an additional variable inside the function, both of which are more cumbersome.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's compllicated, so I could be wrong, but C99 says "If a converted pointer is used to call a function whose type is not compatible with the pointed-to type, the behavior is undefined" (sect. 6.3.2.3 par. 8). The function types are not compatible because "For two function types to be compatible, . . . corresponding parameters shall have compatible types" (sect. 6.7.5.3 par. 15). The parameters are not compatible because they point to incompatible structures (NoddyObject and PyObject): "For two pointer types to be compatible, . . . both shall be pointers to compatible types" (sect. 6.5.7.2 par. 2).

Yes, Python's own code does this function pointer casting a lot itself with success, but it never seemed right to me, and I don't think it is good to claim the cast is "needed" when there is an alternative.

There are a couple of recent bug reports about GCC 8 producing new warnings about explicit function pointer casts, e.g. https://bugs.python.org/issue33012 about METH_NOARGS and PyCFunction.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

bpo-33012 is about wrong number of arguments, right?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think you are right. Looking closer, the bug reports seem to be about GCC's "-Wcast-function-type" warning, and one of the documented exceptions to that warning is "Any parameter of pointer-type matches any other pointer-type."



.. _weakref-support:

Weak Reference Support
----------------------

One of the goals of Python's weak-reference implementation is to allow any type
One of the goals of Python's weak reference implementation is to allow any type
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why this change? Isn’t it an implementation of weak-references, not a reference-implementation that is weak?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We don't use the "weak-reference" spelling anywhere else - they're always just "weak references".

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, there were some odd spellings in that file, including "weak-reference" and "garbage-collector".

}
Py_INCREF(value);
Py_CLEAR(self->first);
self->first = value;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If you’re improving the logic here for replacing an attribute, is there a reason why it has to be different to the INCREF / DECREF dance already done in Noddy_init?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You're right, it would be simpler for the reader to use the same idiom.

Copy link
Contributor

@ncoghlan ncoghlan left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This looks like a nice improvement to me. I've made a bunch of comments inline, but none of them are specific to the changes - they're just a result of reading through this doc for the first time in a long time and considering how it could be made more approachable.



.. _weakref-support:

Weak Reference Support
----------------------

One of the goals of Python's weak-reference implementation is to allow any type
One of the goals of Python's weak reference implementation is to allow any type
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We don't use the "weak-reference" spelling anywhere else - they're always just "weak references".

the :file:`Objects` directory, then search the C source files for ``tp_`` plus
the function you want (for example, ``tp_richcompare``). You will find examples
of the function you want to implement.
get the :term:`CPython` source code. Go to the :file:`Objects` directory,
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.


Python allows the writer of a C extension module to define new types that
can be manipulated from Python code, much like the built-in :class:`str`
and :class:`list` types. This is not hard; the code for all extension types
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't think we need the "This is not hard" comment here, as for folks that aren't experienced C programmers, it is going to be hard.

We may also want to repeat the cross-references to Cython/CFFI/etc, and perhaps point to https://packaging.python.org/guides/packaging-binary-extensions/#an-overview-of-binary-extensions for further discussion of the available alternatives.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Agreed. I'll remove that wording.

That's it! All that remains is to build it; put the above code in a file called
:file:`noddy.c` and ::

from distutils.core import setup, Extension
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Using setuptools is likely to be more robust here, since it tends to be better at finding compilers, emits up to date package metadata, and can be used to build wheel archives in combination with bdist_wheel.

https://packaging.python.org/tutorials/installing-packages/#ensure-pip-setuptools-and-wheel-are-up-to-date is the appropriate reference if folks just want to build locally, while they'll want https://packaging.python.org/tutorials/distributing-packages/ if they plan to actually create an sdist and/or wheel file for their extension module.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Right, I should add a link to that external resource.

easily use the :class:`PyTypeObject` it needs. It can be difficult to share
these :class:`PyTypeObject` structures between extension modules.

In this example we will create a :class:`Shoddy` type that inherits from the
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Seeing Shoddy as the example name makes me think "This is an example of a shoddy class implementation".

Perhaps it would be clearer if the examples were just Example and SubExample rather than Noddy and Shoddy?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Alternatively, since "the Example example class" can read strangely, we could rename the examples based on what they demonstrate:

  • ExtMinimal (replacing Noddy1)
  • ExtWithAttributes (replacing Noddy2)
  • ExtWithSetters (replacing Noddy3)
  • ExtWithGC (replacing Noddy4)
  • ExtWithParent (replacing Shoddy)

Where the Ext prefix is short for Extension type.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

To be honest, the names "Noddy" and "Shoddy" look like a joke by the original author(s). Overall the examples are silly, but it's hard to come with better ones (that still fit in the tutorial's structure).

I don't like the ExtSomething names either. We probably need something simple yet evocative...

At least Shoddy could be renamed SubList, I think.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah, I'm not especially fond of the Ext... names either.

I do like Minimal as the name for the first example, since it does strip class definition down to its essence (i.e. no state, no behaviour).

PublicAttrs would describe the second example, since it exposes the data attributes directly.

GetSetAttrs would describe the third example, since it hides the data attributes behind getter and setter functions.

CyclicRefs would describe the fourth example, since it handles the case where the object ends up referencing itself (directly or indirectly).

And then SubList for the list subclassing example.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I find it nice that the current 4 examples keep the same object name, since each builds on top of the previous one. It feels like a distraction for the object name to change each time. Perhaps call it Something?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

"the Custom example class" and "the example Custom class" both read OK to me (whereas those were the phrases that put me off Example), and I think it sounds OK in "custom module" and "Custom instance" as well.

So if it sounds OK to you as well, I'd suggest going with that.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think Noddy and Shoddy are pretty good names for examples. Let not kill jokes in the documentation for a programming language named after the comic group.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't know if those names are jokes (what does "Noddy" mean exactly)? Those names are a bit confusing.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

While Noddy and Shoddy are presumably meant just in fun, I found myself distracted by their actual meanings while reviewing Antoine's changes:

Noddy = this is pointless ("Yeah, it's a bit of a noddy change, but the boss insisted")
Shoddy = this is poorly implemented ("They did a shoddy job and I'm not happy with it")

For PEP 426, I used names like ComfyChair and SoftCushions in the examples, and picking some thematically appropriate arbitrary nouns from a Monty Python sketch may work here, too.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think it's good to avoid confusion with bizarre names so will probably change to the more neutral "Custom".

@serhiy-storchaka serhiy-storchaka self-requested a review April 2, 2018 22:24
.tp_name = "noddy.Noddy",
.tp_doc = "Noddy objects",
.tp_basicsize = sizeof(NoddyObject),
.tp_itemsize = 0,
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is this needed?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The text explains tp_itemsize.

Py_INCREF(&noddy_NoddyType);
PyModule_AddObject(m, "Noddy", (PyObject *)&noddy_NoddyType);
Py_INCREF(&NoddyType);
PyModule_AddObject(m, "Noddy", (PyObject *) &NoddyType);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Don't know if this is covered by PEP 7, but I think that it is more common to not add a space between closing ) and the casted value.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmm, I'd think the contrary.

(initproc)Noddy_init, /* tp_init */
0, /* tp_alloc */
Noddy_new, /* tp_new */
.tp_name = "noddy.Noddy",
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

"noddy2.Noddy"?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Good catch :-)

Py_INCREF(value);
Py_CLEAR(self->first);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You should use the dance with tmp here. Py_CLEAR(self->first) calls the destructor. The first attribute can be set directly in the destructor or it can be set in other thread when the GIL will be released in the destructor. self->first = value in the following line will rewrite the new value of the first attribute with leaking a reference.

Py_SETREF() could help here, but it is not in a public API.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Py_CLEAR should first transfer the pointer to a temporary before decref'ing it, so I don't see where the problem is.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The problem is virtually the same as in __init__.

  • Thread A calls Noddy_setfirst with the argument valueA, runs to Py_CLEAR(self->first) in which sets tmp = self->first; self->first = NULL and executes Py_XDECREF(tmp) which calls the destructor and releases the GIL. This causes switching to other thread.
  • Thread B calls Noddy_setfirst, and sets self->first to a new value valueB (it was set to NULL in thread A).
  • Thread A continues execution. The next line is self->first = valueA. It is expected that the value of self->first is NULL (as set by Py_CLEAR), but actually it is valueB. After setting self->first = valueA valueB will be leaked.

The same race condition is possible in a single-thread case if the destructor of the old attribute value calls Noddy_setfirst.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for clearing that up. I'll do the change.

};

PyMODINIT_FUNC
PyInit_shoddy(void)
{
PyObject *m;

PyObject* m;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It is more common to add a space before * than after it.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Uh, right. I'm confused between the coding styles of two different projects :-(


This is what a Noddy object will contain. ``PyObject_HEAD`` is mandatory
at the start of each object struct and defines a field called ``ob_base``
of type :c:type:`PyObject`, containing a pointer to a type object and a
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In other order: a reference count and a type object.

>>> "" + noddy.new_noddy()
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: cannot add type "noddy.Noddy" to string
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

TypeError: can only concatenate str (not "noddy.Noddy") to str

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ha, thank you.

This adds the type to the module dictionary. This allows us to create
:class:`Noddy` instances by calling the :class:`Noddy` class::

>>> import noddy
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is there needed a special directive for highlighting this code block as Python session rather of C code?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmm, you're right.

in C!

We want to make sure that the first and last names are initialized to empty
strings, so we provide a ``tp_new`` method::
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think that the original "a new method" means literally a new C function, not a method named "new" or something like. "provide a tp_new method" conflicts with a phrase following the code: "and install it in the tp_new member".

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

No, it was present in several places, and was really meaning "a tp_new method".

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In any case "a tp_new method" looks wrong and confusing to me.

.. we provide a tp_new method ... and install it in the tp_new member.

tp_new is not a method, it is a PyTypeObject member (or slot). I think the original text is a tiny bit more correct. Look at wordings for other slots below.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You're right. I will use the more generic "handler".

It is not required to define a ``tp_new`` member, and indeed many extension
types will simply reuse ``PyType_GenericNew`` as done in the first version
of the ``Noddy`` type above. In this case, we use the ``tp_new`` method
to first initialize the :attr:`first` and :attr:`last` attributes to
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

:c:member:`first` and :c:member:`last`

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmm, right. That won't make much of a difference since this thing isn't documented...

{NULL} /* Sentinel */
};

(note that we used the :const:`METH_NOARGS` flag to indicate that the method
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This hanging comment in parenthesis looks weird. What if move it up?

Now that we've defined the method, we need to create an array of method
definitions (note that we used the :const:`METH_NOARGS` flag to indicate that the method
is expecting no arguments other than *self*)::

Or keep it where it was in the original text?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmm, they look fine to me?

.tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE,

We rename :c:func:`PyInit_noddy` to :c:func:`PyInit_noddy2` and update the module
name in the :c:type:`PyModuleDef` struct.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

After changing the name of the module, update the full name of the class. They should be consistent.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes.

easily use the :class:`PyTypeObject` it needs. It can be difficult to share
these :class:`PyTypeObject` structures between extension modules.

In this example we will create a :class:`Shoddy` type that inherits from the
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think Noddy and Shoddy are pretty good names for examples. Let not kill jokes in the documentation for a programming language named after the comic group.

create the memory for the object with its :c:member:`~PyTypeObject.tp_alloc`,
but let the base class handle it by calling its own :c:member:`~PyTypeObject.tp_new`.

When filling out the :c:func:`PyTypeObject` for the :class:`Shoddy` type, you see
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Don't know what is more appropriate here, but perhaps not :c:func:.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmm, indeed :-)

but let the base class handle it by calling its own :c:member:`~PyTypeObject.tp_new`.

When filling out the :c:func:`PyTypeObject` for the :class:`Shoddy` type, you see
a slot for :c:func:`tp_base`. Due to cross-platform compiler issues, you can't
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

:c:member:


const char *tp_name; /* For printing */

The name of the type - as mentioned in the last section, this will appear in
The name of the type - as mentioned in the previous chapter, this will appear in
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Needed longer dash.

@@ -1035,7 +156,7 @@ example::
static PyObject *
newdatatype_repr(newdatatypeobject * obj)
{
return PyUnicode_FromFormat("Repr-ified_newdatatype{{size:\%d}}",
return PyUnicode_FromFormat("Repr-ified_newdatatype{{size:%d}}",
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ha-ha, TeX artifact!


Any :term:`iterator` object should implement both :c:member:`~PyTypeObject.tp_iter`
and :c:member:`~PyTypeObject.tp_iternext`. An iterator's
:c:member:`~PyTypeObject.tp_iter` handler should return a new reference
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe add a reference to PyObject_SelfIter?

static void
instance_dealloc(PyInstanceObject *inst)
{
/* Allocate temporaries if needed, but do not begin
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This comment is important. PyObject_ClearWeakRefs() should be called before calling any destructors.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why is that?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I suppose that this is because destructors can resurrect the deallocated object (increment the refcount) if there are weak references to it. Following call of PyObject_ClearWeakRefs() will fail with raising SystemError because the refcount is not 0.

Interestingly, but not all builtin types follow this rule. Seems there are bugs.

@pitrou
Copy link
Member Author

pitrou commented Apr 6, 2018

I think I addressed most important comments now (including the renaming to "custom" and "sublist"). There are a couple stylistic comments that I chose not to follow.

Copy link
Member

@serhiy-storchaka serhiy-storchaka left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What about the history? Doesn't mass renaming ruin it?


>>> "" + noddy.new_noddy()
.. code-block:: python
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

"pycon" may be better option.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I didn't know about that.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ok, it doesn't seem to make a difference compared to "python".

@pitrou
Copy link
Member Author

pitrou commented Apr 6, 2018

What about the history? Doesn't mass renaming ruin it?

git is able to detect renames. You may need to use specific command-line options to benefit from it (e.g. git log --follow <filename>).

@pitrou pitrou merged commit 1d80a56 into python:master Apr 7, 2018
@miss-islington
Copy link
Contributor

Thanks @pitrou for the PR 🌮🎉.. I'm working now to backport this PR to: 3.6, 3.7.
🐍🍒⛏🤖

@pitrou pitrou deleted the modernize-extending branch April 7, 2018 16:14
@bedevere-bot
Copy link

GH-6411 is a backport of this pull request to the 3.7 branch.

miss-islington pushed a commit to miss-islington/cpython that referenced this pull request Apr 7, 2018
* bpo-33201: Modernize "Extension types" doc
* Split tutorial and other topics
* Some small fixes
* Address some review comments
* Rename noddy* to custom* and shoddy to sublist
* Fix markup
(cherry picked from commit 1d80a56)

Co-authored-by: Antoine Pitrou <pitrou@free.fr>
@miss-islington
Copy link
Contributor

Sorry, @pitrou, I could not cleanly backport this to 3.6 due to a conflict.
Please backport using cherry_picker on command line.
cherry_picker 1d80a561734b9932961c546b0897405a3bfbf3e6 3.6

pitrou added a commit to pitrou/cpython that referenced this pull request Apr 7, 2018
* bpo-33201: Modernize "Extension types" doc
* Split tutorial and other topics
* Some small fixes
* Address some review comments
* Rename noddy* to custom* and shoddy to sublist
* Fix markup.
(cherry picked from commit 1d80a56)
pitrou added a commit to pitrou/cpython that referenced this pull request Apr 7, 2018
* bpo-33201: Modernize "Extension types" doc
* Split tutorial and other topics
* Some small fixes
* Address some review comments
* Rename noddy* to custom* and shoddy to sublist
* Fix markup.
(cherry picked from commit 1d80a56)
@bedevere-bot
Copy link

GH-6412 is a backport of this pull request to the 3.6 branch.

pitrou added a commit that referenced this pull request Apr 7, 2018
* bpo-33201: Modernize "Extension types" doc
* Split tutorial and other topics
* Some small fixes
* Address some review comments
* Rename noddy* to custom* and shoddy to sublist
* Fix markup
(cherry picked from commit 1d80a56)

Co-authored-by: Antoine Pitrou <pitrou@free.fr>
pitrou added a commit that referenced this pull request Apr 7, 2018
* bpo-33201: Modernize "Extension types" doc
* Split tutorial and other topics
* Some small fixes
* Address some review comments
* Rename noddy* to custom* and shoddy to sublist
* Fix markup.
(cherry picked from commit 1d80a56)
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

7 participants