Työasemien välillä liikkuva kisa-aikainen data on versiosta 9.0.1 alkaen kokonaan XML:ää. Kaikki tietokentät ovat tekstitietoa ja siis editoitavissa ja vaikkapa notepadillä luettavissa.
Versiossa 9.4.1 (Juhannus 2019) poistin tägeistä kaikki #-merkit ja skandimerkit noudattaakseni paremmin XML-määrittelyitä, joten kaikissa samalla kertaa käytössä olevissa työasemissa on ajettava samaa ExTime-versiota.
Versiosta 10 eteenpäin sarjakohtaisia sanomia kehystää s10- ja urheilijakohtaisia u10-tagi. Ja jos jatkossa rakenne selvästi muuttuu, 10 vaihtuu version mukaan. 11-versio varmaan sisältää sitten ajat FIS-vaatimusten mukaan sekunnin 10000 osina, niin hullulta kuin se kuullostaakin.
Versiosta 10.5. eteenpäin ExTime tunnistaa tietoa vastaanottavan tai tietoa lähettävän työaseman. Tämä tehdään uudella alive- sanomalla. Kun ExTime avataan vastaanottamaan tietoa niin jokainen työasema, joka avaa lähetysyhteyden, saa sanoman <alive>NIMI</alive>, jossa NIMI on tämän vastaanottavan työaseman laitenimi. Ja tietoa lähettävä työasema lähettää heti ensimmäisenä sanomanaan oman nimensä <alive>MINUN NIMI</alive>. Jos vastaanottava työasema ei saa mitään tietoa viiteen minuuttiin, se kysyy lähettävältä tilannetietoa taas tällä alive-sanomalla, johon tämä lähettävä vastaa taas omalla nimellään, että hiljaisuus johtuu kisan suvantovaiheesta.
Jokaisessa sanomassa on alussa 10 tavun mittainen otsikkokenttä, jossa on kolme eri tietoa
Esimerkki:
0286.00213
Tulossa oleva sanoma on 286 tavua pitkä ja se on järjestyksessä 213. Sanomaan ei tarvitse lähettää kuittausta, koska viides merkki on piste tai tyhjä (esimerkissä piste luettavuuden takia). Jos viides merkki olisi X tiedon vastaanottajan pitäisi vastata sanomalla
0012.00213OK
eli 12 merkin vakiomittaisella sanomalla, jossa on sama järjestysnumero ja OK, kun sanoma on luettu onnistuneesti. Testit ovat osoittaneet, että kuittausta ei tarvita. TCP/IP-protokolla itsessään sisältää jo riittävästi tarkistuksia.
Esimerkin otsikkoa seuraa siis 286 tavua XML-dataa.
Päivitystietoja edeltää siis <s10> tai <u10> riippuen siitä, onko kyse sarjan (s10) vai urheilijan (u10) tietojen päivityksestä. Itse päivitystapahtumassa on kaksi kokonaisuutta:
Kun kilpailijasta tai sarjasta lähetetään päivitystietoa, kohdistustietoa (<search>..</search>) seuraa vain nyt muuttuneet tiedot. Kohdistustietona on sarjatiedoissa <Class>SNSNSNSN</Class>, jossa SNSNSNSN on maksimissaan kahdeksan merkkiä pitkä sarjan tunnus. Urheilijatiedoissa kohdistustietoja on kaksi, sarjan tunnus ja kilpailijan numero. Samassa sarjassa ei siis saa olla kahta saman numeroista urheilijaa, kilpailussa kylläkin.
Kelloajan koodauksessa desimaalien määrä (siis sekunnin osien) voi vaihdella. Samoin tunnit saattavat puuttua, joten koodaa kelloajan käsittelyt sallimaan vaihtelevan mittaisia kenttiä. Itse lasken ensin, montako erottimilla (kaksoispiste, piste, pilkku tai välityhjä) erotettua kenttää on. Jos kenttiä on vähemmän kuin 4, niin silloin tunteja ei ole mukana. Jos kenttiä on vain kaksi, mukana ovat sekunnit ja sekunnin osat. Jos kelloajassa on esimerkiksi lukema '70.5', niin oma muuntorutiinini tuntee tuon kelloaikana yksi minuutti, 10 sekuntia ja 5 sekunnin kymmenystä. Saman tuloksen antavat merkinnät '1.10,5', '70:5', '70,5' ja '70 5'.
Alapuolen esimerkissä sarjan M12 urheilija, jonka lähtönumero on 21. Sarjan tägi on siis Class ja urheilijan sarjan ClassID.
<search> <ClassID>M12</ClassID> <Bib>21</Bib> <action>Insert</action> </search>
ExTime mahdollistaa koko kilpailun datan lähettämisen bulkkisiirtona. Tämä siirto voidaan kohdistaa vain tietylle työasemalle. Bulkkisiirtossa lähetetään kaikki ne tietokentät, joissa on rasti XML-tietojen määrityksessä. Tietueen alussa on aina em. kohdistustieto, mutta sarjan tiedoissa mukana ei ole toimenpidetietoa (<action> </action>) vaan itse datatiedoissa on sarjan yksilöllinen indeksi (class unique id) ((tägi <Cluniqid>). Jos indeksi on tyhjä (siis nolla), kyseessä ei ole kilpailun sarja vaan kilpailun perustiedot. Sarjan päivitystiedot ovat tägin Classinfo sisällä, urheilijan taas tägin Competitor reunustamina.
Kelloajat ovat yleensä valtakunnan aikaa (GPS-aikaa), eli tapahtumahetken kelloaika. Urheilijan omat ajat (väliaika, loppuaika) saadaan sitten vähentämällä tästä kelloajasta urheilijan todellinen lähtöaika.
Tyypillisin päivitystieto on urheilijan loppuajan välittäminen, siitä esimerkki alla, alussa on kohdistustieto, jossa toimenpiteenä Update, jota seuraa päivittävät tiedot maaliintuloheti, loppuaika ja sijoitus (listaus on lukemisen helpottamiseksi muotoiltu eri riveille sisäänvetojen kanssa):
<u10> <search> <ClassID>NYL</ClassID> <Bib>105</Bib> <action>Update</action> </search> <Competitor> <Finishtime>10:30:57.300</Finishtime> <Totaltime>27:27.3</Totaltime> <Rank>1.</Rank> </Competitor> </u10>
Naisten sarjan NYL urheilija numerolla 105 leikkasi maalilinjaa ajassa 10:30:57.300 ja sai sillä hetkellä sijoituksen 1. Loppuaika 27:27.3 ja sijoitus on laskettu sanomaan valmiiksi, mutta vastaanottava pää laskee ne aina myös itse aikaisemmin saamansa lähtöajan mukaan.
Seuraavassa esimerkissä on kilpailijan tilatietoa muutettu, tilatieto on 32 bitin sana, joka tulostetaan 8 merkin hexa-tietona. Tilatiedoista on erillinen selostus, yleisimpiä ovat keskeytys, poisjäänti, hylkäys ja todellinen, rekisteröity lähtöaika.
<u10> <search> <ClassID>NYL</ClassID> <Bib>103</Bib> <action>Update</action> </search> <Competitor> <Flags>00080000</Flags> </Competitor> </u10>
Tässä esimerkissä on taas kyseisen sarjan (NYL) nimeä muutettu:
<s10> <search> <Class>NYL</Class> <action>Update</action> </search> <Classinfo> <Classname>Naiset, FIS-sarja</Classname> </Classinfo> </s10>
Uusi nimi sarjassa NYL on nyt Naiset, FIS-sarja.
Toimenpidemahdollisuuksia (action) on kolme:
Tapahtumatiedolla ExTimessä tarkoitan väliaikoihin, ammuntatuloksiin tai kierrosaikoihin liittyviä tietoja. Niistä erotuksena urheilijan lähtöaika ja maalintuloaika ovat päivitystietoja. Tapahtumatiedot alkavat intermediate-tägillä. Jos tapahtumassa on mukana Competitor-tägi, on kyseessä urheilijan tapahtumatieto, ilman sitä tapahtuma koskee class-tagissä määritettyä sarjaa.
Tässä esimerkissä on sarjaan MYL lisätty 2 väliaikapistettä (type = V) 5.000 ja 10.000 km kohdille, hitain ja nopein mahdollinen väliaika on määritetty, sektori, johon väliaika kuuluu, sekä väliaikaan liittyvä aikakanava ja/tai käytetyn lukijan (Emit ECB-kellolaite) postcode :
<intermediate> <class>MYL</class> <point>5.000</point> <fast>12:45.0</fast> <slow>17:10.0</slow> <sector>P</sector> <postcode>V:</postcode> <type>V</type> </intermediate> <intermediate> <class>MYL</class> <point>10.000</point> <fast>20:00.0</fast> <slow>24:10.0</slow> <sector>K</sector> <postcode>V:</postcode> <type>V</type> </intermediate>
Niitä seuraa välittömäsi sarjan päivitetyt tiedot, esimerkissä on muuttunut vain sarjan väliaikapisteiden lukumäärä kahdeksi:
<s10> <search> <Class>MYL</Class> <action>Update</action> </search> <Classinfo> <Intermedpoints>2</Intermedpoints> </Classinfo> </s10>
MYL-sarjan urheilija numerolla 156 leikkaa sitten 5 km väliaikapisteen (type=V) kelloajassa 10.46.54,2 ja hiukan myöhemmin 10.km kohdan kellon ollessa 11:06:57,6:
<intermediate> <class>MYL</class> <Competitor>156</Competitor> <point>5.000 km</point> <elapsed>10:46:54.2</elapsed> <type>V</type> </intermediate> <intermediate> <class>MYL</class> <Competitor>156</Competitor> <point>10.000 km</point> <elapsed>11:06:57.6</elapsed> <type>V</type> </intermediate>
Ja kellon ollessa 11:26:58:000 samainen urheilija numerolla 156 saapuu maaliin ja sijoittuu sarjassa MYL sijalle 21:
<u10> <search> <ClassID>MYL</ClassID> <Bib>156</Bib> <action>Update</action> </search><Competitor> <Finishtime>11:26:58.000</Finishtime> <Rank>21.</Rank> </Competitor> </u10>
Tässä esimerkissä on sarjasta MYL poistettu väliaikapiste 10.000 km kohdalta:
<intermediate> <class>MYL</class> <point>10.000 km</point> <action>Delete</action> </intermediate>
Jota sitten seuraa sarjan päivitetyt tiedot:
<s10> <search> <Class>MYL</Class> <action>Update</action> </search> <Classinfo> <Intermedpoints>1</Intermedpoints> </Classinfo> </s10>
Ampumahiihdossa/juoksussa ammuntatapahtumat tunnistetaan tyypillä B, esimerkissä naisten sarjan (NYL) urheilija (nro=4) saa viisi (value=5) ohilaukausta ampumapaikalta, joka on lähdöstä 1 km päässä. Ampumapaikat tunnistetaan etäisyydellä lähdöstä, jolloin väliaikapalvelussa osataan laskea vain siihen asti kertyneet sakot mukaan väliaikaan. Versiosta 9.4.1 lähtien sanomassa tulee mukana myös kaikki urheilijan siihen mennessä saamien (=kirjattujen) sakkojen summa <total>-tägissä.
<intermediate> <class>NYL</class> <Competitor>4</Competitor> <point>1.000 km</point> <value>5</value> <total>18</total> <type>B</type> </intermediate>
Allaolevassa esimerkissä (type=G) MYL-B sarjan urheilija numerolla 51 on kiertänyt yhden sakkoringin (lukijan on paikassa 1.180 km) kellon ollessa 12:22:26:
<intermediate> <class>MYL-B</class> <Competitor>51</Competitor> <point>1.180 km</point> <elapsed>12:22:26.0</elapsed> <type>G</type> </intermediate>
Alla on käyttämäni C-kielen struktuuri kilpailijan tilatiedoista. Tilabittejä on 32 kappaletta, jotka siis tulostetaan XML-tietoihin yhtenä 32 bitin pitkänä hexadesimaalisena kokonaislukuna. Kaikki bitit bivät ole vielä käytössä. Siis kahdeksan kappaletta 4 bitin kenttiä, jokainen arvoltaan 0..9, A..F. Ja Intel-arkkitehtuurin mukaisesti viimeinen merkki koostuu 4:stä ensimmäisestä bitistä (eit, lapped, csped, ral käänteisessä järjestyksessä, eli eit on vähiten merkitsevä bitti).
Jos viimeinen merkki on 1, 3, 5, 7, 9, B, D tai F, urheilijalla ei ole tulosta (bitti eit on 1) ja jos lisäksi toiseksi viimeinen merkki on 1, tuloksen puuttumisen syy on keskeyttäminen, (bitti kes on 1), silloinhan ei oikeastaan muut saman nibblen biteistä voikaan olla päällä.
struct status_bitit_urheilija_v10 // Kilpailijan tilannetiedot { // tavu 1 alkaa, bitit 1..8, tulosbitit. Jos joku näistä on päällä, tulos on tiedossa BOOL eit : 1; // 1 = eitulosta-lippu BOOL lapped : 1; // 1 = kilpailija on ylösajettu, lapped BOOL csped : 1; // 1 = Jyry keskeyttää ja määrää viimeiseksi BOOL ral : 1; // 1 = Jyry rankkaa viimeiseksi BOOL kes : 1; // 1 = keskeyttänyt BOOL hyl : 1; // 1 = hylätty BOOL poi : 1; // 1 = poisjäänyt BOOL laa : 1; // 1 = lääkärintodistuksella poissa // tavu 2 alkaa, bitit 9..16 BOOL yhsta : 1; // 1 = Oli mukana viestin yhteislähdössä BOOL pakkosija : 1; // 1 = sijoitus annettu, ei saa muuttaa BOOL jal : 1; // 1 = jälki-ilmoittautunut BOOL tark : 1; // 1 = emit-tarkistus tehty (suunnistus) BOOL ulko : 1; // 1 = ulkopuolella_kilpailun BOOL on_lahtenyt : 1 ; // 1 = on saanut todellisen lähtöajan lähtöviikseltä/laukauksesta BOOL kilpailijalla_on_kommentti : 1; // TRUE = urheilijalla on rivejä.txt-tiedostossa BOOL FIS_laskettu : 1 ; // FIS-pisteet on laskettu // tavu 3 alkaa, bitit 17..24 BOOL hyvat_pojot : 1; // Tällä rankissa parhaita alkupisteitä, ei käytössä versiossa 9 BOOL tag1_ok : 1; // tunnistin 1 käsitelty BOOL odottaa_FF : 1 ; // on lähellä toista aikaa, tarvitaan päätös maalikameralta BOOL oma_hajonta : 1 ; // TRUE = urheilijalla oma hajonta, 0 = sarjan rata pätee (suunnistus) BOOL FIS_check : 1 ; // on tarkistettu FIS:n kannasta BOOL yritykset_taynna:1;// urheilija ei enää voi yrittää hyppyä tai heittoa (=TRUE) BOOL sisainen_id : 1; // vapaa BOOL distanssi_pisteet: 1; // FIS-pisteet on distanssimatkoilta BOOL sprintti_pisteet: 1; // FIS-pisteet on sprinttimatkoilta BOOL vapaa_25 : 1; // vapaa BOOL vapaa_26 : 1; // vapaa BOOL vapaa_27 : 1; // vapaa BOOL vapaa_28 : 1; // vapaa BOOL vapaa_29 : 1; // vapaa BOOL vapaa_30 : 1; // vapaa BOOL vapaa_31 : 1; // vapaa BOOL vapaa_32 : 1; // vapaa };
Urheilijan maa tai piiri "Nation" Kansainvälinen lisenssinumero "Fiscode" Urheilijan syntymäaika "Yearofbirth" Sukupuoli "Gender" Toisen transponderin numero "Transponder2" Pisteet ajasta "Timepoints" Tilastotulos "Statresult" Pisteet sijoituksesta "Rankpoints" Urheilijan virtuaalisarjat "Extraclasses" Urheilijalle annetut kommentit "Notes" Osatulosten lukumäärä "Parts" KILPAILIJAN_REITTI_KENTTA, 6, "Course" // 28 Urheilijan hajonnan nimi "Variety" Lisäinfo "Info" Suorituksen kesto päivinä "RaceDays" Urheilijan numero "Bib" Urheiljan sarja "ClassID" Urheilijan koko nimi "Name" Osuuden numero "Legno" Lisenssinumero "Lisence" Seuran nimi "Clubname" Lähtöaika "Starttime" Maaliintuloaika "Finishtime" Lopputulos "Totaltime" Aikahyvitys "Penalty" Joukkueen ID-numero "TeamID" Arvontaryhmä "Drawslot" Tilatiedot "Flags" Sijoitus "Rank" Pisteet "Points" Urheilijan nettoaika "NetTime" Transponderin numero "Tag" Kierrosten määrä "Rounds" Tarkistus tehty "Checked" Ratanumero "Lane" Parinumero "Pair" { SKIING, KILPAILIJAN_FISPISTE_KENTTA, 9, "Fispoints" // 13 { SKIING, KILPAILIJAN_FISTULOS_KENTTA, 10, "Racepoints" // 20 { BIATHLON, KILPAN_SAKKO_ERIT_KENTTA, 7, "Details" // 115 s { BIATHLON, KILPAN_SAKOT_SAATU_KENTTA, 13, "Penalties_got" // 23 { BIATHLON, KILPAN_SAKOT_TEHTY_KENTTA, 14, "Penalties_done" // 24 { TRACKFIELD, KILPAILIJAN_PB_KENTTA, 10, "PersonBest" // 31 { ELKSPORT, KILPAILIJAN_ARVIO_P_KENTTA, 10, "FirControl" // 2 { ELKSPORT, KILPAILIJAN_ARVIO_1_KENTTA, 12, "FirEstimate1" // 3 { ELKSPORT, KILPAILIJAN_ARVIO_2_KENTTA, 12, "FirEstimate2" // 4 { ELKSPORT, KILPAILIJAN_AMMUNTA_KENTTA, 11, "FirShooting" // 5 { ELKSPORT, KILPAILIJAN_ARVI2_P_KENTTA, 10, "SecControl" // 6 { ELKSPORT, KILPAILIJAN_ARVI2_1_KENTTA, 12, "SecEstimate1" // 7 { ELKSPORT, KILPAILIJAN_ARVI2_2_KENTTA, 12, "SecEstimate2" // 8 { ELKSPORT, KILPAILIJAN_AMMU2TA_KENTTA, 11, "SecShooting" // 9 { ELKSPORT, KILPAN_AR1PISTEET_KENTTA, 12, "FirEstPoints" // 18 { ELKSPORT, KILPAN_AR2PISTEET_KENTTA, 12, "SecEstPoints" // 19 { SARJA_TIETUE, 0, SARJAN_UNIQUE_ID_KENTTA, 8, "Cluniqid" // 49 1 { SARJA_TIETUE, 0, SARJAN_KISAPAIVA_KENTTA, 8, "Racedate" // 50 '2' { SARJA_TIETUE, 0, SARJAN_PAREMMUUS_KENTTA, 13, "BestSeqFields" // 51 '3' { SARJA_TIETUE, 0, SARJAN_TARKKUUS_KENTTA, 9, "Precision" // 52 '4' { SARJA_TIETUE, 0, SARJAN_MAALISEKTORI_KENTTA, 6, "Sector" // 53 '5' { SARJA_TIETUE, 0, SARJAN_PISTERYHMA_KENTTA, 10, "PointGroup" // 54 '6' { SARJA_TIETUE, 0, SARJAN_VIESTIRYHMA_KENTTA, 10, "RelayGroup" // 55 7 { SARJA_TIETUE, 0, SARJAN_ERA_KENTTA, 4, "Heat" // 57 9 { SARJA_TIETUE, 0, SARJAN_SUMMA_SARJA_KENTTA, 8, "SumClass" // 58 { SARJA_TIETUE, 0, SARJAN_SEURAJ_KOKO_KENTTA, 8, "Teamsize" // 60 < { SARJA_TIETUE, 0, SARJAN_JARJESTYS_KENTTA, 5, "Seqno" // 65 A { SARJA_TIETUE, 0, SARJAN_TUNNUS_KENTTA, 5, "Class" // 66 B { SARJA_TIETUE, 0, SARJAN_JOUKKUE_KOKO_KENTTA, 4, "Legs" // 67 C { SARJA_TIETUE, 0, SARJAN_KILPAILIJA_KENTTA, 8, "Athletes" // 68 D { SARJA_TIETUE, 0, SARJAN_NIMI_KENTTA, 9, "Classname" // 69 E { SARJA_TIETUE, 0, SARJAN_TYYLI_KENTTA, 9, "Technique" // 70 F { SARJA_TIETUE, 0, SARJAN_MATKA_KENTTA, 8, "Distance" // 71 G { SARJA_TIETUE, 0, SARJAN_LAHTOAIKA_KENTTA, 10, "Firststart" // 72 H { SARJA_TIETUE, 0, SARJAN_EKANUMERO_KENTTA, 8, "FirstBib" // 73 I { SARJA_TIETUE, 0, SARJAN_LAHTOVALI_KENTTA, 8, "Interval" // 74 J { SARJA_TIETUE, 0, SARJAN_RINNAN_KENTTA, 8, "Parallel" // 75 K { SARJA_TIETUE, 0, SARJAN_ILMOMAKSU_KENTTA, 3, "Fee" // 76 L { SARJA_TIETUE, 0, SARJAN_JALKIILMO_KENTTA, 7, "LateFee" // 77 M { SARJA_TIETUE, 0, SARJAN_LAHTOPAIKKA_KENTTA, 10, "Startplace" // 78 N { SARJA_TIETUE, 0, SARJAN_STATUSTIETO_KENTTA, 6, "Status" // 79 O { SARJA_TIETUE, 0, SARJAN_RATA_KENTTA, 10, "ClassTrack" // 80 P { SARJA_TIETUE, 0, SARJAN_AIKAPISTEET_KENTTA, 10, "Basepoints" // 81 Q { SARJA_TIETUE, 0, SARJAN_PARAS_AIKA_KENTTA, 16, "ClassFastetsTime" // 84 T { SARJA_TIETUE, 0, SARJAN_HITAIN_AIKA_KENTTA, 16, "ClassSlowestTime" // 85 U { SARJA_TIETUE, 0, SARJAN_VA_PISTEET_KENTTA, 14, "Intermedpoints" // 86 V { SARJA_TIETUE, 0, SARJAN_ERARYHMA_KENTTA, 10, "ClassGroup" // 87 W { SARJA_TIETUE, 0, "Ens. kierroksen pituus", 11, "FirstLapLen" // 89 Y { SARJA_TIETUE, 0, "Kierroksen pituus", 9, "LapLength" // 90 Z { SARJA_TIETUE, 0, "Viim. kierroksen pituus", 10, "LastLapLen" // 91 [ { SARJA_TIETUE, 0, "Sarjan kierrosmäärä", 9, "TotalLaps" // 92 { SARJA_TIETUE, 0, SARJAN_LAJITYYPPI_KENTTA, 9, "Sporttype" // 93 ] { SARJA_TIETUE, 0, SARJAN_KOMME_KENTTA, 8, "Comments" // 94 ^ { SARJA_TIETUE, 0, SARJAN_OSUUSNRO_KENTTA, 9, "Legnumber" // 95 { SARJA_TIETUE, SKIING, SARJAN_CODEX_KENTTA, 5, "Codex" // 56 8 { SARJA_TIETUE, BIATHLON, SARJAN_SAKKOPAIKAT_KENTTA, 18, "ClassPenaltyPoints" // 82 R { SARJA_TIETUE, BIATHLON, SARJAN_SAKKOARVO_KENTTA, 16, "PenaltyBaseValue" // 83 S { SARJA_TIETUE, TRACKFIELD, SARJAN_NOSTO_KENTTA, 8, "Fieldtra" // 59 ';' { VIRT_TIETO, 0, "KILPAILIJA", 10, "Competitor" // 96 @+32 Koontikenttä { VIRT_TIETO, 0, KILPAILIJAN_ETUNIMI_KENTTA, 9, "Firstname" // 10 { VIRT_TIETO, 0, KILPAILIJAN_SUKUNIMI_KENTTA, 8, "Lastname" // 11 { VIRT_TIETO, 0, "SARJAN_ALOITUS_TAGI", 9, "Classinfo" // 64 @ Koontikenttä { VIRT_TIETO, 0, OHITA_TAMA_KENTTA, 8, "notinuse" // 88 X