{"version":3,"sources":["sections/Introduction.js","components/YoutubeEmbed.js","components/TimeHeading.js","images/koe-loki.png","sections/Demo.js","images/fs-test.png","components/LinkBlock.js","sections/Summary.js","images/GitHub.png","images/YTL.png","sections sync /^/.//.*$","components/Section.js","components/Authors.js","images/avatars/eemil.png","images/avatars/ruben.png","images/avatars/mikael.png","App.js","reportWebVitals.js","index.js"],"names":["Introduction","YoutubeEmbed","id","class","title","src","frameborder","gesture","allow","allowfullscreen","TimeHeading","data","time","description","className","Demo","text","language","showLineNumbers","theme","dracula","customStyle","borderRadius","display","codeContainerStyle","highlight","koeLogPicture","alt","LinkBlock","icon","siteName","date","url","href","target","rel","Summary","map","webpackContext","req","webpackContextResolve","__webpack_require__","o","e","Error","code","keys","Object","resolve","module","exports","sections","require","Section","name","default","Authors","author","avatar","join","App","EemilAvatar","RubenAvatar","MikaelAvatar","reportWebVitals","onPerfEntry","Function","then","getCLS","getFID","getFCP","getLCP","getTTFB","ReactDOM","render","StrictMode","document","getElementById"],"mappings":"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","file":"static/js/main.ed534374.chunk.js","sourcesContent":["export default function Introduction() {\n return (\n
\n

\n Ilmoitimme Ylioppilastutkintolautakunnalle keväällä 2021 lukuisista haavoittuvuuksista Abitti-koejärjestelmästä. Haavoittuvuussarja antoi hyökkääjälle täyden pääsyn koetilan palvelimelle ja jokaiseen siihen yhdistäneeseen kokelaan tietokoneeseen pääkäyttäjänä. Haavoittuvuudet koskettivat kaikkia Suomen lukioita ja sähköisiä ylioppilaskirjoituksia. Alttiina lukemiselle, muokkaamiselle, poistamiselle, ja häiriköinnille olivat muun muassa:\n

\n \n

\n Mahdollinen tietovuoto olisi voinut olla hyvin vakava ja sisältää suuria määriä kokelaiden arkaluontoistakin dataa. Puhumattakaan vilpin ja koetilanteen häiriköinnin mahdollisuudesta.\n

\n

\n Tämä kaikki löydettiin noin kolmen kuukauden tiimityön ja lukuisten Abitin lähdekoodin parissa vietettyjen tuntien ansiosta. Nyt kuitenkin pääsemme jakamaan työmme tuloksia ja olemme tästä yhtä innoissamme kuin tekin :)\n

\n

\n Haavoittuvuuden hyödyntämiseen tarkoitettu koodi on tämän postauksen lopussa ;)\n

\n
\n\n )\n};","export function YoutubeEmbed({id}) {\n return (\n
\n \n
\n )\n}","export function TimeHeading({data}) {\n const {title, time, description} = data;\n return (\n
\n

\n {time ? {time} : ''} {title} \n

\n { description ?\n \n {description}\n \n : ''}\n
\n );\n}","export default __webpack_public_path__ + \"static/media/koe-loki.d00fab59.png\";","import { YoutubeEmbed } from '../components/YoutubeEmbed';\nimport { TimeHeading } from '../components/TimeHeading';\n\nimport { CodeBlock, dracula } from \"react-code-blocks\";\n\nimport koeLogPicture from '../images/koe-loki.png';\nimport fsTestPicture from '../images/fs-test.png';\n\nexport default function Demo() {\n return (\n
\n

Demonstraatio ja rekonstruktio

\n \n\n \n

\n Kokelastikun ABITTI2106S levykuvaan on tehty muokkauksia, jotka sallivat pääsyn komentokehotteeseen ja ujuttamaan tiedostojärjestelmään tarvittavat skriptimme. Muutokset tekee videolla hetkellisesti esitelty AntiBitti-työkalumme.\n

\n\n \n

\n Koetilan palvelin tiedustelee kaikilta kokelaslaitteilta dataa 20 - 30 minuutin välein. Korvaamme kokelaslaitteen vastauspalvelimen omalla muokatulla palvelimella “nsamock.js”, jolla voimme itse määrittää koetilan palvelimelle annetun vastauksen. Tätä hyökkäystekniikkaa kutsutaan nimellä “Server Side Request Forgery” (SSRF), jota voisi suomeksi luonnehtia palvelinpuolen pyynnön väärentämiseksi.\n

