HEX
Server: LiteSpeed
System: Linux premium212.web-hosting.com 4.18.0-553.124.4.lve.el8.x86_64 #1 SMP Fri May 15 13:02:13 UTC 2026 x86_64
User: vitanhod (1367)
PHP: 8.2.31
Disabled: NONE
Upload Files
File: //opt/cloudlinux/venv/lib64/python3.11/site-packages/wmt/__pycache__/main.cpython-311.pyc
�

��"j�G���ddlZddlZddlZddlZddlZddlZddlZddlZddl	m	Z	m
Z
ddlmZm
Z
ddlmZddlmZddlmZmZmZmZmZddlmZmZmZmZddlmZdd	lm Z dd
l!m"Z"m#Z#m$Z$m%Z%m&Z&m'Z'm(Z(m)Z)ddl*m+Z+m,Z,ddl-m.Z.dd
l/m0Z0ddl1m2Z2ed��Z3e4gd���Z5ej6d��Z7de8fd�Z9Gd�dej:j;��Z<Gd�dej=��Z>d�Z?d�Z@eGd�d����ZAdeBfd�ZCd�ZDd-d�ZEd-deeAfd�ZFd-d �ZGd!�ZHd"�ZId#�ZJd$�ZKd%�ZLd&�ZMd'�ZNd(�ZOePd)kr�eQejR����ZSe3�Td*eS��ee&eS��5ed+e d,��e#��ejejUe?��ejejVe@��ejW��ZXeX�YeO����ddd��dS#1swxYwYdSdS).�N)�datetime�	timedelta)�List�Optional)�	dataclass)�or_)�ScrapeResult�DomainAlerts�setup_database�
session_scope�cleanup_old_data)�get_domains�setup_logger�save_pid_and_lock�	intersect)�init_sentry_client)�get_pkg_version)�PING_TIMEOUT_STATUS_CODE�
SENTRY_DNS�!ERROR_DOMAINS_PING_RETRY_INTERVAL�ERROR_DOMAINS_ALERT_INTERVAL�
WMT_LOCK_FILE�PING_CONNECTIONS�&LICENSE_EXPIRED_FAREWELL_LETTER_MARKER�LICENSE_CHECK_PAUSE)�Notifier�SupportedNotificationTypes)�ErrorReport)�cfg)�CloudlinuxLicenseLib�wmt_scanner)�is_loopback�
is_private�
is_link_local�is_reservedz
100.64.0.0/10�returnc����	tj|���n#t$rYdSwxYwt�dd��}|�|�t	�fd�t
D����rdS�jdkr�tvrdSdS)z�
    True if `host` is a literal IP in a non-routable / blocked range.
    Hostnames return False -- they are filtered after resolution by _SafeResolver.
    F�ipv4_mappedNc3�8�K�|]}t�|��V��dS�N)�getattr)�.0�prop�ips  ���/builddir/build/BUILDROOT/cl-web-monitoring-tool-3.2.4-1.el8.cloudlinux.x86_64/opt/cloudlinux/venv/lib/python3.11/site-packages/wmt/main.py�	<genexpr>z!_is_blocked_ip.<locals>.<genexpr>Bs-�����
@�
@��7�2�t���
@�
@�
@�
@�
@�
@�T�)�	ipaddress�
ip_address�
ValueErrorr+�any�_BLOCKED_IP_PROPERTIES�version�_CGNAT_NETWORK)�host�mappedr.s  @r/�_is_blocked_ipr<4s����
�
�
!�$�
'�
'���������u�u������R���
-�
-�F�
��
��
�
@�
@�
@�
@�)?�
@�
@�
@�@�@���t�	�z�Q���2��/�/��t��5s��
&�&c�&�eZdZdZd�Zdd�Zd�ZdS)�
_SafeResolverz�
    Wraps aiohttp's ThreadedResolver and drops any resolved address that falls
    in a blocked range, closing the DNS-rebinding-to-internal-host vector.
    c�2�tj|i|��|_dSr*)�aiohttp�ThreadedResolver�	_resolver)�self�args�kwargss   r/�__init__z_SafeResolver.__init__Os�� �1�4�B�6�B�B����r1rNc���K�|�"|j�||���d{V��}n"|j�|||���d{V��}d�|D��}|std|�����|S)Nc�<�g|]}t|d���|��S)r:)r<)r,�hs  r/�
