mirror of
				https://github.com/jart/cosmopolitan.git
				synced 2025-10-26 03:00:57 +00:00 
			
		
		
		
	Actually Portable Python is now outperforming the Python binaries
that come bundled with Linux distros, at things like HTTP serving.
You can now have a fully featured Python install in just one .com
file that runs on six operating systems and is about 10mb in size.
With tuning, the tiniest is ~1mb. We've got most of the libraries
working, including pysqlite, and the repl now feels very pleasant.
The things you can't do quite yet are: threads and shared objects
but that can happen in the future, if the community falls in love
with this project and wants to see it developed further. Changes:
- Add siginterrupt()
- Add sqlite3 to Python
- Add issymlink() helper
- Make GetZipCdir() faster
- Add tgamma() and finite()
- Add legacy function lutimes()
- Add readlink() and realpath()
- Use heap allocations when appropriate
- Reorganize Python into two-stage build
- Save Lua / Python shell history to dotfile
- Integrate Python Lib embedding into linkage
- Make isregularfile() and isdirectory() go faster
- Make Python shell auto-completion work perfectly
- Make crash reports work better if changed directory
- Fix Python+NT open() / access() flag overflow error
- Disable Python tests relating to \N{LONG NAME} syntax
- Have Python REPL copyright() show all notice embeddings
The biggest technical challenge at the moment is working around
when Python tries to be too clever about filenames.
		
	
			
		
			
				
	
	
		
			59 lines
		
	
	
	
		
			2.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			59 lines
		
	
	
	
		
			2.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:4;tab-width:8;coding:utf-8 -*-│
 | |
| │vi: set net ft=c ts=4 sts=4 sw=4 fenc=utf-8                                :vi│
 | |
| ╞══════════════════════════════════════════════════════════════════════════════╡
 | |
| │ Python 3                                                                     │
 | |
| │ https://docs.python.org/3/license.html                                       │
 | |
| ╚─────────────────────────────────────────────────────────────────────────────*/
 | |
| #include "third_party/python/Include/abstract.h"
 | |
| #include "third_party/python/Include/bytesobject.h"
 | |
| #include "third_party/python/Include/object.h"
 | |
| #include "third_party/python/Include/pyerrors.h"
 | |
| #include "third_party/python/Include/unicodeobject.h"
 | |
| /* clang-format off */
 | |
| 
 | |
| /*
 | |
|     Return the file system path representation of the object.
 | |
| 
 | |
|     If the object is str or bytes, then allow it to pass through with
 | |
|     an incremented refcount. If the object defines __fspath__(), then
 | |
|     return the result of that method. All other types raise a TypeError.
 | |
| */
 | |
| PyObject *
 | |
| PyOS_FSPath(PyObject *path)
 | |
| {
 | |
|     /* For error message reasons, this function is manually inlined in
 | |
|        path_converter(). */
 | |
|     _Py_IDENTIFIER(__fspath__);
 | |
|     PyObject *func = NULL;
 | |
|     PyObject *path_repr = NULL;
 | |
| 
 | |
|     if (PyUnicode_Check(path) || PyBytes_Check(path)) {
 | |
|         Py_INCREF(path);
 | |
|         return path;
 | |
|     }
 | |
| 
 | |
|     func = _PyObject_LookupSpecial(path, &PyId___fspath__);
 | |
|     if (NULL == func) {
 | |
|         return PyErr_Format(PyExc_TypeError,
 | |
|                             "expected str, bytes or os.PathLike object, "
 | |
|                             "not %.200s",
 | |
|                             Py_TYPE(path)->tp_name);
 | |
|     }
 | |
| 
 | |
|     path_repr = PyObject_CallFunctionObjArgs(func, NULL);
 | |
|     Py_DECREF(func);
 | |
|     if (NULL == path_repr) {
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     if (!(PyUnicode_Check(path_repr) || PyBytes_Check(path_repr))) {
 | |
|         PyErr_Format(PyExc_TypeError,
 | |
|                      "expected %.200s.__fspath__() to return str or bytes, "
 | |
|                      "not %.200s", Py_TYPE(path)->tp_name,
 | |
|                      Py_TYPE(path_repr)->tp_name);
 | |
|         Py_DECREF(path_repr);
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     return path_repr;
 | |
| }
 |