\n

\n Vastaamme koetilan palvelimen NSA-palvelun (kokeen valvontapalvelu) pyyntöön haavoittuvuusketjun laukaisevalla vastauksella, joka avaa ovet hyökkäyksen seuraaville vaiheille:\n

\n \n
\n\n \n

\n Edellisessä vaiheessa palautimme koetilan palvelimella toimivalle NSA-palvelulle vastauksen, jonka HTTP-statuskoodina oli 307. Tämä koodi pyytää koetilan palvelimen HTTP-asiakasohjelman uudelleenohjaamaan pyyntönsä toiseen osoitteeseen. Annettu osoite on koetilan palvelimella käynnissä olevan palvelun osoite, joka on eristetty palomuurilla muilta kuin palvelimelta itseltään. Ohjaamme siis koetilan palvelimen tekemään pyynnön itseelleen ja muokkaamme samalla pyynnön osoitetta. Palomuuri ohitettu.\n

\n

\n Koetilan palvelimelle annettu uudelleenohjaus sen aikaisemmalle pyynnölle ohjautuu tarkoin valittuun osoitteeseen: yhteen sen sisäiseen web-palvelimeen, joka on haavoittuvainen “Shell injection”-hyökkäykselle:\n

\n \n createAnswersZip()\n .then(zip => fs.writeFileAsync(tmpAnswersPackageFile, zip))\n .then(() => {\n if (config.isProd) {\n childProcess.exec(\n \\`/usr/bin/sudo -u digabi /usr/local/bin/digabi-download-progress\n \\${tmpAnswersPackageFile} \\${req.params.answersFilename}\\`\n )\n }\n })\n .then(() => res.json({ useDevModeAnswerDownloading: !config.isProd }))\n .catch(next)\n)`}\n language=\"js\"\n showLineNumbers={false}\n theme={dracula}\n highlight=\"6-9\"\n customStyle={{borderRadius: \"10px\", display: \"block\", \"overflow-y\": \"hidden\"}}\n codeContainerStyle={{display: \"block\"}}\n />\n

\n Polun “/start-answer-downloader/:answersFilename”-metodin alkuperäinen tarkoitus on toimia tapana kokeen järjestäjälle ladata kokeen vastauksia USB-tikulle tai johonkin muuhun järjestelmän sijaintiin. Tämän takia videolla hyökkäyksen tässä vaiheessa ilmestyy “valitse sijanti”-ikkuna. Tämä ikkuna olisi kuitenkin todella helppoa piilottaa.\n

\n

\n Tutkiessa ylhäällä olevaa koodia tarkemmin voi helposti huomata selkeän “Shell injection”-haavoittuvuuden, kohdassa jossa “req.params.answersFilename” lisätään suoraan komentoriville. Käyttäjän syötettä ei tarkisteta tai sanitoida lainkaan.\n

\n

\n Pääkäyttäjän oikeudet saa koetila-käyttäjältä (jolla Shell injection koodia ajetaan) helposti,\n hyödyntämällä virhettä Abitin sudo-asetuksissa. Koetila-käyttäjä saa ajaa pääkäyttäjänä vain muutaman ohjelman, esimerkiksi sijainnissa\n /tmp/ktp-update-work/ktp-update.sh olevan skriptin. Koetila-käyttäjä voi kuitenkin myös kirjoittaa tähän tiedostoon täysin vapaasti, näin antaen meille helpon tavan saavuttaa pääkäyttäjän oikeudet.\n

\n

\n Lähetetyssä komennossa on kaksi keskeistä osaa:\n

\n
    \n
  1. \n Ensimmäinen rivi avaa kaiken liikenteen Abitin koetilan palvelimelle ja siltä ulos. Teknisesti puhuen se tyhjentää palomuurin asetukset, näin sallien kaiken liikenteen.\n
  2. \n
  3. \n Toinen rivi luo etäyhteyden, jonka avulla voi kätevästi lähettää komentoja koetilan palvelimelle. Se luo ja käynnistää Node.js:llä kirjoitetun TCP Socket-pohjaisen RCE-palvelimen (Remote Code Execution, eli komentojen suoritus etänä).\n
  4. \n
\n\n \n

\n Hyökkääjän laitteella avataan aikaisemmin käynnistetyn RCE-palvelimen asiakasohjelma “nclient.js”. Tämä ohjelma toimii käytännössä samalla tavalla kuin SSH ja omaa kaikki pääkäyttäjän luvat ajaa koodia koetilan palvelimella.\n

\n

\n Videossa demonstroidaan etäyhteyttä useaan otteeseen, esimerkiksi avaamalla koetilan palvelimen ruudulle uuden selainvälilehden Firefoxissa, uuden muistion, sekä lopuksi välittömästi sammuttamalla koetilan palvelimen. \n

\n

\n Pääkäyttäjän oikeudet kuitenkin mahdollistavat paljon vakavampia kokeen valvonnalta piilossa olevia toimia. Se mahdollistaa kaikenlaisen koetilanteen häirinnän, tallenteiden lukemisen, muokkaamisen ja poistamisen. Pääkäyttäjän oikeudet koetilan palvelimella antavat myös paljon mahdollisuuksia muiden sen verkossa olevien Abitti-koneiden kaappaamiseksi, joka vaarantaa myös kokelaiden henkilökohtaisten tietojen turvallisuuden heidän koneillaan, vaikka Abitissa koneiden tallennustila ei olisi kokelaille tavallisesti näkyvissä, sillä pääkäyttäjänä voimme päästä levyihin käsiksi vapaasti.\n

\n\n \n

\n Koetilan palvelimessa on kokelaiden laitteille tarkoitettu automaattinen päivitysmekanismi, joka lähettää päivityspaketin kokelaiden laitteille niiden liittyessä kokeeseen. Kaappaamalla koetilan palvelimen, voimme luoda oman päivityspaketin, joka antaa meille yksinkertaisesti tavan suorittaa koodia kokelaiden laitteilla pääkäyttäjän oikeuksilla.\n

\n

\n Polkuun /var/lib/ktpjs/public/koe-update/ tehdään zip-arkisto nimellä “koe-update.zip”, joka sisältää skriptitiedoston “koe-update.sh”. Kokelaan laite suorittaa tämän skriptin avattuaan päivityspaketin.\n

\n

\n Esimerkki mahdollisesta hyökkäyksessä käytettävästä skriptistä:\n

\n > /home/digabi/pwned.txt\n\nexport DISPLAY=:0\n\nsudo -u digabi notify-send \"ABITTI HAS BEEN PWNED!\"\n`}\n language={(process.env.NODE_ENV === 'development' ? 'text' : 'shell')}\n showLineNumbers={false}\n theme={dracula}\n customStyle={{borderRadius: \"10px\", display: \"block\", \"overflow-y\": \"hidden\"}}\n codeContainerStyle={{display: \"block\"}}\n />;\n

\n Esimerkkiskripti luo tiedostojärjestelmän juureen tiedoston, jonka luomiseen on normaalisti oikeus vain pääkäyttäjällä. Se myös lähettää työpöytäilmoituksen koetilapalvelimen ruudulle.\n

\n

\n Avaamalla kokelaan “digabi-koe” Systemd-palvelun lokin, voi huomata päivityspaketin sisältäneen koodin tulleen suoritetuksi pääkäyttäjänä:\n

\n \"Kuvankaappaus\n

\n Tässä vielä skriptin luoma tiedosto tiedostojärjestelmän juuressa:\n

\n \"Kuvankaappaus\n
\n );\n}","export default \"\"","export function LinkBlock({data}) {\n const {title, icon, siteName, date, url} = data;\n return (\n \n {title} {date ? {date} : ''}\n \n {siteName}\n {siteName}\n \n \n );\n}","import { LinkBlock } from '../components/LinkBlock';\n\nimport GitHubIcon from '../images/GitHub.png';\nimport YTLIcon from '../images/YTL.png';\n\nexport default function Summary() {\n return (\n
\n

