Ongelmia UDP listener threadin kanssa (Python 2.7)

Anonyymi

Olen koodaamassa qRadarille plugaria erästä systeemiä varten. Koodi on tarkoitus tehdä Pythonilla (versio 2.7) ja REST-toteutukseen käytetään Flask-kirjastoa. Olen koodannut luokan, jonka yhtenä osana on erityinen listener-threadi (jonka erillinen manager-thread käynnistää). Toteutus käyttää pythonin socketia, mutta jostain syystä bindaus feilaa. Koodin alkuosa (jossa alustetaan ja bindataan socket) on seuraavanlainen:

try:
# Set timeout for socket to prevent thread from getting stuck on termination
self.listener_error = "Setting socket timeout failed"
socket.setdefaulttimeout(2)
# Initiate socket
self.listener_error = "Instantiating socket failed"
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)

# Enable address reuse
self.listener_error = "Enabling adress reuse failed"
s.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)

# Bind the socket to given address
self.listener_error = "Binding socket failed"
s.bind((self.ip_address, int(self.port)))
self.is_listening = True
self.is_connected = True

Muuttujaa listener_error käytetään tarkemman virheviestin välittämiseen kutsujalle. Kutsuttaessa tulee virhe "binding socket failed", joten jostain syystä bindaus ei toimi. Missä voisi olla vika ? ip_address ja port asetetaan ylemmällä tasolla ja asetusfunktiossa tarkistetaan niiden kelvollisuus, joten siitä ei pitäisi olla kyse. Voin toki varmuuden vuoksi lisätä ne virheilmoitukseen.

Koodissa muuttujaa is_listening käytetään managerissa tarkistamaan onko listener käynnissä (en tunne Pythonia niin hyvin, enkä äkkiseltään keksinyt parempaa tapaa) ja is_connectedia tarkastellaan ylemmän tason funktiossa, jotta selviäisi onnistuiko kytkentä tietyn ajan sisällä.

6

