mirror of
				https://github.com/python/cpython.git
				synced 2025-10-24 18:33:49 +00:00 
			
		
		
		
	 93b7ed7c6b
			
		
	
	
		93b7ed7c6b
		
			
		
	
	
	
	
		
			
			SimpleNamespace({'a': 1, 'b': 2}) and SimpleNamespace([('a', 1), ('b', 2)])
are now the same as SimpleNamespace(a=1, b=2).
		
	
			
		
			
				
	
	
		
			311 lines
		
	
	
	
		
			8.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			311 lines
		
	
	
	
		
			8.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // namespace object implementation
 | |
| 
 | |
| #include "Python.h"
 | |
| #include "pycore_modsupport.h"    // _PyArg_NoPositional()
 | |
| #include "pycore_namespace.h"     // _PyNamespace_Type
 | |
| 
 | |
| #include <stddef.h>               // offsetof()
 | |
| 
 | |
| 
 | |
| typedef struct {
 | |
|     PyObject_HEAD
 | |
|     PyObject *ns_dict;
 | |
| } _PyNamespaceObject;
 | |
| 
 | |
| 
 | |
| static PyMemberDef namespace_members[] = {
 | |
|     {"__dict__", _Py_T_OBJECT, offsetof(_PyNamespaceObject, ns_dict), Py_READONLY},
 | |
|     {NULL}
 | |
| };
 | |
| 
 | |
| 
 | |
| // Methods
 | |
| 
 | |
| static PyObject *
 | |
| namespace_new(PyTypeObject *type, PyObject *args, PyObject *kwds)
 | |
| {
 | |
|     PyObject *self;
 | |
| 
 | |
|     assert(type != NULL && type->tp_alloc != NULL);
 | |
|     self = type->tp_alloc(type, 0);
 | |
|     if (self != NULL) {
 | |
|         _PyNamespaceObject *ns = (_PyNamespaceObject *)self;
 | |
|         ns->ns_dict = PyDict_New();
 | |
|         if (ns->ns_dict == NULL) {
 | |
|             Py_DECREF(ns);
 | |
|             return NULL;
 | |
|         }
 | |
|     }
 | |
|     return self;
 | |
| }
 | |
| 
 | |
| 
 | |
| static int
 | |
| namespace_init(_PyNamespaceObject *ns, PyObject *args, PyObject *kwds)
 | |
| {
 | |
|     PyObject *arg = NULL;
 | |
|     if (!PyArg_UnpackTuple(args, _PyType_Name(Py_TYPE(ns)), 0, 1, &arg)) {
 | |
|         return -1;
 | |
|     }
 | |
|     if (arg != NULL) {
 | |
|         PyObject *dict;
 | |
|         if (PyDict_CheckExact(arg)) {
 | |
|             dict = Py_NewRef(arg);
 | |
|         }
 | |
|         else {
 | |
|             dict = PyObject_CallOneArg((PyObject *)&PyDict_Type, arg);
 | |
|             if (dict == NULL) {
 | |
|                 return -1;
 | |
|             }
 | |
|         }
 | |
|         int err = (!PyArg_ValidateKeywordArguments(dict) ||
 | |
|                    PyDict_Update(ns->ns_dict, dict) < 0);
 | |
|         Py_DECREF(dict);
 | |
|         if (err) {
 | |
|             return -1;
 | |
|         }
 | |
|     }
 | |
|     if (kwds == NULL) {
 | |
|         return 0;
 | |
|     }
 | |
|     if (!PyArg_ValidateKeywordArguments(kwds)) {
 | |
|         return -1;
 | |
|     }
 | |
|     return PyDict_Update(ns->ns_dict, kwds);
 | |
| }
 | |
| 
 | |
| 
 | |
| static void
 | |
| namespace_dealloc(_PyNamespaceObject *ns)
 | |
| {
 | |
|     PyObject_GC_UnTrack(ns);
 | |
|     Py_CLEAR(ns->ns_dict);
 | |
|     Py_TYPE(ns)->tp_free((PyObject *)ns);
 | |
| }
 | |
| 
 | |
| 
 | |
| static PyObject *
 | |
| namespace_repr(PyObject *ns)
 | |