Yhteenveto

\n

\n Haluamme kiittää ylioppilastutkintolautakuntaa hyvin asiallisesta tietoturvailmoituksen käsittelystä ja ripeästä toiminnasta. He ovat hoitaneet tapauksen mallikkaasti sisäisesti ja ulkoisesti. Etenkin hyvänä yksityiskohtana haluamme tuoda esiin huolellisesti suunnitellun julkaisuaikataulun tekemistä korjausta varten, jossa otettiin huomioon erilaisia vaihtoehtoja huomioimalla niiden uhat ja edut.\n

\n\n

Haavoittuvuuden lähdekoodi

\n

\n Hyökkäyksen suorittamiseen tarvittava koodi on saatavilla täältä: \n

\n \n

\n Haavoittuvuuden vaiheita voit kokeilla viimeisimmällä toimivalla kokelaan tikun versiolla ABITTI2106S (tai vanhemmat, 1.10.2019 asti) ja palvelimen tikun versiolla SERVER21066 (tai vanhemmat, 1.10.2019 asti). Tämä haavoittuvuus on korjattu viime päivityksissä, joten sen testaaminen oikeissa koetilanteessa on turhaa ja voi pahimmillaan johtaa jopa kahden vuoden vankeustuomioon (ks. RL 38:7 a § ja 8 §).\n