<listcomp>z)_SafeResolver.resolve.<locals>.<listcomp>Ws)��B�B�B�a��q��y�(A�(A�B��B�B�Br1zNo routable address for )rB�resolve�OSError)rCr:�port�family�hosts�safes      r/rKz_SafeResolver.resolveRs������>��.�0�0��t�<�<�<�<�<�<�<�<�E�E��.�0�0��t�V�D�D�D�D�D�D�D�D�E�B�B�5�B�B�B���	=��;�T�;�;�<�<�<��r1c��HK�|j����d{V��dSr*)rB�close)rCs r/rRz_SafeResolver.close\s2�����n�"�"�$�$�$�$�$�$�$�$�$�$�$r1)rN)�__name__�
__module__�__qualname__�__doc__rFrKrR�r1r/r>r>IsS��������
C�C�C�����%�%�%�%�%r1r>c�$��eZdZdZd�fd�	Z�xZS)�_SafeTCPConnectorz�
    Rejects IP-literal hosts in a blocked range before connecting. aiohttp's
    stock _resolve_host short-circuits IP literals and skips the resolver, which
    would otherwise leave redirect targets like http://169.254.169.254/ unfiltered.
    Nc���K�t|��rtd|�����t���|||����d{V��S)NzBlocked non-routable address: )�traces)r<rL�super�
_resolve_host)rCr:rMr[�	__class__s    �r/r]z_SafeTCPConnector._resolve_hostgs`������$���	C��A�4�A�A�B�B�B��W�W�*�*�4��f�*�E�E�E�E�E�E�E�E�Er1r*)rSrTrUrVr]�
__classcell__)r^s@r/rYrY`sQ���������F�F�F�F�F�F�F�F�F�Fr1rYc���tj��d�tj�����D��}t�dt
|����dS)Nc�&�i|]\}}|dk�||��S)�report_emailrW)r,�k�vs   r/�
<dictcomp>zreload_conf.<locals>.<dictcomp>os(��P�P�P�4�1�a�A��<O�<O�!�Q�<O�<O�<Or1zReloading config: %s)r�reload�to_dict�items�logger�info�str)�
sig_number�frame�
config_logs   r/�reload_confromsS���J�L�L�L�P�P�3�;�=�=�#6�#6�#8�#8�P�P�P�J�
�K�K�&��J���8�8�8�8�8r1c�.�tjd��dS)zp
    Shutdown to call finally block to
    close all fds, remove lock and file
    see: save_pid_and_lock()
    rN)�sys�exit)rlrms  r/�shutdownrsss���H�Q�K�K�K�K�Kr1c�h�eZdZUeed<dZeeed<dZeeed<dZ	eeed<dS)�ScrapeCoroResult�urlN�
response_code�response_time_ms�original_domain)
rSrTrUrk�__annotations__rwr�intrxryrWr1r/ruru|sX�������	�H�H�H�#'�M�8�C�=�'�'�'�&*��h�s�m�*�*�*�%)�O�X�c�]�)�)�)�)�)r1ruc��ttjdt��}	t|��}n#tt
f$r
t}YnwxYwt
d|��S)N�ping_connections�)r+rrr{�	TypeErrorr5�max)�values r/�get_connection_limitr��s`���C�G�/�1A�B�B�E�!��E�
�
�����z�"�!�!�!� ����!�����q�%�=�=�s�2�A
�A
c�����K�|4�d{V����fd�}|�d��r|}d|td��d�z}n<|�d��rd|td��d�z}|}n
d|��}d|��}||���d{V��}||_|jdkr|cddd���d{V��S||���d{V��}||_|cddd���d{V��S#1�d{V��swxYwYdS)a
    Main 'pinger'
    1. Requests domains
     - if domain responded - keep status code
     - if no response for timeout - keep Timeout status code
     - if unreachable (ConnectionError or so) - keep 523 status code
       (same logic as go implementation)
    Nc
��,�K�tj��}	��|����4�d{V��}t|d��rt|j��n|}t||jtdtj��|z
z�����cddd���d{V��S#1�d{V��swxYwYdS#tj	j
$rt|t�dz���cYStj
j$rt|d���cYSwxYw)N��timeoutrvi�)rwrxi)rw)�time�get�hasattrrkrvru�statusr{�
concurrent�futures�TimeoutErrorrr@�client_exceptions�ClientError)�
target_url�start�resp�	final_url�ping_timeout�sessions    ��r/�_fetchzping.<locals>._fetch�s�������I�K�K�E�
G�"�;�;�z�<�;�H�H��������D�18��u�1E�1E� U��D�H�
�
�
�:�I�+�!���),�T�T�Y�[�[�5�5H�-I�)J�)J���������������������������������������%�2�
N�
N�
N�'�
�6N�9E��9L�N�N�N�N�N�N��,�8�
G�
G�
G�(�
�#�F�F�F�F�F�F�
G���s;�B<�A!B)�B<�)
B3�3B<�6B3�7B<�<.D�,$D�Dzhttps://zhttp://��)�
startswith�lenryrw)	rvr�r��	semaphorer��	https_url�http_url�https_result�http_results	 ``      r/�pingr��s'�������&�&�&�&�&�&�&�&�	G�	G�	G�	G�	G�	G�(�>�>�*�%�%�	'��I� �3�s�:���'7�'7�#8�8�H�H�
�^�^�I�
&�
&�	'�"�S��Y�����%9�9�I��H�H�(�3�(�(�I�&����H�#�V�I�.�.�.�.�.�.�.�.��'*��$��%��,�,��E&�&�&�&�&�&�&�&�&�&�&�&�&�&�H#�F�8�,�,�,�,�,�,�,�,��&)��#��M&�&�&�&�&�&�&�&�&�&�&�&�&�&�&�&�&�&�&�&�&�&�&�&����&�&�&�&�&�&s�BC%�9C%�%
C/�2C/c#�K�t��}|�|t|��z}t�dt	|����|D]*}tj|��st||||��V��+dS)z�
    ping_timeout: specified in config timeout time (s) for request
    semaphore: semaphore obj to handle asyncio tasks
    ping_target_domains: mostly needed for re-pinging error domains
    Nz Those domains will be pinged: %s)r�setri�debugrkr�is_domain_ignoredr�)r�r�r��ping_target_domains�domains�domains      r/�	executorsr��s������m�m�G��&��3�*�+�+�+��
�L�L�3�S��\�\�B�B�B��A�A���$�V�,�,�	A��v�|�W�i�@�@�@�@�@��A�Ar1c���K�tt||||����}t|��dkrgSd�|D��}	tjtj|�|����d{V��S#tj$rjg}|D]*}|���s|����+|D]3}	|�	|�d{V�����#tj
$rY�0wxYw|cYSwxYw)Nrc�6�g|]}tj|����SrW)�asyncio�create_task)r,�coros  r/rJz scrape_sites.<locals>.<listcomp>�s#��>�>�>�4�W�
 ��
&�
&�>�>�>r1r�)�listr�r�r��wait_for�gatherr��done�cancel�append�CancelledError)	�ping_site_timeout�
ping_intervalr�r�r��
coroutines�tasks�results�tasks	         r/�scrape_sitesr��s1�����i� 1�7�I�GZ�[�[�\�\�J�
�:���!����	�>�>�:�>�>�>�E���%�g�n�e�&<�m�T�T�T�T�T�T�T�T�T�T����
�
�
�*,���	�	�D��9�9�;�;�
����
�
�
���	�	�D�
����T�z�z�z�z�z�z�*�*�*�*���)�
�
�
���
��������
���s7�'A,�,AC%�/C�
C%�C�C%�C�C%�$C%c
�|�t��}i}g}t|��5}|D]�}|�t|jd|j|j�����|�|jp|j��|jdkr|j||j<�s|�|j����|�	t���tj�
|�����td���d���ddd��n#1swxYwYt!��}|�|t|��z}t|��|z
}	t|��5}|	D]&}
|�t|
d������'	ddd��n#1swxYwY|S)	a
    - obtains all scrape coro results from asyncio tasks
    - saves ping results to ScrapeResult table
    - updates 'is_resolved' field in DomainAlerts table, in case
      error domain`s status code was changed to 200
    - returns domains with non-200 status code code
    T)�website�is_finishedrwrxr�)�is_resolvedF)�synchronize_sessionN)r�r�)r�r�addr	rvrwrxryr��queryr
�filterr��in_�update�dictr)�engine�pingedr��finished_domains�errors_domains�resolvedr��resultr��unfinished_domains�
unfinisheds           r/�manage_ping_resultsr��sY���u�u���N��H�	�v�	�	�G�'��	,�	,�F��K�K���
� �$�2�!'�!8�	���
�
�
�
� � ��!7�!E�6�:�F�F�F��#�s�*�*�-3�-A��v�z�*�*�����
�+�+�+�+��
�
�l�#�#�
�V�L�(�,�,�X�6�6�
7�
7�
�V�D�T�*�*�*��V�
F�
F�
F�!G�G�G�G�G�G�G�G�G�G�G����G�G�G�G�"�m�m�G��&��3�*�+�+�+���W���(8�8��	�v�	�	��'�,�	�	�J��K�K��"�!����
�
�
�
�	��������������������s$�C>D,�,D0�3D0�:*F1�1F5�8F5c	���tj��tt���z
}t	|��5}|�tj���tj�	t|�������ttj
|ktjdk����}d�|D��cddd��S#1swxYwYdS)z�
    - gets websites that must NOT be included in alert email:
          less than ERROR_DOMAINS_ALERT_INTERVAL passed or is_resolved marker was not
          changed from last alerting
    )�hoursFc��g|]	}|j��
SrW�r��r,�rows  r/rJz%get_recent_alerts.<locals>.<listcomp>s��8�8�8����8�8�8r1N)r�nowrrrr�r
r�r�r�r��keysr�
alert_timer�)r��
alert_domains�repeat_intervalr��recently_alerteds     r/�get_recent_alertsr�
s���l�n�n�y�7S�'T�'T�'T�T�O�	�v�	�	�9�'�"�=�=��)=�>�>�
�V�L�(�,�,�T�-�2D�2D�2F�2F�-G�-G�H�H���/�/�A�$�0�E�9�;�;�<�<�	�9�8�'7�8�8�8�9�9�9�9�9�9�9�9�9�9�9�9����9�9�9�9�9�9s�B%C+�+C/�2C/c	�Z�t�dtt|���������d�|���D��}t
tjtj	d|itj������|S)z�
    prepares needed error report object with error domains
    to be alerted and sends this mail
    returns alerted domains
    zAlerts will be sent for %sc
��g|]W\}}t|d�ttt	|������t|�������XS)z, )rv�code�count_errors)r�join�maprkr�r�)r,r��codess   r/rJzalert.<locals>.<listcomp>$sh�����
�F�E�	�����3�s�C��J�J�/�/�0�0��U���	
�	
�	
���r1�error_report��target_email�
from_email�report�notification_type)
rirjrkr�r�rhrrr�r�r�ALERT�notify)�domains_datar�s  r/�alertr�s����K�K�,��D��*�*�,�,�-�-�.�.�0�0�0���*�/�/�1�1�
���L�
��%��>��L�
�5�:�
<�<�<�=C�F�H�H�H��r1c	�`��t||����fd�|���D��}|sOt�dt	t|�������t��dSt|��tj
��}t|��5}|�t���tj�����}d�|D��}|D]�}||vr^|�t���tj|k���t'|d������d|�t||�������	ddd��dS#1swxYwYdS)aJ
    - gets recently alerted domains (those that must not be alerted again)
      and does not include them for alerting
    - calls alerting for left domains
    - updates DomainAlerts table:
      if website was not alerted -> adds new record
      if website was alerted before -> updates alert time and is_resolved marker
    c�$��i|]\}}|�v�	||��
SrWrW)r,rcrdr�s   �r/rez flush_alerts.<locals>.<dictcomp>@s*���\�\�\���A�!�K[�B[�B[��1�B[�B[�B[r1zDAll domains "%s" were alerted or still not resolved in last %d hoursNc��g|]	}|j��
SrWr�r�s  r/rJz flush_alerts.<locals>.<listcomp>Ns��0�0�0����0�0�0r1F)r�r�)r�r�)r�rhrirjrkr�r�rr�rr�rr�r
�
with_entitiesr��allr�r�r�r�)	r�r��domains_to_alertr�r��websites�urlsr�r�s	        @r/�flush_alertsr�6s����)���?�?��\�\�\�\��)<�)<�)>�)>�\�\�\�������Z���]�/�/�1�1�2�2�3�3�0�	2�	2�	2�	��	�
����
�,�.�.�C�	�v�	�	�J�'��=�=��.�.�
�]�<�/�
0�
0�
�S�U�U�	�1�0�x�0�0�0��&�	J�	J�F���~�~��
�
�l�+�+��V�L�0�F�:�;�;��V�D�C�U�C�C�C�D�D�D�D����L��C�H�H�H�I�I�I�I�
	J�J�J�J�J�J�J�J�J�J�J�J�J����J�J�J�J�J�Js�4C!F#�#F'�*F'c�0�|rtjjrdSdS)NTF)r�alert_notifications_enabled)�
error_domainss r/�should_be_repingedr�Xs ������<���t��5r1c��tj�t��r5t�d��tjt��dSdS)NzCloudLinux license was updated)�os�path�existsrrirj�removerWr1r/�cleanup_farewell_letter_markerr�^sM��	�w�~�~�<�=�=�:����4�5�5�5�
�	�8�9�9�9�9�9�:�:r1c�d�	tj�t��s_t�d��t
tjtj	itj������dSdS#t$rt�d��YdSwxYw)z@
    Sends farewell letter once (if it was not sent before)
    z/Going to send last email about expired license!r�z$Error while managing farewell letterN)r�r�r�rri�warningrrr�r�r�FAREWELLr��	Exception�	exceptionrWr1r/�manage_license_farewellrds���	A��w�~�~�D�E�E�	P��N�N�L�M�M�M�� �-��>��"<�"E�	
G�
G�
G�HN�v�x�x�x�x�x�
	P�	P���A�A�A����?�@�@�@�@�@�@�A���s�BB�$B/�.B/c
��K�tj��}tjjdz}	t��t	|��t��}t
tjdd��rtj|���}nt|t�����}tj|���4�d{V��}tj
|��}ttjj|||���d{V��}t!||��}	t#|	���r�t$�dt)|	����	tjjdz}|t*kr�tj��|z
}
tjt/t1t*|
z
d
��d�����d{V��|
t*krottjj||||	�������d{V��}t!|||	�����}t5|	|��}
nnt5||	��}
|	}n��|
rWt$�d
t)t7|
���������t9||
��ng}	ddd���d{V��n#1�d{V��swxYwYn*#t:$rt$�d��YnwxYw	tjjdz}tj��|z
}t/t1||z
d
��d��}tj|���d{V��||krn�mns#	tjjdz}tj��|z
}t/t1||z
d
��d��}tj|���d{V��||krn�mwxYw|S)aF
    Scanner logic:
    1. Scrapes domains and obtains ping results;
    2. Manage ping results (e.g: saving to DB)
    3. In case error domains found -> start re-pinging
       Re-pinging:
         - in min(ping_interval, 5 mins)
         - flush alerts if needed
    4. Sleep for ping_interval until next ping iteration
    �<�allow_private_targetsF)�limit)r	�resolver)�	connectorNz8Those domains are unsuccessful: %s 
 Try to re-ping themTr�
)r�z1Domains with unsuccessful status code found: "%s"zError during ping iteration!)r�rr�r�r
r�r+r@�TCPConnectorrYr>�
ClientSessionr��	Semaphorer�r�r�r�rirjrkr�sleep�minr�r�rr�r�rr)�previously_erroredr�r��ping_interval_seconds�connections_limitrr�r��ping_resultr��elapsed_for_ping�ping_retry_result�retry_errorsr��elapsed�
sleep_times                r/�scrape_iterationrts�����
�I�K�K�E��G�1�B�6��<�&�(�(�(��� � � �0�2�2���3�7�3�U�;�;�	]��,�3D�E�E�E�I�I�)�0A�M�O�O�\�\�\�I��(�9�=�=�=�*	(�*	(�*	(�*	(�*	(�*	(�*	(���)�*;�<�<�I� ,���$�&;�W�i�!�!�������K�0���D�D�M�"�-�0�0�"
(����W�Y\�]j�Yk�Yk�l�l�l��,/�G�,A�B�,F�)�,�/P�P�P�+/�9�;�;��+>�(�%�m���$E�HX�$X�Z[� \� \�^`�a�a����������,�.O�O�O�6B� #�� 4�6K�W�V_�4A�4F�4F�4H�4H�7�7�7�1�1�1�1�1�1�-�,?� &�(9�=�;M�;M�;O�;O�,�,�L�-6�m�\�,R�,R�M�!�P�)2�2D�m�(T�(T�
�-:�*��/�2!�8��K�K� S� #�D��);�);�)=�)=�$>�$>� ?� ?�A�A�A� ���7�7�7��&(�"�'�U*	(�*	(�*	(�*	(�*	(�*	(�*	(�*	(�*	(�*	(�*	(�*	(�*	(�*	(�*	(�*	(�*	(�*	(�*	(�*	(�*	(�*	(�*	(����*	(�*	(�*	(�*	(���V�9�9�9����7�8�8�8�8�8�9����	�$'�G�$9�B�$>�!��i�k�k�E�)�G��S�!6��!@�!�D�D�b�I�I�J��-�
�+�+�+�+�+�+�+�+�+��.�.�.��
	���
	�$'�G�$9�B�$>�!��i�k�k�E�)�G��S�!6��!@�!�D�D�b�I�I�J��-�
�+�+�+�+�+�+�+�+�+��.�.�.��
	�	�	�	�	��sP�BK�?G$J6�$K�6
K�K�K�K�M!�$K/�,M!�.K/�/M!�!A0Oc��K�t��}g}d}	t�����rd}t||���d{V��}n~|dz
}|dkrt�d��|dkrt
��d}tjjdz}t|t��}tj|���d{V����)z�
    Main loop for wmt_scanner_solo service
    each 'while: True' iteration returns errored domains
    (domains that responded with non-200 status code)
    rTNr~z)Seems your CloudLinux license is expired!�r)
rr �get_license_statusrrirrrr�rrr�r)r�r�license_attemptrrs     r/�scrape_loopr �s������
�
�F����O�,��!�!�4�4�6�6�	,��O�'7�8J�F�'S�'S�!S�!S�!S�!S�!S�!S����q� �O��!�#�#����J�K�K�K��!�#�#�'�)�)�)�"#��$'�G�$9�B�$>�!��2�4G�H�H�J��-�
�+�+�+�+�+�+�+�+�+�,r1�__main__zPID: %szweb-monitoring-toolzcl-web-monitoring-toolr*)Zr�r3r�rqr@�signalr��concurrent.futuresr�rr�typingrr�dataclassesr�
sqlalchemyr�wmt.dbr	r
rrr
�wmt.common.utilsrrrr�clsentryr�clsentry.utilsr�wmt.common.constrrrrrrrr�wmt.common.notificationrr�wmt.common.reportr�
wmt.commonr�	cllicenser ri�	frozensetr7�
ip_networkr9�boolr<�abc�AbstractResolverr>r
rYrorsrur{r�r�r�r�r�r�r�r�r�r�rrr rSrk�getpid�pidrj�SIGUSR1�SIGTERM�get_event_loop�loop�run_until_completerWr1r/�<module>r<s:����������	�	�	�	�
�
�
�
�����
�
�
�
���������(�(�(�(�(�(�(�(�!�!�!�!�!�!�!�!�!�!�!�!�!�!���������������������U�T�T�T�T�T�T�T�T�T�T�T�'�'�'�'�'�'�*�*�*�*�*�*�	�	�	�	�	�	�	�	�	�	�	�	�	�	�	�	�	�	�	�	�I�H�H�H�H�H�H�H�)�)�)�)�)�)�������*�*�*�*�*�*�	��m�	$�	$��#��#`�#`�#`�a�a��&��%�o�6�6���D�����*%�%�%�%�%�G�K�0�%�%�%�.
F�
F�
F�
F�
F��,�
F�
F�
F�9�9�9�����*�*�*�*�*�*�*���*��c�����/�/�/�dA�A�A�A���jn�o�kA�����('�'�'�'�T9�9�9����4J�J�J�D���:�:�:�
A�
A�
A� K�K�K�\,�,�,�6�z���

�#�i�b�i�k�k�
�
�C�
�K�K�	�3����	�	�=�#�	.�	.�/�/���0�*�?�+C�D�D�%�	'�	'�	'�	��
�f�n�k�2�2�2���
�f�n�h�/�/�/�%�w�%�'�'�������
�
�.�.�.�/�/�/�/�/�/�/�/�/�/�/�/����/�/�/�/�/�/��s�;A/G7�7G;�>G;