| {
 | |
|     int i, loop_error = 0;
 | |
|     PyObject *pairs = NULL, *d = NULL, *keys = NULL, *keys_iter = NULL;
 | |
|     PyObject *key;
 | |
|     PyObject *separator, *pairsrepr, *repr = NULL;
 | |
|     const char * name;
 | |
| 
 | |
|     name = Py_IS_TYPE(ns, &_PyNamespace_Type) ? "namespace"
 | |
|                                                : Py_TYPE(ns)->tp_name;
 | |
| 
 | |
|     i = Py_ReprEnter(ns);
 | |
|     if (i != 0) {
 | |
|         return i > 0 ? PyUnicode_FromFormat("%s(...)", name) : NULL;
 | |
|     }
 | |
| 
 | |
|     pairs = PyList_New(0);
 | |
|     if (pairs == NULL)
 | |
|         goto error;
 | |
| 
 | |
|     assert(((_PyNamespaceObject *)ns)->ns_dict != NULL);
 | |
|     d = Py_NewRef(((_PyNamespaceObject *)ns)->ns_dict);
 | |
| 
 | |
|     keys = PyDict_Keys(d);
 | |
|     if (keys == NULL)
 | |
|         goto error;
 | |
| 
 | |
|     keys_iter = PyObject_GetIter(keys);
 | |
|     if (keys_iter == NULL)
 | |
|         goto error;
 | |
| 
 | |
|     while ((key = PyIter_Next(keys_iter)) != NULL) {
 | |
|         if (PyUnicode_Check(key) && PyUnicode_GET_LENGTH(key) > 0) {
 | |
|             PyObject *value, *item;
 | |
| 
 | |
|             value = PyDict_GetItemWithError(d, key);
 | |
|             if (value != NULL) {
 | |
|                 item = PyUnicode_FromFormat("%U=%R", key, value);
 | |
|                 if (item == NULL) {
 | |
|                     loop_error = 1;
 | |
|                 }
 | |
|                 else {
 | |
|                     loop_error = PyList_Append(pairs, item);
 | |
|                     Py_DECREF(item);
 | |
|                 }
 | |
|             }
 | |
|             else if (PyErr_Occurred()) {
 | |
|                 loop_error = 1;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         Py_DECREF(key);
 | |
|         if (loop_error)
 | |
|             goto error;
 | |
|     }
 | |
| 
 | |
|     separator = PyUnicode_FromString(", ");
 | |
|     if (separator == NULL)
 | |
|         goto error;
 | |
| 
 | |
|     pairsrepr = PyUnicode_Join(separator, pairs);
 | |
|     Py_DECREF(separator);
 | |
|     if (pairsrepr == NULL)
 | |
|         goto error;
 | |
| 
 | |
|     repr = PyUnicode_FromFormat("%s(%S)", name, pairsrepr);
 | |
|     Py_DECREF(pairsrepr);
 | |
| 
 | |
| error:
 | |
|     Py_XDECREF(pairs);
 | |
|     Py_XDECREF(d);
 | |
|     Py_XDECREF(keys);
 | |
|     Py_XDECREF(keys_iter);
 | |
|     Py_ReprLeave(ns);
 | |
| 
 | |
|     return repr;
 | |
| }
 | |
| 
 | |
| 
 | |
| static int
 | |
| namespace_traverse(_PyNamespaceObject *ns, visitproc visit, void *arg)
 | |
| {
 | |
|     Py_VISIT(ns->ns_dict);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| static int
 | |
| namespace_clear(_PyNamespaceObject *ns)
 | |
| {
 | |
|     Py_CLEAR(ns->ns_dict);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| static PyObject *
 | |
| namespace_richcompare(PyObject *self, PyObject *other, int op)
 | |
| {
 | |
|     if (PyObject_TypeCheck(self, &_PyNamespace_Type) &&
 | |
|         PyObject_TypeCheck(other, &_PyNamespace_Type))
 | |
|         return PyObject_RichCompare(((_PyNamespaceObject *)self)->ns_dict,
 | |
|                                    ((_PyNamespaceObject *)other)->ns_dict, op);
 | |
|     Py_RETURN_NOTIMPLEMENTED;
 | |
| }
 | |
| 
 | |
| 
 | |
| PyDoc_STRVAR(namespace_reduce__doc__, "Return state information for pickling");
 | |
| 
 | |
| static PyObject *
 | |
| namespace_reduce(_PyNamespaceObject *ns, PyObject *Py_UNUSED(ignored))
 | |
| {
 | |
|     PyObject *result, *args = PyTuple_New(0);
 | |
| 
 | |
|     if (!args)
 | |
|         return NULL;
 | |
| 
 | |
|     result = PyTuple_Pack(3, (PyObject *)Py_TYPE(ns), args, ns->ns_dict);
 | |
|     Py_DECREF(args);
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| 
 | |
| static PyObject *
 | |
| namespace_replace(PyObject *self, PyObject *args, PyObject *kwargs)
 | |
| {
 | |
|     if (!_PyArg_NoPositional("__replace__", args)) {
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     PyObject *result = PyObject_CallNoArgs((PyObject *)Py_TYPE(self));
 | |
|     if (!result) {
 | |
|         return NULL;
 | |
|     }
 | |
|     if (PyDict_Update(((_PyNamespaceObject*)result)->ns_dict,
 | |
|                       ((_PyNamespaceObject*)self)->ns_dict) < 0)
 | |
|     {
 | |
|         Py_DECREF(result);
 | |
|         return NULL;
 | |
|     }
 | |
|     if (kwargs) {
 | |
|         if (PyDict_Update(((_PyNamespaceObject*)result)->ns_dict, kwargs) < 0) {
 | |
|             Py_DECREF(result);
 | |
|             return NULL;
 | |
|         }
 | |
|     }
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| 
 | |
| static PyMethodDef namespace_methods[] = {
 | |
|     {"__reduce__", (PyCFunction)namespace_reduce, METH_NOARGS,
 | |
|      namespace_reduce__doc__},
 | |
|     {"__replace__", _PyCFunction_CAST(namespace_replace), METH_VARARGS|METH_KEYWORDS,
 | |
|      PyDoc_STR("__replace__($self, /, **changes)\n--\n\n"
 | |
|         "Return a copy of the namespace object with new values for the specified attributes.")},
 | |
|     {NULL,         NULL}  // sentinel
 | |
| };
 | |
| 
 | |
| 
 | |
| PyDoc_STRVAR(namespace_doc,
 | |
| "SimpleNamespace(mapping_or_iterable=(), /, **kwargs)\n\
 | |
| --\n\n\
 | |
| A simple attribute-based namespace.");
 | |
| 
 | |
| PyTypeObject _PyNamespace_Type = {
 | |
|     PyVarObject_HEAD_INIT(&PyType_Type, 0)
 | |
|     "types.SimpleNamespace",                    /* tp_name */
 | |
|     sizeof(_PyNamespaceObject),                 /* tp_basicsize */
 | |
|     0,                                          /* tp_itemsize */
 | |
|     (destructor)namespace_dealloc,              /* tp_dealloc */
 | |
|     0,                                          /* tp_vectorcall_offset */
 | |
|     0,                                          /* tp_getattr */
 | |
|     0,                                          /* tp_setattr */
 | |
|     0,                                          /* tp_as_async */
 | |
|     (reprfunc)namespace_repr,                   /* tp_repr */
 | |
|     0,                                          /* tp_as_number */
 | |
|     0,                                          /* tp_as_sequence */
 | |
|     0,                                          /* tp_as_mapping */
 | |
|     0,                                          /* tp_hash */
 | |
|     0,                                          /* tp_call */
 | |
|     0,                                          /* tp_str */
 | |
|     PyObject_GenericGetAttr,                    /* tp_getattro */
 | |
|     PyObject_GenericSetAttr,                    /* tp_setattro */
 | |
|     0,                                          /* tp_as_buffer */
 | |
|     Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC |
 | |
|         Py_TPFLAGS_BASETYPE,                    /* tp_flags */
 | |
|     namespace_doc,                              /* tp_doc */
 | |
|     (traverseproc)namespace_traverse,           /* tp_traverse */
 | |
|     (inquiry)namespace_clear,                   /* tp_clear */
 | |
|     namespace_richcompare,                      /* tp_richcompare */
 | |
|     0,                                          /* tp_weaklistoffset */
 | |
|     0,                                          /* tp_iter */
 | |
|     0,                                          /* tp_iternext */
 | |
|     namespace_methods,                          /* tp_methods */
 | |
|     namespace_members,                          /* tp_members */
 | |
|     0,                                          /* tp_getset */
 | |
|     0,                                          /* tp_base */
 | |
|     0,                                          /* tp_dict */
 | |
|     0,                                          /* tp_descr_get */
 | |
|     0,                                          /* tp_descr_set */
 | |
|     offsetof(_PyNamespaceObject, ns_dict),      /* tp_dictoffset */
 | |
|     (initproc)namespace_init,                   /* tp_init */
 | |
|     PyType_GenericAlloc,                        /* tp_alloc */
 | |
|     (newfunc)namespace_new,                     /* tp_new */
 | |
|     PyObject_GC_Del,                            /* tp_free */
 | |
| };
 | |
| 
 | |
| 
 | |
| PyObject *
 | |
| _PyNamespace_New(PyObject *kwds)
 | |
| {
 | |
|     PyObject *ns = namespace_new(&_PyNamespace_Type, NULL, NULL);
 | |
|     if (ns == NULL)
 | |
|         return NULL;
 | |
| 
 | |
|     if (kwds == NULL)
 | |
|         return ns;
 | |
|     if (PyDict_Update(((_PyNamespaceObject *)ns)->ns_dict, kwds) != 0) {
 | |
|         Py_DECREF(ns);
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     return (PyObject *)ns;
 | |
| }
 |