Aiviro Utilities
The section is a hub for essential tools like constants, methods, and classes. It’s sorted into different categories, each handling specific tasks like spatial objects, logging, parameter controls, constants, keyboard and exceptions. These tools are vital for different functions within the Aiviro package.
Area & Secret
- class aiviro.Area(x_min: int, y_min: int, x_max: int, y_max: int)
Rectangular area defined in x and y axes.
- Parameters:
x_min – Lower x-axis coordinate.
y_min – Lower y-axis coordinate.
x_min – Upper x-axis coordinate.
y_min – Upper y-axis coordinate.
- Example:
>>> import aiviro >>> r = aiviro.create_desktop_robot() >>> r.add_mask(aiviro.Area(x_min=750, y_min=850, x_max=1100, y_max=950))
- classmethod from_points(p1: Point, p2: Point) Area
Create an area from two points.
- Parameters:
p1 – First point.
p2 – Second point.
- Returns:
Area.
- copy(x_min: int | None = None, y_min: int | None = None, x_max: int | None = None, y_max: int | None = None) Area
Create a copy of the area with updated coordinates.
- Parameters:
x_min – New x_min value.
y_min – New y_min value.
x_max – New x_max value.
y_max – New y_max value.
- is_intersecting(other: Area, strict: bool = False) bool
Check if two areas are intersecting. https://stackoverflow.com/questions/20925818/algorithm-to-check-if-two-boxes-overlap
- Parameters:
other – Other area.
strict – If True, areas must have common points to be considered intersecting.
- Returns:
True if areas are intersecting.
- is_inside(other: Area, strict: bool = False) bool
Determine if self-area is inside other-area.
- Parameters:
other – Other area.
strict – If True, the area should be strictly inside (not touching the border) of other area.
- Returns:
True if area is inside other area.
- class aiviro.BoundBox(x_min: int, y_min: int, x_max: int, y_max: int, label: ~aiviro.core.constants.ui_constants.AiviroCoreElementLabels = AiviroCoreElementLabels.DEFAULT, score: float = 0.0, extra: ~typing.Dict | None = None, *, text: str = '', text_type: str = '', text_boxes: ~typing.List[~aiviro.core.utils.bound_box.bbox.BoundBox] = <factory>, regex_match: ~_regex.Match | None = None, or_index: int = -1, extra_data: ~typing.Dict[str, ~typing.Any] = <factory>)
Bounding box, object returned after finding an element.
- Example:
>>> # Get the text of the bound-box. >>> import aiviro >>> r = aiviro.create_desktop_robot() >>> box = r.get(aiviro.Text("Hello, World!")) >>> box.text 'Hello, World!'
>>> # Get the regex match of the bound-box. >>> import aiviro >>> r = aiviro.create_desktop_robot() >>> box = r.get( ... aiviro.OnTheRight( ... aiviro.RegexText(r"(\d\d\.\d\d\.\d\d\d\d)"), ... aiviro.Text("Date") ... ) ... ) >>> box.regex_match.group(1) '13.11.2020'
>>> # Get the index of the Or element in the bound-box. >>> import aiviro >>> r = aiviro.create_desktop_robot() >>> or_box = r.get( ... aiviro.Or(aiviro.Text("not-found"), aiviro.Text("found")) ... ) >>> or_box.or_index 1
- classmethod from_points(p1: Point, p2: Point, label: AiviroCoreElementLabels = AiviroCoreElementLabels.DEFAULT) BoundBox
Create a box from two points.
- Parameters:
p1 – First point.
p2 – Second point.
label – Label of the box.
- classmethod from_area(area_box: Area, label: AiviroCoreElementLabels = AiviroCoreElementLabels.AREA_TYPE_BOX) BoundBox
Create a box from an area.
- Parameters:
area_box – Area.
label – Label of the box.
- copy(x_min: int | None = None, y_min: int | None = None, x_max: int | None = None, y_max: int | None = None, label: AiviroCoreElementLabels | None = None, score: float | None = None, extra: Dict | None = None) BoundBox
Create a copy of the box with updated values.
- Parameters:
x_min – New x_min value.
y_min – New y_min value.
x_max – New x_max value.
y_max – New y_max value.
label – New label value.
score – New score value.
extra – New extra data
- class aiviro.Secret(value: str | Secret)
Holds a string value that should not be revealed in logs, tracebacks, etc. To retrieve the value use the str() function.
- Example:
>>> Secret("some string") Secret('********') >>> str(Secret("some string")) some string
Note
If you use
print()
(or its variants) directly with this object it is automatically cast tostr()
, so it will display the secret value. Use castrepr()
to avoid this functionality.
Convertors
All convertoring methods can be accessed by aiviro.*
notation.
- convertors.convert_get_result_to_box() BoundBox | None
Converts the result of a get or other commands to a single box.
>>> import aiviro >>> r = aiviro.create_desktop_robot() >>> my_text = aiviro.convert_get_result_to_box( ... r.get(aiviro.Text("Order Number")) ... )
- Parameters:
boxes – The result of a robot’s command.
- Raises:
ValueError – If the result contains more than one box, in that case use convert_get_result_to_list.
- Returns:
A single box.
- convertors.convert_get_result_to_list() List[BoundBox]
Converts the result of a get or other commands to a list of boxes. This is useful for solving typing problems in the code, as the result of the get command can be a single box, a list of boxes, or a list of lists of boxes.
- Example:
>>> import aiviro >>> r = aiviro.create_desktop_robot() >>> my_text = aiviro.convert_get_result_to_list( ... r.get(aiviro.Text("text-1"), aiviro.Text("text-2")) ... )
- Parameters:
boxes – The result of a robot’s command.
- Returns:
A list of boxes.
- convertors.convert_to_decimal() Decimal | None
Converts a string, float, or integer to a Decimal number.
- Parameters:
amount – The amount to convert.
- Returns:
The converted amount as a Decimal number.
- convertors.convert_to_datetime(language_hint: List[str] | None = None) datetime | None
Converts a string to a datetime object.
- Parameters:
text_date – The date to convert.
language_hint – A list of language hints to help the parser, by default [“cs”].
- Returns:
The converted date as a datetime object.
- convertors.convert_to_word_boxes() List[BoundBox]
Selects word boxes from a list of line boxes. By default, Aiviro connects several words that logically belong together into a single box and calls it a line box.
- Parameters:
line_boxes – The list of line boxes, most likely resulting from a
get
command.- Returns:
The list of boxes that contain individual words.
Logging
All logging methods can be accessed by aiviro.*
notation.
- logging.init_logging(config_file: GlobalConfig | None = None) LoggingContext
Initialize Aiviro logging.
- Parameters:
log_folder – Path to the folder where logs will be stored. If not specified, the default folder based on the configuration will be used.
config_file – GlobalConfig instance. If not specified, the default configuration will be used.
- Returns:
LoggingContext instance, which can be used to add additional handlers
- Example:
>>> import aiviro >>> if __name__ == "__main__": >>> aiviro.init_logging() >>> # some code
>>> import aiviro >>> from typing import List >>> from logging import StreamHandler, DEBUG, Filter, LogRecord >>> >>> class LoggerNameFilter(Filter): ... def __init__(self, loggers_name: List[str]): ... super().__init__() ... self._loggers_name = loggers_name ... ... def filter(self, record: LogRecord) -> bool: ... return record.name in self._loggers_name ... >>> if __name__ == "__main__": >>> l_ctx = aiviro.init_logging() >>> >>> # you can add additional handlers, for example >>> # this handler will print only logs from specified loggers to stdout >>> stdout_handler = StreamHandler(stream=sys.stdout) >>> stdout_handler.setLevel(DEBUG) >>> stdout_handler.addFilter( ... LoggerNameFilter([ ... "aiviro.module.pdf.robot", ... "aiviro.core.find.GetElements", ... "aiviro.core.services.SearchObjectService", ... ]) ... ) >>> # add handler to the main aiviro logger >>> main_logger = l_ctx.main_logger >>> main_logger.addHandler(stdout_handler)
- logging.stop_logging()
Stop logging and close all handlers.
- logging.get_logger() Logger
Get an ‘aiviro’ logger with the specified name.
- Parameters:
name – Name of the logger. For aiviro logger the name must start with aiviro.. Otherwise, the logger won’t be a child of the aiviro logger.
- Returns:
Logger with the specified name.
- Example:
>>> import aiviro >>> logger = aiviro.get_logger("scenario.module") >>> logger.info("This is info message") >>> print(logger.name) aiviro.scenario.module
- logging.get_logger_advanced(robot_id: str = '') AiviroLoggerAdapter
Get an advanced logger, containing robot_id and other information. The logger supports providing images and data to the log message.
- Parameters:
name – Name of the logger. Internally
get_logger()
is used to get the logger.robot_id – Robot ID, which will be added to every log message.
- Example:
>>> import aiviro >>> from aiviro.core.utils.file_utils import load_image >>> logger = aiviro.get_logger_advanced("aiviro.scenario.module") >>> np_image = load_image("path/to/image.png") >>> logger.info("This is info message", image=np_image)
>>> import aiviro >>> r = aiviro.create_desktop_robot() >>> boxes = r.get(aiviro.Text("some text")) >>> logger.info("Found boxes", data=boxes)
- logging.disable_logger() Generator
Get a context manager which disables the logger with the specified name Upon exiting the context manager, the logger will be enabled again.
- Parameters:
name – Name of the logger to disable. Internally
get_logger()
is used to get the logger.
Control Parameters
- class aiviro.core.constants.ui_constants.FindMethod(value)
Different methods to process texts can be used with the alias
aiviro.find_method
.- Example:
>>> import aiviro >>> aiviro.Button("Log in", aiviro.find_method.EQUAL) >>> aiviro.Text(r"^\d\d$", aiviro.find_method.REGEX) >>> aiviro.Button("OK", aiviro.find_method.EQUAL) >>> aiviro.Text("Total", aiviro.find_method.EQUAL.EXCLUSIVE) >>> aiviro.Text("[1] Tab Text")
>>> import aiviro >>> from typing import Tuple >>> def my_text_normalization(ocr_text: str) -> str: ... return ocr_text.replace("O", "0") ... >>> aiviro.user_config.text_normalization = my_text_normalization >>> aiviro.Text(r"^\d+$", aiviro.find_method.REGEX.USER_NORMALIZATION)
- AUTO = 1
Method
It tries several predefined methods and selects the one with matching text. This method is used by default.
- DYNAMIC = 2
Method
Comparison of text is based on dynamic method. Method finds each word in the text, it’s compared as EQUAL. e.g. “Hello world” will be found in “Hi Hello world !”.
- EQUAL = 4
Method
Comparison of text is based on equality. Texts have to be equal with every character to pass
- SUBSTRING = 8
Method
Comparison of text is based on substring equality. Text has to be equal to a substring with every character to pass
- SIMILAR = 16
Method
Comparison of text is based on their respective similarity. The texts can have minor changes and still pass.
- REGEX = 32
Method
Comparison of text is based on regular expression. Text has to exactly match the regular expression to pass.
- CASE_SENSITIVE = 64
Flag
Must be combined with some other comparison method. Comparison of text is case-sensitive. This option is ignored forREGEX
.
- EXCLUSIVE = 128
Flag
Must be combined with some other comparison method. Find elements that contain only the specified text and nothing else.
- GLOBAL = 256
Flag
Must be combined with some other comparison method. Returns all occurrences in line not only the first one.
- USER_NORMALIZATION = 512
Flag
Must be combined with some other comparison method. Apply user’s normalization for text. SeeUserConfig
for more information.
- CASE_INSENSITIVE = 1024
Flag
[DEPRECATED] It’ll be removed in the future versions. (Changed in version 3.19.0) Now automatically every text is taken as case-insensitive
- class aiviro.core.utils.search_objects.TableAnchor(value)
Table anchors can be used with the alias
aiviro.table_anchor
. It defines which anchor of the element should be used for findingAbove
orBelow
elements.- Example:
>>> import aiviro >>> aiviro.Below( ... aiviro.Text(r'^\d\w\d$', aiviro.find_method.REGEX), ... aiviro.Text("Column-header"), ... aiviro.table_anchor.RIGHT ... ) >>> aiviro.Below( ... aiviro.Text(""), ... aiviro.Text("Records on the right"), ... aiviro.table_anchor.RIGHT.THRESHOLD ... ) >>> aiviro.Below( ... aiviro.RegexText(r"\d{8}"), ... aiviro.Text("Customer No."), ... aiviro.table_anchor.MIDDLE.DISTANCE ... )
- LEFT = 1
Method
The distance between the most left point of reference and other objects is calculated, based on x-axis, to find the closest elements.
- RIGHT = 2
Method
The distance between the most right points of reference and other objects is calculated, based on x-axis, to find the closest elements.
- MIDDLE = 4
Method
The distance between the middle point of reference and other objects is calculated, based on x-axis, to find the closest elements.
- DISTANCE = 8
Flag
The distance option must be used with aLEFT
,RIGHT
,MIDDLE
option, it calculates the distance between reference and other objects based on both axis.
- THRESHOLD = 16
Flag
The threshold is applied for distance between reference and searching object, it finds strictly aligned objects, based on the selected anchor
- NONE = 32
Method
All elements Below or Above from reference object are found.
- class aiviro.core.constants.ui_constants.LayoutAnchor(value)
Anchor positions can be used with the alias
aiviro.layout_anchor
.- Example:
>>> import aiviro >>> r = aiviro.create_desktop_robot() >>> r.check_are_in_column( # elements are in column with alignment on the left ... aiviro.Input("Username"), ... aiviro.Input("Password"), ... anchor=aiviro.layout_anchor.LEFT ... )
- LEFT = 'LEFT'
Use left points of bounding boxes as anchor points
- RIGHT = 'RIGHT'
Use right points of bounding boxes as anchor points
- MIDDLE = 'MIDDLE'
Use center points of bounding boxes as anchor points
- TOP = 'TOP'
Use top points of bounding boxes as anchor points
- BOTTOM = 'BOTTOM'
Use bottom points of bounding boxes as anchor points
- BOTH = 'BOTH'
Use both left and right or top and bottom points of bounding boxes as anchor points
- class aiviro.core.constants.ui_constants.LayoutPositions(value)
Layout positioning can be used with the alias
aiviro.layout_positions
.- IS_ABOVE = 0
Above a position
- IS_BELOW = 1
Below a position
- IS_RIGHT_OF = 2
Right of a position
- IS_LEFT_OF = 3
Left of a position
- class aiviro.core.constants.ui_constants.LayoutComparisonMode(value)
Layout comparison modes can be used with the alias
aiviro.layout_comparison_mode
.- Example:
>>> import aiviro >>> r = aiviro.create_desktop_robot() >>> r.check_is_positioned( # the whole Button object is below Input field ... aiviro.Button("Log in"), ... aiviro.Input("Username"), ... direction_type=aiviro.layout_positions.IS_BELOW, ... comparison_mode=aiviro.layout_comparison_mode.WHOLE ... )
- WHOLE = 'COMPARISON_WHOLE'
The entire area is strictly in the specified direction from given object
- CENTER = 'COMPARISON_CENTER'
The center is in specified direction from the center of the given object
- EXTREME = 'COMPARISON_EXTREME'
there is at least one pixel in the specified direction from the given object
- class aiviro.core.constants.ui_constants.ScrollOption(value)
Layout positioning can be used with the alias
aiviro.scroll_option
.- MOUSE_UP = 'up'
Scroll with the mouse up
- MOUSE_DOWN = 'down'
Scroll with the mouse down
- PAGE_UP = 'page_up'
Use page up as a scroll
- PAGE_DOWN = 'page_down'
Use page down as a scroll
- class aiviro.core.constants.ui_constants.BoundaryType(value)
Can be used with the alias
aiviro.boundary_type
.- Example:
>>> import aiviro >>> r = aiviro.create_desktop_robot() >>> r.set_working_area(aiviro.BoundaryArea( ... aiviro.Button("Login"), ... aiviro.Button("Cancel"), ... aiviro.boundary_type.DEFAULT ... )) >>> r.add_mask(aiviro.BoundaryArea( ... aiviro.Button("Login"), ... aiviro.Button("Cancel"), ... aiviro.boundary_type.EXCLUDE_FIRST.RIGHT_EDGE ... ))
- DEFAULT = 1
Both boundaries included into the resulting box
- RIGHT_EDGE = 2
Expand right side to the edge of the display
- TOP_EDGE = 4
Expand top side to the edge of the display
- BOTTOM_EDGE = 8
Expand bottom side to the edge of the display
- LEFT_EDGE = 16
Expand left side to the edge of the display
- EXCLUDE_FIRST_WIDTH = 32
Exclude width of the first boundary (uses right side of the boundary instead of left)
- EXCLUDE_FIRST_HEIGHT = 64
Exclude height of the first boundary (uses bottom side of the boundary instead of top)
- EXCLUDE_SECOND_WIDTH = 128
Exclude width of the second boundary (uses left side of the boundary instead of right)
- EXCLUDE_SECOND_HEIGHT = 256
Exclude height of the second boundary (uses top side of the boundary instead of bottom)
- TOP_LEFT_CORNER = 20
Expand boundary to the top left corner of the screen
- TOP_RIGHT_CORNER = 6
Expand boundary to the top right corner of the screen
- BOTTOM_LEFT_CORNER = 24
Expand boundary to the bottom left corner of the screen
- BOTTOM_RIGHT_CORNER = 10
Expand boundary to the bottom right corner of the screen
- BOTH_WIDTH = 18
Expand boundary to both vertical sides of the screen
- BOTH_HEIGHT = 12
Expand boundary to both horizontal sides of the screen
- EXCLUDE_FIRST = 96
Exclude the whole first boundary
- EXCLUDE_SECOND = 384
Exclude the whole second boundary
- EXCLUDE_BOTH_WIDTH = 160
Exclude width of both boundaries
- EXCLUDE_BOTH_HEIGHT = 320
Exclude height of both boundaries
- EXCLUDE_BOTH = 480
Exclude both boundaries
- class aiviro.core.constants.ui_constants.WorkingAreaLimit(value)
Can be used with the alias
aiviro.working_area_limit
.- Example:
>>> import aiviro >>> r = aiviro.create_desktop_robot() >>> # set some working-area >>> r.limit_working_area( ... aiviro.Text("Cancel"), ... top=aiviro.working_area_limit.MIN, ... right=aiviro.working_area_limit.MAX ... ) >>> # set another working-area >>> r.limit_working_area( ... aiviro.Text("Cancel"), ... bottom=aiviro.working_area_limit.MAX, ... right=aiviro.working_area_limit.MIN ... )
- MIN = 'min'
Use MIN option for selected axis
- MAX = 'max'
Use MAX option for selected axis
Constants
Every constant can be access by aiviro.*
notation.
>>> import aiviro
>>> DB_NAME = "production" if aiviro.AIVIRO_DEBUG_VALUE else "testing"
- aiviro.core.constants.environment_constants.AIVIRO_DEBUG_KEY = 'AIVIRO_DEBUG'
The name of the environment variable that indicates whether the debug mode is on or off.
- aiviro.core.constants.environment_constants.AIVIRO_DEBUG_VALUE = 1
The integer value that indicates whether the debug mode is on or off.
- aiviro.core.constants.environment_constants.IS_RUNNING_IN_PREFECT = False
The boolean value that indicates whether script is running in Prefect environment.
- aiviro.core.constants.environment_constants.SCENARIO_CONFIG_FOLDER_KEY = 'AIVIRO__CONFIG_FOLDER'
The name of the environment variable that indicates the path to the scenario configuration folder.
- aiviro.core.constants.environment_constants.SCENARIO_CONFIG_FOLDER_PATH = None
The path to the folder containing the scenario configuration files.
- aiviro.core.utils.locale.LOCALE_KEY = 'AIVIRO__LOCALE'
The name of the environment variable that indicates which localization should be used.
- aiviro.core.utils.locale.LOCALE_VALUE = 'cs_CZ'
The value of loaded localization, defaults to “cs_CZ”.
Keyboard
- class aiviro.core.services.device.keyboard.BaseKey(value)
can be used with the alias
aiviro.key
.- ESCAPE = ('escape',)
- F1 = ('f1',)
- F2 = ('f2',)
- F3 = ('f3',)
- F4 = ('f4',)
- F5 = ('f5',)
- F6 = ('f6',)
- F7 = ('f7',)
- F8 = ('f8',)
- F9 = ('f9',)
- F10 = ('f10',)
- F11 = ('f11',)
- F12 = ('f12',)
- GRAVE = ('`',)
- KEY_1 = ('1',)
- KEY_2 = ('2',)
- KEY_3 = ('3',)
- KEY_4 = ('4',)
- KEY_5 = ('5',)
- KEY_6 = ('6',)
- KEY_7 = ('7',)
- KEY_8 = ('8',)
- KEY_9 = ('9',)
- KEY_0 = ('0',)
- MINUS = ('-',)
- EQUAL = ('=',)
- DOUBLE_HYPHEN = ('=',)
- BACKSPACE = ('\x08',)
- TAB = ('\t',)
- KEY_Q = ('q',)
- KEY_W = ('w',)
- KEY_E = ('e',)
- KEY_R = ('r',)
- KEY_T = ('t',)
- KEY_Y = ('y',)
- KEY_U = ('u',)
- KEY_I = ('i',)
- KEY_O = ('o',)
- KEY_P = ('p',)
- LEFT_SQUARE_BRACKET = ('[',)
- RIGHT_SQUARE_BRACKET = (']',)
- ENTER = ('enter',)
- CAPS_LOCK = ('capslock',)
- KEY_A = ('a',)
- KEY_S = ('s',)
- KEY_D = ('d',)
- KEY_F = ('f',)
- KEY_G = ('g',)
- KEY_H = ('h',)
- KEY_J = ('j',)
- KEY_K = ('k',)
- KEY_L = ('l',)
- SEMICOLON = (';',)
- APOSTROPHE = ("'",)
- LEFT_SHIFT = ('shiftleft',)
- KEY_Z = ('z',)
- KEY_X = ('x',)
- KEY_C = ('c',)
- KEY_V = ('v',)
- KEY_B = ('b',)
- KEY_N = ('n',)
- KEY_M = ('m',)
- COMMA = (',',)
- PERIOD = ('.',)
- SLASH = ('/',)
- RIGHT_SHIFT = ('shiftright',)
- BACKSLASH = ('\\',)
- LEFT_CONTROL = ('ctrlleft',)
- LEFT_WINDOW = ('winleft',)
- LEFT_ALT = ('altleft',)
- SPACE = (' ',)
- RIGHT_ALT = ('altright',)
- MENU = ('menu',)
- RIGHT_WINDOW = ('winright',)
- RIGHT_CONTROL = ('ctrlright',)
- INSERT = ('insert',)
- DELETE = ('delete',)
- HOME = ('home',)
- END = ('end',)
- PAGE_UP = ('pageup',)
- PAGE_DOWN = ('pagedown',)
- LEFT_ARROW = ('left',)
- RIGHT_ARROW = ('right',)
- UP_ARROW = ('up',)
- DOWN_ARROW = ('down',)
- NUMPAD_0 = ('kp0',)
- NUMPAD_1 = ('kp1',)
- NUMPAD_2 = ('kp2',)
- NUMPAD_3 = ('kp3',)
- NUMPAD_4 = ('kp4',)
- NUMPAD_5 = ('kp5',)
- NUMPAD_6 = ('kp6',)
- NUMPAD_7 = ('kp7',)
- NUMPAD_8 = ('kp8',)
- NUMPAD_9 = ('kp9',)
- NUMPAD_DECIMAL = ('kp.',)
- NUMPAD_DIVIDE = ('kp/',)
- NUMPAD_ASTERISK = ('kp*',)
- NUMPAD_MINUS = ('kp-',)
- NUMPAD_PLUS = ('kp+',)
- NUMPAD_ENTER = ('kpenter',)
Exceptions
Every exception can be access by aiviro.*Error
notation.
>>> import aiviro
>>> try:
... # some code
>>> except aiviro.CheckCommandError:
... # process exception
- exception aiviro.core.utils.exceptions.SearchObjectError
- exception aiviro.core.utils.exceptions.CheckCommandError
- exception aiviro.core.utils.exceptions.TooManyElementsError
- exception aiviro.core.utils.exceptions.ElementIndexError
- exception aiviro.core.utils.exceptions.ReferenceElementError
- exception aiviro.core.utils.exceptions.AndElementError
- exception aiviro.core.utils.exceptions.WorkingAreaError
- exception aiviro.core.utils.exceptions.ScreenNotStableError
- exception aiviro.core.utils.exceptions.ScrollEndException
- exception aiviro.core.utils.exceptions.ImageNotSetError
- exception aiviro.core.utils.exceptions.BoundaryElementError