PostgreSQL Source Code git master
plpy_util.c
Go to the documentation of this file.
1/*
2 * utility functions
3 *
4 * src/pl/plpython/plpy_util.c
5 */
6
7#include "postgres.h"
8
9#include "mb/pg_wchar.h"
10#include "plpy_elog.h"
11#include "plpy_util.h"
12#include "plpython.h"
13
14/*
15 * Convert a Python unicode object to a Python string/bytes object in
16 * PostgreSQL server encoding. Reference ownership is passed to the
17 * caller.
18 */
19PyObject *
20PLyUnicode_Bytes(PyObject *unicode)
21{
22 PyObject *bytes,
23 *rv;
24 char *utf8string,
25 *encoded;
26
27 /* First encode the Python unicode object with UTF-8. */
28 bytes = PyUnicode_AsUTF8String(unicode);
29 if (bytes == NULL)
30 PLy_elog(ERROR, "could not convert Python Unicode object to bytes");
31
32 utf8string = PyBytes_AsString(bytes);
33 if (utf8string == NULL)
34 {
35 Py_DECREF(bytes);
36 PLy_elog(ERROR, "could not extract bytes from encoded string");
37 }
38
39 /*
40 * Then convert to server encoding if necessary.
41 *
42 * PyUnicode_AsEncodedString could be used to encode the object directly
43 * in the server encoding, but Python doesn't support all the encodings
44 * that PostgreSQL does (EUC_TW and MULE_INTERNAL). UTF-8 is used as an
45 * intermediary in PLyUnicode_FromString as well.
46 */
48 {
49 PG_TRY();
50 {
51 encoded = pg_any_to_server(utf8string,
52 strlen(utf8string),
53 PG_UTF8);
54 }
55 PG_CATCH();
56 {
57 Py_DECREF(bytes);
59 }
60 PG_END_TRY();
61 }
62 else
63 encoded = utf8string;
64
65 /* finally, build a bytes object in the server encoding */
66 rv = PyBytes_FromStringAndSize(encoded, strlen(encoded));
67
68 /* if pg_any_to_server allocated memory, free it now */
69 if (utf8string != encoded)
70 pfree(encoded);
71
72 Py_DECREF(bytes);
73 return rv;
74}
75
76/*
77 * Convert a Python unicode object to a C string in PostgreSQL server
78 * encoding. No Python object reference is passed out of this
79 * function. The result is palloc'ed.
80 */
81char *
82PLyUnicode_AsString(PyObject *unicode)
83{
84 PyObject *o = PLyUnicode_Bytes(unicode);
85 char *rv = pstrdup(PyBytes_AsString(o));
86
87 Py_XDECREF(o);
88 return rv;
89}
90
91/*
92 * Convert a C string in the PostgreSQL server encoding to a Python
93 * unicode object. Reference ownership is passed to the caller.
94 */
95PyObject *
96PLyUnicode_FromStringAndSize(const char *s, Py_ssize_t size)
97{
98 char *utf8string;
99 PyObject *o;
100
101 utf8string = pg_server_to_any(s, size, PG_UTF8);
102
103 if (utf8string == s)
104 {
105 o = PyUnicode_FromStringAndSize(s, size);
106 }
107 else
108 {
109 o = PyUnicode_FromString(utf8string);
110 pfree(utf8string);
111 }
112
113 return o;
114}
115
116PyObject *
118{
119 return PLyUnicode_FromStringAndSize(s, strlen(s));
120}
#define PG_RE_THROW()
Definition: elog.h:412
#define PG_TRY(...)
Definition: elog.h:371
#define PG_END_TRY(...)
Definition: elog.h:396
#define ERROR
Definition: elog.h:39
#define PG_CATCH(...)
Definition: elog.h:381
#define PLy_elog
int GetDatabaseEncoding(void)
Definition: mbutils.c:1261
char * pg_any_to_server(const char *s, int len, int encoding)
Definition: mbutils.c:676
char * pg_server_to_any(const char *s, int len, int encoding)
Definition: mbutils.c:749
char * pstrdup(const char *in)
Definition: mcxt.c:1696
void pfree(void *pointer)
Definition: mcxt.c:1521
@ PG_UTF8
Definition: pg_wchar.h:232
char * PLyUnicode_AsString(PyObject *unicode)
Definition: plpy_util.c:82
PyObject * PLyUnicode_Bytes(PyObject *unicode)
Definition: plpy_util.c:20
PyObject * PLyUnicode_FromStringAndSize(const char *s, Py_ssize_t size)
Definition: plpy_util.c:96
PyObject * PLyUnicode_FromString(const char *s)
Definition: plpy_util.c:117
static pg_noinline void Size size
Definition: slab.c:607