blob: 8e02a7ac302290ac8cd58fb6e645f799c0c1bdf4 [file] [log] [blame]
import inflection
import re
class NamedElement(object):
def __init__(self, **kwargs):
super(NamedElement, self).__init__()
self.name = kwargs.pop('name', "unnamed")
self.description = kwargs.pop('description', "")
def __getattribute__(self, name):
lam = {'CamelCase': lambda: inflection.camelize(self.name),
'camelCase': lambda: NamedElement.lower_camel_case(self.name),
'snake_case': lambda: inflection.underscore(self.name)}\
.get(name)
if lam:
return NamedElement.__fixup_name(lam())
try:
return super(NamedElement, self).__getattribute__(name)
except Exception:
raise AttributeError("Attribute '%s' not found in %s.NamedElement"
% (name, self.__module__))
""" Some names are reserved in some languages. Fixup names to avoid using
reserved words.
"""
@staticmethod
def __fixup_name(name):
# List of reserved words from http://en.cppreference.com/w/cpp/keyword
cppReserved = frozenset({
"alignas", "alignof", "and", "and_eq", "asm", "auto",
"bitand", "bitor", "bool", "break", "case", "catch", "char",
"char8_t", "char16_t", "char32_t", "class", "compl", "concept",
"const", "consteval", "constexpr", "constinit", "const_cast",
"continue", "co_await", "co_return", "co_yield", "decltype",
"default", "delete", "do", "double", "dynamic_cast", "else",
"enum", "explicit", "export", "extern", "false", "float", "for",
"friend", "goto", "if", "inline", "int", "long", "mutable",
"namespace", "new", "noexcept", "not", "not_eq", "nullptr",
"operator", "or", "or_eq", "private", "protected", "public",
"register", "reinterpret_cast", "requires", "return", "short",
"signed", "sizeof", "static", "static_assert", "static_cast",
"struct", "switch", "template", "this", "thread_local", "throw",
"true", "try", "typedef", "typeid", "typename", "union",
"unsigned", "using", "virtual", "void", "volatile", "wchar_t",
"while", "xor", "xor_eq"})
while(name in cppReserved):
name = name + "_"
return name
# Normally you can use inflection.camelize(string, False) to return
# a lowerCamelCase string, but it doesn't work well for acronyms.
# An input like "MACAddress" will become "mACAddress". Try to handle
# acronyms in a better way.
@staticmethod
def lower_camel_case(name):
# Get the UpperCamelCase variation of the name first.
upper_name = inflection.camelize(name)
# If it is all upper case, return as all lower. ex. "MAC"
if re.match(r"^[A-Z]*$", upper_name):
return upper_name.lower()
# If it doesn't start with 2 upper case, it isn't an acronym.
if not re.match(r"^[A-Z]{2}", upper_name):
return upper_name[0].lower() + upper_name[1:]
# If it is upper case followed by 'v[0-9]', treat it all as one word.
# ex. "IPv6Address" -> "ipv6Address"
if re.match(r"^[A-Z]+v[0-9]", upper_name):
return re.sub(
r"^([A-Z]+)(.*)$",
lambda m: m.group(1).lower() + m.group(2),
upper_name)
# Anything left has at least two sequential upper-case, so it is an
# acronym. Switch all but the last upper-case (which starts the next
# word) to lower-case.
# ex. "MACAddress" -> "macAddress".
return re.sub(
r"^([A-Z]+)([A-Z].*)$",
lambda m: m.group(1).lower() + m.group(2),
upper_name)