import os
import sys
import re
import tempfile
import urllib
import posixpath
from types import IntType
from robot.errors import DataError, FrameworkError
from robottypes import is_str, is_list, unic, is_integer
from normalizing import normpath
from error import get_error_message
def get_doc(obj):
try:
doc = obj.__doc__
except AttributeError:
doc = ''
if not doc:
return ''
return '\n'.join([ line.strip() for line in doc.strip().splitlines() ])
def printable_name(string, code_style=False):
"""Generates and returns printable name from the given string.
Examples:
'simple' -> 'Simple'
'name with spaces' -> 'Name With Spaces'
'more spaces' -> 'More Spaces'
'Cases AND spaces' -> 'Cases AND Spaces'
'' -> ''
If 'code_style' is True:
'mixedCAPSCamel' -> 'Mixed CAPS Camel'
'camelCaseName' -> 'Camel Case Name'
'under_score_name' -> 'Under Score Name'
'under_and space' -> 'Under And Space'
'miXed_CAPS_nAMe' -> 'MiXed CAPS NAMe'
'' -> ''
"""
if code_style:
string = string.replace('_', ' ')
parts = string.split()
if len(parts) == 0:
return ''
elif len(parts) == 1 and code_style:
parts = _splitCamelCaseString(parts[0])
parts = [ part[0].upper() + part[1:] for part in parts if part != '' ]
return ' '.join(parts)
def _splitCamelCaseString(string):
parts = []
current_part = []
string = ' ' + string + ' ' # extra spaces make going through string easier
for i in range(1, len(string)-1):
# on 1st/last round prev/next is ' ' and char is 1st/last real char
prev, char, next = string[i-1:i+2]
if _isWordBoundary(prev, char, next):
parts.append(''.join(current_part))
current_part = [ char ]
else:
current_part.append(char)
parts.append(''.join(current_part)) # append last part
return parts
def _isWordBoundary(prev, char, next):
if char.isupper():
return (prev.islower() or next.islower()) and prev.isalnum()
if char.isdigit():
return prev.isalpha()
return prev.isdigit()
def cygpath(path, options='-wa'):
"""Converts given path using Cygwin command 'cygpath'.
Default options convert unixy path to an absolute Windows path.
Returns path unmodified if not in Cygwin.
"""
if 'cygwin' not in sys.platform:
return path
output = os.popen('cygpath %s "%s"' % (options, path))
winpath = output.read().splitlines()[0]
output.close()
return winpath
def get_temp_dir(extrapath=None):
tempdir = os.path.normpath(tempfile.gettempdir())
if not os.path.isdir(tempdir):
raise EnvironmentError("Temp directory '%s' does not exist" % tempdir)
if extrapath is not None:
tempdir = os.path.join(tempdir, extrapath)
# make sure there's no file with the same name as tempdir
if os.path.isfile(tempdir):
i = 1
while os.path.isfile('%s%d' % (tempdir, i)):
i += 1
tempdir = '%s%d' % (tempdir, i)
if not os.path.exists(tempdir):
os.mkdir(tempdir)
return tempdir
def plural_or_not(list_or_int):
if is_integer(list_or_int):
return list_or_int != 1 and "s" or ""
else:
return len(list_or_int) != 1 and "s" or ""
def seq2str(sequence, quote="'", sep=', ', lastsep=' and ', quote_all=False):
"""Returns sequence in format 'item 1', 'item 2' and 'item 3' """
def elem(string):
if not is_str and not quote_all:
return unic(string)
return quote + unic(string) + quote
if not is_list(sequence):
return elem(sequence)
elif len(sequence) == 0:
return ''
elif len(sequence) == 1:
return elem(sequence[0])
ret = []
for i in range(len(sequence)):
if i + 2 == len(sequence):
ret.append(elem(sequence[i]) + lastsep + elem(sequence[i+1]))
break
ret.append(elem(sequence[i]))
return sep.join(ret)
def seq2str2(sequence):
"""Returns sequence in format [ item 1 | item 2 | ... ] """
if not sequence:
return '[ ]'
return '[ %s ]' % ' | '.join([unic(item) for item in sequence])
def printable_name_from_path(path):
"""Create suite name from given path that points to file or directory.
Examples:
'/tmp/tests.py' -> 'Tests'
'/var/data/my_tests/ -> 'My Tests'
'c:\path\my_tests.html' -> 'My Tests'
'd:\MY TESTS.HTML' -> 'MY TESTS'
'e:\myTestCases.html -> 'My Test Cases'
"""
# Get name of the file/dir without leading path and possible extension
name = os.path.splitext(os.path.basename(os.path.normpath(path)))[0]
name = _remove_prefix(name)
return printable_name(name, code_style=True)
def _remove_prefix(name):
return name.split('__', 1)[-1]
def get_directory(path):
"""Returns the directory part of the given path.
If path already is a directory returns it as is, otherwise returns the
directory containing the file
"""
if path.lower().startswith('http://'):
return posixpath.dirname(path)
path = normpath(path)
if os.path.isdir(path):
return path
return os.path.dirname(path)
def get_link_path(target, base):
target = normpath(target)
base = normpath(base)
if not os.path.isdir(base):
base = get_directory(base)
common = os.path.commonprefix([base,target])
# if in Windows and base and link on different drives
if common == '':
return 'file:///' + target.replace('\\', '/')
base_to_common = _base_to_common(base, common)
common_to_target = _commom_to_target(common, target)
link = os.path.join(base_to_common, common_to_target)
return link.replace('\\', '/')
def _base_to_common(base, common):
parts = _split_to_parts(base, common)
parts = ['..'] * len(parts)
return os.sep.join(parts)
def _commom_to_target(common, target):
parts = _split_to_parts(target, common)
return os.sep.join(parts)
def _split_to_parts(path, prefix):
prefix = os.path.normpath(prefix)
parts = []
prevpath = None
while os.path.normpath(path) != prefix and path != prevpath:
prevpath = path
path, part = os.path.split(path)
parts.insert(0, part)
return parts
def calc_percents(passed, failed):
total = passed + failed
if total == 0:
return 0.0, 0.0
pass_percent = 100.0 * passed / total
fail_percent = 100.0 * failed / total
if 0 < pass_percent < 0.1:
return 0.1, 99.9
if 0 < fail_percent < 0.1:
return 99.9, 0.1
pass_percent = round(pass_percent, 1)
fail_percent = round(fail_percent, 1)
return pass_percent, fail_percent
def percents_to_widths(num1, num2):
if num1 + num2 == 0:
return 0.00, 0.00
# Make small percentages better visible
if 0 < num1 < 1:
num1, num2= 1.0, 99.0
elif 0 < num2 < 1:
num1, num2= 99.0, 1.0
# Handle situation where both are rounded up
while num1 + num2 > 100:
num1, num2 = _subtract_from_larger(num1, num2, 0.1)
# Make sure both pass and fail bar fit into 100% also in IE
num1, num2 = _subtract_from_larger(num1, num2, 0.01)
return num1, num2
def _subtract_from_larger(num1, num2, subtr):
if num1 > num2:
num1 -= subtr
else:
num2 -= subtr
return num1, num2
def get_not_none(*args):
for arg in args:
if arg is not None:
return arg
raise FrameworkError('No non-None item found')
文件路径:robotframework-2.1.2\src\robot\utils\misc.py
功能:包含一些杂项处理,比如获取文档、转为可视性好的串、获取临时目录、单复数判断、序列和字符串的转换等,这些函数具有较广的通用性。
阅读(39327) | 评论(0) | 转发(0) |