217

    Vastaukset

    Anonyymi (Kirjaudu / Rekisteröidy)
    5000
    • Anonyymi

      Tuohan on erinomainen.

      T. Urpo

    • Anonyymi

      Ei ole kovin helppoa jäljittää virheen syytä kun ainoa tieto on, että virhe tulee nimen omaan bindissa (seuraavana vaiheena on varsinainen vastaanottosilmukka ja sitä ennen virheilmoitukseksi vaihdetaan "internal operation error", mutta siihen asti ei selvästikään päästä). Olisiko mahdollista esimerkiksi except-lohkossa hakea tarkempi virheviesti ja lisätä se perusvirheeseen ja jos on, miten ?

    • Anonyymi

      Sellainen lisäys vielä, että lisäsin except lohkoon määrittelyn "...socket.error as msg" ja except-lohkon alle rivin

      self.listener_error = self.listener_error ": " msg

      Tuon pitäisi antaa vielä tarkempi virhekuvaus, mutta jostain syystä mitään ei tule. Edes kaksoispistettä ei tule. Ikään kuin tuossa ei tulisikaan exceptionia vaan listener-thread jäisi jotenkin jumiin tuohon bindiin ja ylemmän tason funktiossa tulee timeout (odotetaan silmukassa 100 kertaa 0.1 sekunnin delayn kanssa, siis yhteensä 10 sekuntia). En tiedä onko tuollainen odotus oikea tapa, mutta en keksi parempaakaan.

      Listenerin aktivointi toimii siis siten, että ylemmän tason funktio kutsuu listener-luokan (johon manageri ja listener-funktio kuuluvat) funktiota, joka asettaa IP-osoitteen ja portin. Funktio tarkistaa, että arvot ovat kelvolliset ja jos ne ovat asettaa ne muuttujiin "uusi IP-osoite" ja "uusi portti". Manager-threadi (joka siis pyörii taustalla jatkuvasti omana threadinaan) huomaa, että IP-osoite tai portti on muuttunut ja instantioi listener-threadin. Jos kaikki menee oikein listener-thread tekee alustukset ja asettaa sitten tuon is_connected-lipun. Samalla se funktio, joka asetti alun perin arvot on odottamassa silmukassa. Jos se huomaa is_connected-lipun asettuneen (se nollattiin alussa) ennen ajan loppumista se palauttaa "OK" statuksen, muuten virheen.

      Testasin myös soketin alustamista suoraan palvelimella Pythonin interaktiivisessa tilassa. Siis:

      # python
      >>> import socket
      >>> s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
      >>> socket.setdefaulttimeout(2)
      >>> s.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
      >>> s.bind(("xxx.xxx.xxx.xxx", 4740))
      >>> data = s.recvfrom(1000)
      >>> print data
      >>> "bla bla bla"

      Eli interaktiivisessa moodissa toimi ilman ongelmia. IP-osoite oli sama kuin softalle välitetty. Alkaa vaikuttaa aika mysteeriseltä.

      Voisiko olla, että tuo int() portin numeron määrittelyssä on jotenkin väärin tai bind ei jotenkin tykkää siitä ? Molemmat parametrit (IP-osoite ja portti) välitetään funktiolle normaalin HTML-formin kautta input-kentistä POST-metodin välityksellä (ja näyttäisivät kyllä olevan oiken virheilmoitukseen lisäämäni tulostuksen perusteella).

      • Anonyymi

        Ota try hetkeksi pois, niin jysähtää kunnolla ja saat pythonin error viestin.

        Laita väliprinttejä ja tarkista muuttujien, kuten ip osoitteen ja portin arvot, että ovat järkevät. Tai vaihtoehtoisesti aja koodi rivi kerrallaan debuggerilla.


      • Anonyymi

        Niin ja toki voi olla.

        type() funktiolla voit tarkistaa, mitä tyyppiä muuttujasi ovat.


    • Anonyymi

      Itse asiassa vaikuttaa siltä, että tässä tapauksessa qRadar blokkaa bindauksen (luultavasti turvalisuussyistä). Muutin except-rivin muotoon

      except socket.error as msg:

      ja lisäsin except-lohkon alle rivin

      self.socket_error = self.socket_eror ": " msg

      Jolloin exceptin antama virheviesti pitäisi olla mukana virhetekstissä (virheteksti tulee näkyviin ja se osoittaa, että ongelma on nimen omaan bindissa), mutta sitä ei näy. Virhesivulla näkyy ainoastaan perusvirhe ("bind failed"), ei kaksoispistettä tai lisätekstiä.

      Teoriani on, että qRadar blokkaa turvallisuussyistä omien sovellusten bindausyritykset, mutta ei kuitenkaan aiheuta poikkeusta. Kun bindaus blokataan kutsuvassa koodissa tulee timeout ja listenerin aktivointi feilaa.

      Kenties qRadarin asetuksista löytyy keino sallia bindaus. Jos ei, niin täytyy keksiä jotain muuta, esmes aktivoida serverillä ulkoinen listener (luin jostain sovelluksesta joka voisi sopia, muistaakseni nimeltään socat) ja ohjata sen tulostus tiedostoon, josta haetaan tiedot...

    Ketjusta on poistettu 0 sääntöjenvastaista viestiä.

    Luetuimmat keskustelut

    1. Ei sua enään tunnista

      Kun olet vanhentunut ja lihonut.
      Ikävä
      90
      5177
    2. Huomenet naiselle

      Harmittaa ettei ehkä nähdä enää koskaan. Näillä mennään sitten.
      Ikävä
      53
      4066
    3. Mikä kaivatussasi kolahti?

      Mikä oli erityistä?
      Ikävä
      25
      2179
    4. Etsin vastaantulevista sua

      Nyt kun sua ei oo, ikävöin sua niin v*tusti. 😔Jokaisesta etsin samoja piirteitä, samantyyppistä olemusta, samanlaista s
      Ikävä
      27
      1887
    5. Kaikesta muusta

      Mulla on hyvä fiilis. Mä selviän tästä ja sit musta tulee parempi ihminenkin. Ainut, mitä mun pitää nyt välttää on se ko
      Ikävä
      16
      1795
    6. Hyvää yötä.

      Miten äkäpussi kesytetään? 😉 pus
      Ikävä
      15
      1434
    7. Ajatteletko koskaan

      Yhteisiä työvuosia ja millaista silloin oli? Haluaisin palata niihin vuosiin 🥹
      Ikävä
      28
      1250
    8. Tekis mieli lähestyä sua

      Mutta pelkään että peräännyt ja en haluis häiritä sua... En tiedä mitä tekisin olet ihana salaa sua rakastan...💗
      Ikävä
      20
      1036
    9. T, miten mun pitäis toimia

      Olen niin toivottoman ihastunut suhun...ollut jo liian,monta,vuotta. Lähestynkö viestillä? Miten? Sun katse...mä en kest
      Ikävä
      43
      981
    10. Kyllä hävettää!

      Olla taivalkoskelta jos vuoden taivalkoskelainen on tuommoinen tumpelo.
      Taivalkoski
      21
      980
    Aihe