\n\n

\n Lue Ylioppilastutkintolautakunnan katsaus tästä haavoittuvuussarjasta seuraavasta blogiartikkelista, sekä sen tietoturvakäytännöistä ja vapaaehtoisten tietoturvaraporttien tärkeydestä: \n

\n \n
\n );\n}","export default \"\"","export default \"\"","var map = {\n\t\"./Demo\": 14,\n\t\"./Demo.js\": 14,\n\t\"./Introduction\": 13,\n\t\"./Introduction.js\": 13,\n\t\"./Summary\": 15,\n\t\"./Summary.js\": 15\n};\n\n\nfunction webpackContext(req) {\n\tvar id = webpackContextResolve(req);\n\treturn __webpack_require__(id);\n}\nfunction webpackContextResolve(req) {\n\tif(!__webpack_require__.o(map, req)) {\n\t\tvar e = new Error(\"Cannot find module '\" + req + \"'\");\n\t\te.code = 'MODULE_NOT_FOUND';\n\t\tthrow e;\n\t}\n\treturn map[req];\n}\nwebpackContext.keys = function webpackContextKeys() {\n\treturn Object.keys(map);\n};\nwebpackContext.resolve = webpackContextResolve;\nmodule.exports = webpackContext;\nwebpackContext.id = 32;","const sections = require.context('../sections', true);\n\nexport function Section({name}) {\n return sections('./' + name).default();\n}","export function Authors({data, date}) {\n return (\n
\n
\n {data.map(author => (\n {author.name}\n ))}\n
\n
\n {data.map(author => author.name).join(', ')}\n {date}\n
\n
\n )\n}","export default __webpack_public_path__ + \"static/media/eemil.85dd9b69.png\";","export default __webpack_public_path__ + \"static/media/ruben.46b5a711.png\";","export default __webpack_public_path__ + \"static/media/mikael.82ff0020.png\";","import './App.css';\n\nimport { Section } from './components/Section';\nimport { Authors } from './components/Authors';\n\nimport EemilAvatar from './images/avatars/eemil.png';\nimport RubenAvatar from './images/avatars/ruben.png';\nimport MikaelAvatar from './images/avatars/mikael.png';\n\nfunction App() {\n return (\n
\n
\n

Abitti Open Access

\n
\n
\n
\n
\n
\n
\n
\n \n
\n

\n 2021 © Testausserveri\n

\n
\n
\n
\n );\n}\n\nexport default App;\n","const reportWebVitals = onPerfEntry => {\n if (onPerfEntry && onPerfEntry instanceof Function) {\n import('web-vitals').then(({ getCLS, getFID, getFCP, getLCP, getTTFB }) => {\n getCLS(onPerfEntry);\n getFID(onPerfEntry);\n getFCP(onPerfEntry);\n getLCP(onPerfEntry);\n getTTFB(onPerfEntry);\n });\n }\n};\n\nexport default reportWebVitals;\n","import React from 'react';\nimport ReactDOM from 'react-dom';\nimport './index.css';\nimport App from './App';\nimport reportWebVitals from './reportWebVitals';\n\nReactDOM.render(\n \n \n ,\n document.getElementById('root')\n);\n\n// If you want to start measuring performance in your app, pass a function\n// to log results (for example: reportWebVitals(console.log))\n// or send to an analytics endpoint. Learn more: https://bit.ly/CRA-vitals\nreportWebVitals();\n"],"sourceRoot":""}