Use to_str/to_bytes, replacing old versions
This commit is contained in:
commit
c3ebd8d251
24 changed files with 130 additions and 121 deletions
|
|
@ -87,6 +87,12 @@ Web/Django standard initiative (@strikaco)
|
||||||
|
|
||||||
- Swap argument order of `evennia.set_trace` to `set_trace(term_size=(140, 40), debugger='auto')`
|
- Swap argument order of `evennia.set_trace` to `set_trace(term_size=(140, 40), debugger='auto')`
|
||||||
since the size is more likely to be changed on the command line.
|
since the size is more likely to be changed on the command line.
|
||||||
|
- `utils.to_str(text, session=None)` now acts as the old `utils.to_unicode` (which was removed).
|
||||||
|
This converts to the str() type (not to a byte-string as in Evennia 0.8), trying different
|
||||||
|
encodings. This function will also force-convert any object passed to it into a string (so
|
||||||
|
`force_string` flag was removed and assumed always set).
|
||||||
|
- `utils.to_bytes(text, session=None)` replaces the old `utils.to_str()` functionality and converts
|
||||||
|
str to bytes.
|
||||||
|
|
||||||
|
|
||||||
## Evennia 0.8 (2018)
|
## Evennia 0.8 (2018)
|
||||||
|
|
|
||||||
|
|
@ -814,13 +814,7 @@ class DefaultAccount(with_metaclass(TypeclassBase, AccountDB)):
|
||||||
kwargs["options"] = options
|
kwargs["options"] = options
|
||||||
|
|
||||||
if text is not None:
|
if text is not None:
|
||||||
if not (isinstance(text, str) or isinstance(text, tuple)):
|
kwargs['text'] = to_str(text)
|
||||||
# sanitize text before sending across the wire
|
|
||||||
try:
|
|
||||||
text = to_str(text, force_string=True)
|
|
||||||
except Exception:
|
|
||||||
text = repr(text)
|
|
||||||
kwargs['text'] = text
|
|
||||||
|
|
||||||
# session relay
|
# session relay
|
||||||
sessions = make_iter(session) if session else self.sessions.all()
|
sessions = make_iter(session) if session else self.sessions.all()
|
||||||
|
|
|
||||||
|
|
@ -2077,7 +2077,7 @@ class CmdExamine(ObjManipCommand):
|
||||||
"""
|
"""
|
||||||
if crop:
|
if crop:
|
||||||
if not isinstance(value, str):
|
if not isinstance(value, str):
|
||||||
value = utils.to_str(value, force_string=True)
|
value = utils.to_str(value)
|
||||||
value = utils.crop(value)
|
value = utils.crop(value)
|
||||||
|
|
||||||
string = "\n %s = %s" % (attr, value)
|
string = "\n %s = %s" % (attr, value)
|
||||||
|
|
|
||||||
|
|
@ -107,7 +107,7 @@ class CommandTest(EvenniaTest):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
# clean out evtable sugar. We only operate on text-type
|
# clean out evtable sugar. We only operate on text-type
|
||||||
stored_msg = [args[0] if args and args[0] else kwargs.get("text", utils.to_str(kwargs, force_string=True))
|
stored_msg = [args[0] if args and args[0] else kwargs.get("text", utils.to_str(kwargs))
|
||||||
for name, args, kwargs in receiver.msg.mock_calls]
|
for name, args, kwargs in receiver.msg.mock_calls]
|
||||||
# Get the first element of a tuple if msg received a tuple instead of a string
|
# Get the first element of a tuple if msg received a tuple instead of a string
|
||||||
stored_msg = [smsg[0] if isinstance(smsg, tuple) else smsg for smsg in stored_msg]
|
stored_msg = [smsg[0] if isinstance(smsg, tuple) else smsg for smsg in stored_msg]
|
||||||
|
|
|
||||||
|
|
@ -486,7 +486,7 @@ class TestDefaultCallbacks(CommandTest):
|
||||||
try:
|
try:
|
||||||
self.char2.msg = Mock()
|
self.char2.msg = Mock()
|
||||||
self.call(ExitCommand(), "", obj=self.exit)
|
self.call(ExitCommand(), "", obj=self.exit)
|
||||||
stored_msg = [args[0] if args and args[0] else kwargs.get("text", utils.to_str(kwargs, force_string=True))
|
stored_msg = [args[0] if args and args[0] else kwargs.get("text", utils.to_str(kwargs))
|
||||||
for name, args, kwargs in self.char2.msg.mock_calls]
|
for name, args, kwargs in self.char2.msg.mock_calls]
|
||||||
# Get the first element of a tuple if msg received a tuple instead of a string
|
# Get the first element of a tuple if msg received a tuple instead of a string
|
||||||
stored_msg = [smsg[0] if isinstance(smsg, tuple) else smsg for smsg in stored_msg]
|
stored_msg = [smsg[0] if isinstance(smsg, tuple) else smsg for smsg in stored_msg]
|
||||||
|
|
@ -507,7 +507,7 @@ class TestDefaultCallbacks(CommandTest):
|
||||||
try:
|
try:
|
||||||
self.char2.msg = Mock()
|
self.char2.msg = Mock()
|
||||||
self.call(ExitCommand(), "", obj=back)
|
self.call(ExitCommand(), "", obj=back)
|
||||||
stored_msg = [args[0] if args and args[0] else kwargs.get("text", utils.to_str(kwargs, force_string=True))
|
stored_msg = [args[0] if args and args[0] else kwargs.get("text", utils.to_str(kwargs))
|
||||||
for name, args, kwargs in self.char2.msg.mock_calls]
|
for name, args, kwargs in self.char2.msg.mock_calls]
|
||||||
# Get the first element of a tuple if msg received a tuple instead of a string
|
# Get the first element of a tuple if msg received a tuple instead of a string
|
||||||
stored_msg = [smsg[0] if isinstance(smsg, tuple) else smsg for smsg in stored_msg]
|
stored_msg = [smsg[0] if isinstance(smsg, tuple) else smsg for smsg in stored_msg]
|
||||||
|
|
|
||||||
|
|
@ -582,7 +582,7 @@ class DefaultObject(with_metaclass(TypeclassBase, ObjectDB)):
|
||||||
if not (isinstance(text, str) or isinstance(text, tuple)):
|
if not (isinstance(text, str) or isinstance(text, tuple)):
|
||||||
# sanitize text before sending across the wire
|
# sanitize text before sending across the wire
|
||||||
try:
|
try:
|
||||||
text = to_str(text, force_string=True)
|
text = to_str(text)
|
||||||
except Exception:
|
except Exception:
|
||||||
text = repr(text)
|
text = repr(text)
|
||||||
kwargs['text'] = text
|
kwargs['text'] = text
|
||||||
|
|
|
||||||
|
|
@ -1039,7 +1039,7 @@ def node_aliases(caller):
|
||||||
|
|
||||||
def _caller_attrs(caller):
|
def _caller_attrs(caller):
|
||||||
prototype = _get_menu_prototype(caller)
|
prototype = _get_menu_prototype(caller)
|
||||||
attrs = ["{}={}".format(tup[0], utils.crop(utils.to_str(tup[1], force_string=True), width=10))
|
attrs = ["{}={}".format(tup[0], utils.crop(utils.to_str(tup[1]), width=10))
|
||||||
for tup in prototype.get("attrs", [])]
|
for tup in prototype.get("attrs", [])]
|
||||||
return attrs
|
return attrs
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -569,7 +569,7 @@ def protfunc_parser(value, available_functions=None, testing=False, stacktrace=F
|
||||||
value = value.dbref
|
value = value.dbref
|
||||||
except AttributeError:
|
except AttributeError:
|
||||||
pass
|
pass
|
||||||
value = to_str(value, force_string=True)
|
value = to_str(value)
|
||||||
|
|
||||||
available_functions = PROT_FUNCS if available_functions is None else available_functions
|
available_functions = PROT_FUNCS if available_functions is None else available_functions
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -145,7 +145,7 @@ class ScriptDB(TypedObject):
|
||||||
pass
|
pass
|
||||||
if isinstance(value, (str, int)):
|
if isinstance(value, (str, int)):
|
||||||
from evennia.objects.models import ObjectDB
|
from evennia.objects.models import ObjectDB
|
||||||
value = to_str(value, force_string=True)
|
value = to_str(value)
|
||||||
if (value.isdigit() or value.startswith("#")):
|
if (value.isdigit() or value.startswith("#")):
|
||||||
dbid = dbref(value, reqhash=False)
|
dbid = dbref(value, reqhash=False)
|
||||||
if dbid:
|
if dbid:
|
||||||
|
|
|
||||||
|
|
@ -78,7 +78,7 @@ class ServerConfig(WeakSharedMemoryModel):
|
||||||
#@property
|
#@property
|
||||||
def __value_get(self):
|
def __value_get(self):
|
||||||
"Getter. Allows for value = self.value"
|
"Getter. Allows for value = self.value"
|
||||||
return pickle.loads(self.db_value)
|
return pickle.loads(utils.to_bytes(self.db_value))
|
||||||
|
|
||||||
#@value.setter
|
#@value.setter
|
||||||
def __value_set(self, value):
|
def __value_set(self, value):
|
||||||
|
|
|
||||||
|
|
@ -290,7 +290,7 @@ class SshProtocol(Manhole, session.Session):
|
||||||
text = args[0] if args else ""
|
text = args[0] if args else ""
|
||||||
if text is None:
|
if text is None:
|
||||||
return
|
return
|
||||||
text = to_str(text, force_string=True)
|
text = to_str(text)
|
||||||
|
|
||||||
# handle arguments
|
# handle arguments
|
||||||
options = kwargs.get("options", {})
|
options = kwargs.get("options", {})
|
||||||
|
|
|
||||||
|
|
@ -18,7 +18,7 @@ from evennia.server.portal import ttype, mssp, telnet_oob, naws, suppress_ga
|
||||||
from evennia.server.portal.mccp import Mccp, mccp_compress, MCCP
|
from evennia.server.portal.mccp import Mccp, mccp_compress, MCCP
|
||||||
from evennia.server.portal.mxp import Mxp, mxp_parse
|
from evennia.server.portal.mxp import Mxp, mxp_parse
|
||||||
from evennia.utils import ansi
|
from evennia.utils import ansi
|
||||||
from evennia.utils.utils import to_str
|
from evennia.utils.utils import to_bytes
|
||||||
|
|
||||||
_RE_N = re.compile(r"\|n$")
|
_RE_N = re.compile(r"\|n$")
|
||||||
_RE_LEND = re.compile(br"\n$|\r$|\r\n$|\r\x00$|", re.MULTILINE)
|
_RE_LEND = re.compile(br"\n$|\r$|\r\n$|\r\x00$|", re.MULTILINE)
|
||||||
|
|
@ -243,7 +243,7 @@ class TelnetProtocol(Telnet, StatefulTelnetProtocol, Session):
|
||||||
line (str): Line to send.
|
line (str): Line to send.
|
||||||
|
|
||||||
"""
|
"""
|
||||||
line = self.encode_output(line)
|
line = to_bytes(line, self)
|
||||||
# escape IAC in line mode, and correctly add \r\n (the TELNET end-of-line)
|
# escape IAC in line mode, and correctly add \r\n (the TELNET end-of-line)
|
||||||
line = line.replace(IAC, IAC + IAC)
|
line = line.replace(IAC, IAC + IAC)
|
||||||
line = line.replace(b'\n', b'\r\n')
|
line = line.replace(b'\n', b'\r\n')
|
||||||
|
|
@ -316,7 +316,6 @@ class TelnetProtocol(Telnet, StatefulTelnetProtocol, Session):
|
||||||
text = args[0] if args else ""
|
text = args[0] if args else ""
|
||||||
if text is None:
|
if text is None:
|
||||||
return
|
return
|
||||||
text = to_str(text, force_string=True)
|
|
||||||
|
|
||||||
# handle arguments
|
# handle arguments
|
||||||
options = kwargs.get("options", {})
|
options = kwargs.get("options", {})
|
||||||
|
|
@ -343,7 +342,7 @@ class TelnetProtocol(Telnet, StatefulTelnetProtocol, Session):
|
||||||
strip_ansi=nocolor, xterm256=xterm256)
|
strip_ansi=nocolor, xterm256=xterm256)
|
||||||
if mxp:
|
if mxp:
|
||||||
prompt = mxp_parse(prompt)
|
prompt = mxp_parse(prompt)
|
||||||
prompt = self.encode_output(prompt)
|
prompt = to_bytes(prompt, self)
|
||||||
prompt = prompt.replace(IAC, IAC + IAC).replace(b'\n', b'\r\n')
|
prompt = prompt.replace(IAC, IAC + IAC).replace(b'\n', b'\r\n')
|
||||||
prompt += IAC + GA
|
prompt += IAC + GA
|
||||||
self.transport.write(mccp_compress(self, prompt))
|
self.transport.write(mccp_compress(self, prompt))
|
||||||
|
|
|
||||||
|
|
@ -28,7 +28,7 @@ header where applicable.
|
||||||
from builtins import object
|
from builtins import object
|
||||||
import re
|
import re
|
||||||
import json
|
import json
|
||||||
from evennia.utils.utils import to_str, is_iter
|
from evennia.utils.utils import is_iter
|
||||||
|
|
||||||
# MSDP-relevant telnet cmd/opt-codes
|
# MSDP-relevant telnet cmd/opt-codes
|
||||||
MSDP = b'\x45'
|
MSDP = b'\x45'
|
||||||
|
|
@ -46,11 +46,6 @@ GMCP = b'\xc9'
|
||||||
from twisted.conch.telnet import IAC, SB, SE
|
from twisted.conch.telnet import IAC, SB, SE
|
||||||
|
|
||||||
|
|
||||||
def force_str(inp):
|
|
||||||
"""Helper to shorten code"""
|
|
||||||
return to_str(inp, force_string=True)
|
|
||||||
|
|
||||||
|
|
||||||
# pre-compiled regexes
|
# pre-compiled regexes
|
||||||
# returns 2-tuple
|
# returns 2-tuple
|
||||||
msdp_regex_table = re.compile(br"%s\s*(\w*?)\s*%s\s*%s(.*?)%s"
|
msdp_regex_table = re.compile(br"%s\s*(\w*?)\s*%s\s*%s(.*?)%s"
|
||||||
|
|
|
||||||
|
|
@ -195,7 +195,6 @@ class WebSocketClient(WebSocketServerProtocol, Session):
|
||||||
return
|
return
|
||||||
|
|
||||||
flags = self.protocol_flags
|
flags = self.protocol_flags
|
||||||
text = to_str(text, force_string=True)
|
|
||||||
|
|
||||||
options = kwargs.pop("options", {})
|
options = kwargs.pop("options", {})
|
||||||
raw = options.get("raw", flags.get("RAW", False))
|
raw = options.get("raw", flags.get("RAW", False))
|
||||||
|
|
|
||||||
|
|
@ -369,7 +369,7 @@ class AjaxWebClientSession(session.Session):
|
||||||
return
|
return
|
||||||
|
|
||||||
flags = self.protocol_flags
|
flags = self.protocol_flags
|
||||||
text = utils.to_str(text, force_string=True)
|
text = utils.to_str(text)
|
||||||
|
|
||||||
options = kwargs.pop("options", {})
|
options = kwargs.pop("options", {})
|
||||||
raw = options.get("raw", flags.get("RAW", False))
|
raw = options.get("raw", flags.get("RAW", False))
|
||||||
|
|
|
||||||
|
|
@ -4,12 +4,14 @@ This module defines a generic session class. All connection instances
|
||||||
|
|
||||||
"""
|
"""
|
||||||
from builtins import object
|
from builtins import object
|
||||||
|
from django.conf import settings
|
||||||
|
|
||||||
import time
|
import time
|
||||||
|
|
||||||
#------------------------------------------------------------
|
# ------------------------------------------------------------
|
||||||
# Server Session
|
# Server Session
|
||||||
#------------------------------------------------------------
|
# ------------------------------------------------------------
|
||||||
|
|
||||||
|
|
||||||
class Session(object):
|
class Session(object):
|
||||||
"""
|
"""
|
||||||
|
|
@ -135,41 +137,6 @@ class Session(object):
|
||||||
if self.account:
|
if self.account:
|
||||||
self.protocol_flags.update(self.account.attributes.get("_saved_protocol_flags", {}))
|
self.protocol_flags.update(self.account.attributes.get("_saved_protocol_flags", {}))
|
||||||
|
|
||||||
# helpers
|
|
||||||
|
|
||||||
def encode_output(self, text):
|
|
||||||
"""
|
|
||||||
Encode the given text for output, following the session's protocol flag.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
text (str or bytes): the text to encode to bytes.
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
encoded_text (bytes): the encoded text following the session's
|
|
||||||
protocol flag. If the converting fails, log the error
|
|
||||||
and send the text with "?" in place of problematic
|
|
||||||
characters. If the specified encoding cannot be found,
|
|
||||||
the protocol flag is reset to utf-8.
|
|
||||||
In any case, returns bytes.
|
|
||||||
|
|
||||||
Note:
|
|
||||||
If the argument is bytes, return it as is.
|
|
||||||
|
|
||||||
"""
|
|
||||||
if isinstance(text, bytes):
|
|
||||||
return text
|
|
||||||
|
|
||||||
try:
|
|
||||||
encoded = text.encode(self.protocol_flags["ENCODING"])
|
|
||||||
except LookupError:
|
|
||||||
self.protocol_flags["ENCODING"] = 'utf-8'
|
|
||||||
encoded = text.encode('utf-8')
|
|
||||||
except UnicodeEncodeError:
|
|
||||||
print("An error occurred during string encoding to {encoding}. Will remove errors and try again.".format(encoding=self.protocol_flags["ENCODING"]))
|
|
||||||
encoded = text.encode(self.protocol_flags["ENCODING"], errors="replace")
|
|
||||||
|
|
||||||
return encoded
|
|
||||||
|
|
||||||
# access hooks
|
# access hooks
|
||||||
|
|
||||||
def disconnect(self, reason=None):
|
def disconnect(self, reason=None):
|
||||||
|
|
|
||||||
|
|
@ -675,7 +675,7 @@ class ANSIString(with_metaclass(ANSIMeta, str)):
|
||||||
"""
|
"""
|
||||||
string = args[0]
|
string = args[0]
|
||||||
if not isinstance(string, str):
|
if not isinstance(string, str):
|
||||||
string = to_str(string, force_string=True)
|
string = to_str(string)
|
||||||
parser = kwargs.get('parser', ANSI_PARSER)
|
parser = kwargs.get('parser', ANSI_PARSER)
|
||||||
decoded = kwargs.get('decoded', False) or hasattr(string, '_raw_string')
|
decoded = kwargs.get('decoded', False) or hasattr(string, '_raw_string')
|
||||||
code_indexes = kwargs.pop('code_indexes', None)
|
code_indexes = kwargs.pop('code_indexes', None)
|
||||||
|
|
|
||||||
|
|
@ -801,7 +801,7 @@ class EvEditor(object):
|
||||||
try:
|
try:
|
||||||
self._buffer = self._loadfunc(self._caller)
|
self._buffer = self._loadfunc(self._caller)
|
||||||
if not isinstance(self._buffer, str):
|
if not isinstance(self._buffer, str):
|
||||||
self._buffer = to_str(self._buffer, force_string=True)
|
self._buffer = to_str(self._buffer)
|
||||||
self._caller.msg("|rNote: input buffer was converted to a string.|n")
|
self._caller.msg("|rNote: input buffer was converted to a string.|n")
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
from evennia.utils import logger
|
from evennia.utils import logger
|
||||||
|
|
|
||||||
|
|
@ -207,8 +207,8 @@ class EvForm(object):
|
||||||
self.filename = filename
|
self.filename = filename
|
||||||
self.input_form_dict = form
|
self.input_form_dict = form
|
||||||
|
|
||||||
self.cells_mapping = dict((to_str(key, force_string=True), value) for key, value in cells.items()) if cells else {}
|
self.cells_mapping = dict((to_str(key), value) for key, value in cells.items()) if cells else {}
|
||||||
self.tables_mapping = dict((to_str(key, force_string=True), value) for key, value in tables.items()) if tables else {}
|
self.tables_mapping = dict((to_str(key), value) for key, value in tables.items()) if tables else {}
|
||||||
|
|
||||||
self.cellchar = "x"
|
self.cellchar = "x"
|
||||||
self.tablechar = "c"
|
self.tablechar = "c"
|
||||||
|
|
@ -378,8 +378,8 @@ class EvForm(object):
|
||||||
kwargs.pop("width", None)
|
kwargs.pop("width", None)
|
||||||
kwargs.pop("height", None)
|
kwargs.pop("height", None)
|
||||||
|
|
||||||
new_cells = dict((to_str(key, force_string=True), value) for key, value in cells.items()) if cells else {}
|
new_cells = dict((to_str(key), value) for key, value in cells.items()) if cells else {}
|
||||||
new_tables = dict((to_str(key, force_string=True), value) for key, value in tables.items()) if tables else {}
|
new_tables = dict((to_str(key), value) for key, value in tables.items()) if tables else {}
|
||||||
|
|
||||||
self.cells_mapping.update(new_cells)
|
self.cells_mapping.update(new_cells)
|
||||||
self.tables_mapping.update(new_tables)
|
self.tables_mapping.update(new_tables)
|
||||||
|
|
|
||||||
|
|
@ -142,7 +142,7 @@ class SharedMemoryModelBase(ModelBase):
|
||||||
if _GA(cls, "_is_deleted"):
|
if _GA(cls, "_is_deleted"):
|
||||||
raise ObjectDoesNotExist("Cannot set %s to %s: Hosting object was already deleted!" % (fname, value))
|
raise ObjectDoesNotExist("Cannot set %s to %s: Hosting object was already deleted!" % (fname, value))
|
||||||
if isinstance(value, (str, int)):
|
if isinstance(value, (str, int)):
|
||||||
value = to_str(value, force_string=True)
|
value = to_str(value)
|
||||||
if (value.isdigit() or value.startswith("#")):
|
if (value.isdigit() or value.startswith("#")):
|
||||||
# we also allow setting using dbrefs, if so we try to load the matching object.
|
# we also allow setting using dbrefs, if so we try to load the matching object.
|
||||||
# (we assume the object is of the same type as the class holding the field, if
|
# (we assume the object is of the same type as the class holding the field, if
|
||||||
|
|
|
||||||
|
|
@ -421,7 +421,7 @@ def parse_inlinefunc(string, strip=False, available_funcs=None, stacktrace=False
|
||||||
# execute the inlinefunc at this point or strip it.
|
# execute the inlinefunc at this point or strip it.
|
||||||
kwargs["inlinefunc_stack_depth"] = depth
|
kwargs["inlinefunc_stack_depth"] = depth
|
||||||
retval = "" if strip else func(*args, **kwargs)
|
retval = "" if strip else func(*args, **kwargs)
|
||||||
return utils.to_str(retval, force_string=True)
|
return utils.to_str(retval)
|
||||||
retval = "".join(_run_stack(item) for item in stack)
|
retval = "".join(_run_stack(item) for item in stack)
|
||||||
if stacktrace:
|
if stacktrace:
|
||||||
out = "STACK: \n{} => {}\n".format(stack, retval)
|
out = "STACK: \n{} => {}\n".format(stack, retval)
|
||||||
|
|
|
||||||
|
|
@ -784,37 +784,86 @@ def latinify(unicode_string, default='?', pure_ascii=False):
|
||||||
return ''.join(converted)
|
return ''.join(converted)
|
||||||
|
|
||||||
|
|
||||||
def to_str(obj, encoding='utf-8', force_string=False):
|
def to_bytes(text, session=None):
|
||||||
"""
|
"""
|
||||||
This function is deprecated in the Python 3 version of Evennia and is
|
Try to encode the given text to bytes, using encodings from settings or from Session. Will
|
||||||
likely to be phased out in future releases.
|
always return a bytes, even if given something that is not str or bytes.
|
||||||
|
|
||||||
---
|
|
||||||
This encodes a unicode string back to byte-representation,
|
|
||||||
for printing, writing to disk etc.
|
|
||||||
|
|
||||||
Args:
|
Args:
|
||||||
obj (any): Object to encode to bytecode.
|
text (any): The text to encode to bytes. If bytes, return unchanged. If not a str, convert
|
||||||
encoding (str, optional): The encoding type to use for the
|
to str before converting.
|
||||||
encoding.
|
session (Session, optional): A Session to get encoding info from. Will try this before
|
||||||
force_string (bool, optional): Always convert to string, no
|
falling back to settings.ENCODINGS.
|
||||||
matter what type `obj` is initially.
|
|
||||||
|
|
||||||
Notes:
|
Returns:
|
||||||
Non-string objects are let through without modification - this
|
encoded_text (bytes): the encoded text following the session's protocol flag followed by the
|
||||||
is required e.g. for Attributes. Use `force_string` to force
|
encodings specified in settings.ENCODINGS. If all attempt fail, log the error and send
|
||||||
conversion of objects to strings.
|
the text with "?" in place of problematic characters. If the specified encoding cannot
|
||||||
|
be found, the protocol flag is reset to utf-8. In any case, returns bytes.
|
||||||
|
|
||||||
|
Note:
|
||||||
|
If `text` is already bytes, return it as is.
|
||||||
|
|
||||||
"""
|
"""
|
||||||
if isinstance(obj, (str, bytes )):
|
if isinstance(text, bytes):
|
||||||
return obj
|
return text
|
||||||
|
if not isinstance(text, str):
|
||||||
|
# convert to a str representation before encoding
|
||||||
|
try:
|
||||||
|
text = str(text)
|
||||||
|
except Exception:
|
||||||
|
text = repr(text)
|
||||||
|
|
||||||
if force_string:
|
default_encoding = session.protocol_flags.get("ENCODING", 'utf-8') if session else 'utf-8'
|
||||||
# some sort of other object. Try to
|
try:
|
||||||
# convert it to a string representation.
|
return text.encode(default_encoding)
|
||||||
obj = str(obj)
|
except (LookupError, UnicodeEncodeError):
|
||||||
|
for encoding in settings.ENCODINGS:
|
||||||
|
try:
|
||||||
|
return text.encode(encoding)
|
||||||
|
except (LookupError, UnicodeEncodeError):
|
||||||
|
pass
|
||||||
|
# no valid encoding found. Replace unconvertable parts with ?
|
||||||
|
return text.encode(default_encoding, errors="replace")
|
||||||
|
|
||||||
return obj
|
|
||||||
|
def to_str(text, session=None):
|
||||||
|
"""
|
||||||
|
Try to decode a bytestream to a python str, using encoding schemas from settings
|
||||||
|
or from Session. Will always return a str(), also if not given a str/bytes.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
text (any): The text to encode to bytes. If a str, return it. If also not bytes, convert
|
||||||
|
to str using str() or repr() as a fallback.
|
||||||
|
session (Session, optional): A Session to get encoding info from. Will try this before
|
||||||
|
falling back to settings.ENCODINGS.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
decoded_text (str): The decoded text.
|
||||||
|
|
||||||
|
Note:
|
||||||
|
If `text` is already str, return it as is.
|
||||||
|
"""
|
||||||
|
if isinstance(text, str):
|
||||||
|
return text
|
||||||
|
if not isinstance(text, bytes):
|
||||||
|
# not a byte, convert directly to str
|
||||||
|
try:
|
||||||
|
return str(text)
|
||||||
|
except Exception:
|
||||||
|
return repr(text)
|
||||||
|
|
||||||
|
default_encoding = session.protocol_flags.get("ENCODING", 'utf-8') if session else 'utf-8'
|
||||||
|
try:
|
||||||
|
return text.decode(default_encoding)
|
||||||
|
except (LookupError, UnicodeDecodeError):
|
||||||
|
for encoding in settings.ENCODINGS:
|
||||||
|
try:
|
||||||
|
return text.decode(encoding)
|
||||||
|
except (LookupError, UnicodeDecodeError):
|
||||||
|
pass
|
||||||
|
# no valid encoding found. Replace unconvertable parts with ?
|
||||||
|
return text.decode(default_encoding, errors="replace")
|
||||||
|
|
||||||
|
|
||||||
def validate_email_address(emailaddress):
|
def validate_email_address(emailaddress):
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue