Asynchronous operation without window freezing

I have a mobile application on kivy, it works well until I try to implement asynchronous execution of site parsing in it, so that the main gui kivy window does not hang and I can interact with it during the execution of a long process. Then the second file (ckecker), site parsing, does not work correctly, especially where the wait for js execution is performed. It either does not wait for js to execute, or waits but with an error somewhere in the process of waiting and outputting the result, as a result the result is incorrect.

I attached a code example without asynchronous execution. I repeat, the second script (checker) works normally only in this mode (when a function is called in it via asyncio.run(), and inside it new_loop) otherwise it does not work normally with the site on which the js script is executed. (Selenium is not recommended).

I also processed my question in chatgpt - it didn’t help. I tried the following options:

  • standard kivymd capabilities (from kivymd.utils.asynckivy import start, sleep)
  • asyncio (and the entire main.run() in it and purely the load function or only the second script in the call)
  • threading / concurrent.futures (purely through them and together with asyncio)
  • asynckivy (which is installed separately) also didn’t work
    Nothing helped. The application managed to start working asynchronously, but the result from the second script was then executed/came incorrectly.

main.py

from kivy.lang import Builder
from kivy.core.clipboard import Clipboard as Cb
from kivy.core.window import Window
from kivy.properties import ObjectProperty
from kivy.utils import platform
from kivy.clock import Clock
from kivy.uix.modalview import ModalView
from kivy.uix.label import Label
from kivymd.app import MDApp
from kivymd.uix.menu import MDDropdownMenu
from kivymd.uix.button import MDRoundFlatIconButton
from kivymd.uix.scrollview import MDScrollView
import cheker as checkN

KV = '''
<ContentNavigationDrawer>
    MDList:
        OneLineIconListItem:
            text: "Home"
            theme_text_color: "Custom"
            text_color: 1, 1, 1, 1
            on_press:
                root.nav_drawer.set_state("close")
                root.screen_manager.current = "scr 1"
            IconLeftWidget:
                icon: 'home'
                
MDScreen:
    MDBoxLayout:
        id: home
        orientation: 'vertical'
        MDTopAppBar:
            pos_hint: {"top": 1}
            elevation: 4
            title: "OSINT"
            left_action_items: [["menu", lambda x: nav_drawer.set_state("open")]]
        MDNavigationLayout:
            id: navig
            MDScreenManager:
                id: screen_manager
                MDScreen:
                    id: scr_1
                    name: "scr 1"
                    MDBottomNavigation:
                        id: sc_0
                        selected_color_background: "orange"
                        text_color_active: "lightgrey"
                        MDBottomNavigationItem:
                            id: sc_1
                            name: 'screen 1'
                            text: 'Home'
                            icon: 'home'
                            MDRoundFlatIconButton:
                                id: lst1
                                text: 'Choose option'
                                pos_hint: {'center_x':0.5, 'center_y':0.7}
                                on_release: app.menu.open()
                            MDTextField:
                                id: txt
                                hint_text: ''
                                helper_text: ''
                                helper_text_mode: 'persistent'
                                text: ''
                                mode: 'rectangle'
                                pos_hint: {'center_x':0.5, 'center_y':0.45}
                                size_hint: 0.28, 0.15
                            MDRoundFlatIconButton:
                                id: startbutton
                                text: 'Start'
                                pos_hint: {'center_x':0.5, 'center_y':0.20}
                                on_release: app.start()
                        MDBottomNavigationItem:
                            id: sc_3
                            name: 'screen 3'
                            text: 'Result'
                            icon: 'text'
                            ScrollView:
                                MDLabel:
                                    id: scroll
                                    markup: True
                                    text: ''
                                    theme_text_color: "Custom"
                                    text_color: 1, 1, 1, 1
                                    font_size: '15sp'
                                    multiline: True
                                    text_size: self.width, None
                                    size_hint_y: None
                                    height: self.texture_size[1]
                                    on_touch_down: app.linki(app.bufftext)
                                    on_ref_press: app.webopen(args[1])
            MDNavigationDrawer:
                id: nav_drawer
                radius: (0, 16, 16, 0)
                ContentNavigationDrawer:
                    screen_manager: screen_manager
                    nav_drawer: nav_drawer
'''

class MyApp(MDApp):
    def __init__(self, **kwargs):
        super().__init__(**kwargs)

        self.list_info = ['Number']
        self.dl = Window.width
        self.sh = Window.height

        self.screen = Builder.load_string(KV)

        menu_items = [
            {
                'id': 'm1',
                'text': f'{i}',
                'viewclass': 'OneLineListItem',
                'on_release': lambda x=f'{i}': self.menu_callback(x),
            } for i in self.list_info
        ]
        self.menu = MDDropdownMenu(
            caller=self.screen.ids.lst1,
            items=menu_items,
            width_mult=5,
        )

    def menu_callback(self, text_item):
        self.screen.ids.lst1.text = text_item
        self.screen.ids.txt.hint_text = text_item
        lst2 = self.screen.ids.sc_1
        self.menu.dismiss()

    def build(self):
        self.theme_cls.material_style = 'M3'
        self.theme_cls.theme_style = 'Dark'
        self.title = 'OSINT'
        return self.screen

    def start(self):
        self.screen.ids.sc_0.switch_tab('screen 3')
        self.root.ids.scroll.text = f'Please wait'
        Clock.schedule_once(lambda dt: self.load(), 1.0)

    def load(self):
        self.res = ''

        opt1 = self.screen.ids.lst1.text
        opt3 = self.screen.ids.txt.text

        if opt1 != 'Choose option':
            if opt1 == 'Number':
                self.res = checkN.parse_num(opt3)
        else:
            self.root.ids.scroll.text = 'Select/enter the correct data.'

        if self.res != '':
            self.root.ids.scroll.text = self.res
        else:
            self.root.ids.scroll.text = 'No results'

if __name__ == "__main__":
    MyApp().run()

cheker.py

import requests
from bs4 import BeautifulSoup
from fake_useragent import UserAgent
import asyncio
from requests_html import HTMLSession, AsyncHTMLSession

def parse_num(numbers):
    try:
        res1 = europe(numbers)
    except:
        res1 = f'{color.RED}[info]:{color.END}\nError internet/proxy_server/site. Try again.\n'

    try:
        res2 = asyncio.run(all_num(numbers))
    except:
        res2 = f'{color.RED}[whosenumber.info]:{color.END}\nError internet/proxy_server/site. Try again.\n'
        
    try:
        res4 = europe(numbers)
    except:
        res4 = f'{color.RED}[ua.tellows.org]:{color.END}\nError internet/proxy_server/site. Try again.\n'

    res = res1+res2+res4
    return res

async def all_num(numbers):
    uu='https://whosenumber.info'
    headers = {'User-Agent': f'{UserAgent().random}', }
    url = f'{uu}/'
    new_loop = asyncio.new_event_loop()
    asyncio.set_event_loop(new_loop)
    session = AsyncHTMLSession()

    response = session.get(f'{url}{numbers}', headers=headers, timeout=60)

    await response.html.arender(sleep=10)
    soup = BeautifulSoup(response.html.html, 'html.parser')
    try:
        divF_tag = soup.find('div', attrs={'style':'padding:10px; padding-left:20px;'})
        ttl = ''
        try:
            divS_tag = divF_tag.find('span', attrs={'id':'t_reiting'})
            ttl = divS_tag.get_text()
        except:
            pass
        comm = ''
        try:
            ij = 1
            for p_tag in divF_tag.findAll('div', attrs={'style': 'margin-left:15px;margin-top:10px;color:#6a6a6a; padding-bottom:10px;'}):
                try:
                    pp_tag = p_tag.get_text()
                    pp_tag = pp_tag.replace('<br/>', '')
                    pp_tag = pp_tag.replace('<br>', '')
                    pp_tag = pp_tag.replace('<b/>', '')
                    pp_tag = pp_tag.replace('<b>', '')
                    pp_tag = pp_tag.replace('</b>', '')
                    pp_tag = pp_tag.replace('</br>', '')
                    comm += f'    [{ij}]: {pp_tag}\n'
                    ij+=1
                except:
                    continue
        except:
            pass
        logger.info('all - there is a result')
        resF = (f'{color.RED}[whosenumber.info] - info:{color.END}\n'
                f'{color.BOLD}Rating: {color.END}{ttl}\n{color.BOLD}Comments: {color.END}\n{comm}\n')
    except:
        logger.info('all - result error')
        resF = (f'{color.RED}[whosenumber.info] - info:{color.END}\n'
                f'Error internet/proxy_server/site. Try again.\n')

    res = f'\n{resF}'
    await session.close()
    new_loop.close()
    return res

def europe(numbers):
    uu = 'https://ua.tellows.org'
    headers = {'User-Agent': f'{UserAgent().random}', }
    url = f'{uu}/num/'

    response = requests.get(f'{url}{numbers}', headers=headers, timeout=60)

    soup = BeautifulSoup(response.text, 'html.parser')
    try:
        ttl = ""
        comm = ""
        for divF_tag in soup.findAll('div', attrs={'id': 'userratings'}):
            try:
                for divS_tag in divF_tag.findAll('div', attrs={'class': 'col-md-4 mt-2'}):
                    try:
                        for h5_tag in divS_tag.findAll('h5'):
                            ttl += f"{color.BOLD}{h5_tag.get_text()}{color.END}\n"
                        for th_tag in divS_tag.findAll('th'):
                            tt = str(th_tag).replace("<th>\n                    ", "")
                            tt = tt.replace("                        ", "")
                            tt = tt.replace('<br/>\n<span class="small">', ": ")
                            tt = tt.replace("</span>\n</th>", "")
                            ttl += f"    {tt}\n"
                    except:
                        continue
            except:
                continue
        for comms_tag in soup.findAll('ol', attrs={'id': 'singlecomments'}):
            try:
                ii = 1
                comm += f"{color.BOLD}Comments: {color.END}\n"
                for comm_tag in comms_tag.findAll('div', attrs={'class': 'col comment-body'}):
                    try:
                        for pCom_tag in comm_tag.findAll('p', attrs={'class': 'mb-0'}):
                            comm += f"    [{ii}]{pCom_tag.get_text()}\n"
                            ii += 1
                    except:
                        continue
                    try:
                        jj = 1
                        for dCom_tag in comm_tag.findAll('div', attrs={'class': 'ccomment'}):
                            try:
                                comm += f"        [0{jj}]{dCom_tag.get_text()}\n"
                                jj += 1
                            except:
                                continue
                    except:
                        continue
            except:
                continue
        logger.info('Europe - there is a result')
        resF = (f'{color.RED}[ua.tellows.org] - info:{color.END}\n'
                f'{ttl}\n{comm}\n')
    except:
        logger.info('Europe - result error')
        resF = (f'{color.RED}[ua.tellows.org] - info:{color.END}\n'
                f'{color.BOLD}Info: {color.END}Error internet/proxy_server/site. Try again.\n')
    res = f'\n{resF}'
    response.close()
    return res

Please test the source code I provided and try to run asynchronous execution of kivy, so that the gui window does not freeze, but at the same time the second script is executed normally (the result came from parsing the site), especially in the function where the expectation of the site js is executed. For the test, the number that needs to be entered in the application: +37129502847. I took it from the parsed site, so that I would definitely get the result.

I don’t know if Kivy does sub-processing, but Python is a single threaded language. Even if you set up threads, it is still single threaded and just jumps between these threads. Async code is also executed on the same thread where the async event loop is. Unless you are doing sub-processing or have disabled the GIL, any sufficiently large computation will lock down your Python application.