Mastodon CZ Sraz 2024, tj. druhý ročník srazu uživatelů Mastodonu a Fediverse proběhne 11. října 2024 v Praze.
Projekty Tor a Tails se spojují. Tails bude začleněn do struktury projektu Tor pro snazší spolupráci, lepší udržitelnost, snížení režie, rozšiřování povědomí o větším počtu digitálních hrozeb a lepší ochranu před sledováním a cenzurováním.
Společnost System76 vydala druhou alfa verzi desktopového prostředí COSMIC Epoch 1 společně s alfa verzí Pop!_OS 24.04 LTS.
Byl vydán PostgreSQL 17. Přehled novinek v poznámkách k vydání.
Společnost Meta Platforms na své dvoudenní konferenci Meta Connect 2024 představuje novinky: brýle Orion (původně Project Nazare) pro rozšířenou realitu (AR, Augmented Reality), nový VR headset Meta Quest 3S, vylepšené chytré brýle Ray-Ban Meta nebo Llama 3.2, tj. nejnovější verzi svého velkého jazykového modelu.
Počítačová hra XBill (Wikipedie) oslavila v létě 30 let. V devadesátkách Linuxáci poctivě bránili Billovi v instalaci Windows. Bohužel hry XSteve a XSatya nevznikly. A nemáme ani hru XTim. Proto není Linux na desktopu tak rozšířený ☺. Mimochodem, po 25 letech byl aktualizován XBill pro PalmOS.
Počítačová hra Elite (Wikipedie), 3D vesmírní simulátor, byla vydána před 40 lety, 20. září 1984. Při té příležitosti byly zveřejněny další zdrojové kódy Elite pro platformy Apple, Atari, C64, NES a SNES a nedokončené Elite II pro BBC Micro.
V květnu bylo oznámeno, že dnes budou zveřejněny zdrojové kódy přehrávače Winamp. Stalo se tak (𝕏). Zdrojové kódy jsou k dispozici na GitHubu. Nejedná se ale o svobodný a otevřený software (licence).
Fiala navrhne odvolání Bartoše z postu vicepremiéra pro digitalizaci a ministra pro místní rozvoj ke 30. září. Důvodem je nezvládnutí digitalizace stavebního řízení, podle premiéra ji Bartoš není schopen dotáhnout do konce. „Po projednání analýzy digitálního stavebního řízení na vládě minulou středu a po dnešním ranním rozhovoru s panem vicepremiérem Ivanem Bartošem jsem bohužel nabyl jistoty, že není schopen tuto digitalizaci
… více »Komunikační platforma Telegram začne po tlaku úřadů poskytovat vládám více informací o svých uživatelích. V pondělí to oznámil její zakladatel a generální ředitel Pavel Durov. Ten už několik týdnů ve Francii čelí obvinění, že nedělá dost pro to, aby platformu nevyužívaly i kriminální živly. To chce Durov nyní také změnit, informují tiskové agentury.
V diskusích pod články ze seriálu o CouchDB jsem často zmiňoval MongoDB jako databázi, která řeší některé problémy mnohem lépe. V tomto článku ji tedy trochu přiblížím – nezvládneme sice popsat tak rozsáhlou část API jako v případě CouchDB, protože Mongo je o poznání rozsáhlejší, uvidíme toho ale dost na to, abychom si udělali rozumnou představu, v čem spočívají jeho silné a slabé stránky.
Nejdřív ale k instalaci: Z webu lze stáhnout zkompilovanou stabilní verzi (1.2.2) nebo vývojovou (1.3.2), s balíčky je to zatím horší. Zdrojový archiv už nějakou dobu obsahuje adresář debian
a vytvořit si vlastní balíček pro Debian nebo Ubuntu tak není např. s pomocí pbuilder
u žádný velký problém, ostatním bude prozatím stačit, když jednoduše spustí bin/mongod
. Použijeme verzi 1.3.2, která je sice vývojová (Mongo používá konvenci sudých minor verzí pro označení stabilních vydání a lichých pro vydání vývojová), ale obsahuje řadu oprav a vylepšení a např. pro sharding je to dnes jediná použitelná možnost.
Mongo funguje jako klasický databázový server, ke kterému se klienty připojují přes síť (obvykle na portu 27017) a komunikují s ním speciálním protokolem (ovšem na portu 28017 najdete jednoduché HTTP rozhraní). Součástí instalace je knihovna pro C++, jako součásti projektu jsou ale vyvíjeny a podporovány i knihovny pro Javu, Python, Ruby, PHP a další jazyky. My si prozatím vystačíme s javascriptovou konzolí, kterou spustíme příkazem mongo
.
Jednotlivé databáze není potřeba vytvářet, vzniknou při prvním zápisu. Směle do toho:
> use pokus switched to db pokus
Aktuální databáze je přístupná jako objekt db
, stručnou nápovědu získáte zavoláním db.help()
.
Databáze se skládá z kolekcí (paralela s tabulkami z SQL je nabíledni). Ani ty není nutné předem vytvářet, stačí do nich zapsat. Kolekce jsou přirozeně součástmi objektu db
.
> db.bands.insert({name: "Old Blind Dogs", genres: ["celtic", "folk", "world music"], since: 1990}) ObjectId("4b76ef5780b71853fe6c89b6") > db.bands.count() 1 > db.bands.find() { "_id" : ObjectId("4b76ef5780b71853fe6c89b6"), "name" : "Old Blind Dogs", "genres" : [ "celtic", "folk", "world music" ], "since" : 1990 }
Na kolekci lze také volat metodu help()
(a klidně i na neexistující, zkuste db.foo.help()
). Nepřekvapí, že Mongo ukládá JSONové dokumenty (uvnitř se ovšem používá BSON, binární serializace JSONu s několika přidanými typy, a velikost dokumentu je omezena na 4 MB), ani že jim automaticky přiřazuje UUID. Jedna věc už však zřejmá není, ale je velmi důležitá: Klientské knihovny kvůli výkonu nečekají na dokončení odeslaných příkazů („fire and forget“) a očekávají, že vše proběhne v pořádku. Je však k disposici bezpečný režim (safe mode), který po každém příkazu volá getLastError a zajišťuje tak, že se o případné chybě dozvíme ihned.
Vidíme ale už jednu zajímavější věc – find? Klasické dotazování není v NoSQL světě zrovna obvyklé, tady Mongo docela vyčnívá a stává se přístupnější lidem z SQL světa. Ať máme trochu víc dat a můžeme si pohrát:
> db.bands.insert({name: "The Pogues", genres: ["celtic", "punk"], since: 1982}) ObjectId("4b76f01480b71853fe6c89b7") > db.bands.insert({name: "Planxty", genres: ["celtic", "traditional"], since: 1972}) ObjectId("4b76f01480b71853fe6c89b8") > db.bands.insert({name: "The Bothy Band", genres: ["celtic", "traditional"], since: 1974}) ObjectId("4b76f01480b71853fe6c89b9") > db.bands.insert({name: "The Chieftains", genres: ["celtic", "traditional"], since: 1962}) ObjectId("4b76f01480b71853fe6c89ba") > db.bands.insert({name: "The Real McKenzies", genres: ["celtic", "punk"], since: 1992}) ObjectId("4b76f01480b71853fe6c89bb") > db.bands.insert({name: "Cruachan", genres: ["celtic", "metal"], since: 1992}) ObjectId("4b76f01480b71853fe6c89bc") > db.bands.insert({name: "Capercaillie", genres: ["celtic", "world music"], since: 1984}) ObjectId("4b76f01580b71853fe6c89bd") > db.bands.count() 8
Dotazovací jazyk spíš než co jiného připomíná QBE. Je poměrně bohatý, takže ho zde nebudeme popisovat celý (vizte dokumentaci), ale pár příkladů si ukážeme:
// podle konkrétní hodnoty > db.bands.find({since: 1990}) { "_id" : ObjectId("4b76ef5780b71853fe6c89b6"), "name" : "Old Blind Dogs", "genres" : [ "celtic", "folk", "world music" ], "since" : 1990 } // lze používat běžné operátory < ($lt), > ($gt), <= ($lte), >= ($gte) i <> ($ne) > db.bands.find({since: {$gte: 1990}}) { "_id" : ObjectId("4b76ef5780b71853fe6c89b6"), "name" : "Old Blind Dogs", "genres" : [ "celtic", "folk", "world music" ], "since" : 1990 } { "_id" : ObjectId("4b76f01480b71853fe6c89bb"), "name" : "The Real McKenzies", "genres" : [ "celtic", "punk" ], "since" : 1992 } { "_id" : ObjectId("4b76f01480b71853fe6c89bc"), "name" : "Cruachan", "genres" : [ "celtic", "metal" ], "since" : 1992 } // hledat se dá i v poli > db.bands.find({genres: "traditional"}) { "_id" : ObjectId("4b76f01480b71853fe6c89b8"), "name" : "Planxty", "genres" : [ "celtic", "traditional" ], "since" : 1972 } { "_id" : ObjectId("4b76f01480b71853fe6c89b9"), "name" : "The Bothy Band", "genres" : [ "celtic", "traditional" ], "since" : 1974 } { "_id" : ObjectId("4b76f01480b71853fe6c89ba"), "name" : "The Chieftains", "genres" : [ "celtic", "traditional" ], "since" : 1962 } // můžeme říct, že chceme jen některé položky > db.bands.find({genres: "traditional"}, {name: 1}) { "_id" : ObjectId("4b76f01480b71853fe6c89b8"), "name" : "Planxty" } { "_id" : ObjectId("4b76f01480b71853fe6c89b9"), "name" : "The Bothy Band" } { "_id" : ObjectId("4b76f01480b71853fe6c89ba"), "name" : "The Chieftains" }
Popis různých operátorů najdete v dokumentaci. Pokud se v tuhle chvíli ptáte, jak to bude s rychlostí v případě velkých databází, uvažujete správně. Mongo zpracovává dotazy poměrně tradičním způsobem: Vytvoří množinu prováděcích plánů, vybere z nich ten nejvhodnější a ten provede. Klíčovým prvkem pro rychlost provádění dotazu je, také tradičně, přítomnost indexů. Existuje pochopitelně i funkce explain
– všimněte si zejména položek n
a nscanned
:
> db.bands.find({genres: "traditional"}).explain() { "cursor" : "BasicCursor", "startKey" : { }, "endKey" : { }, "nscanned" : 8, "n" : 3, "millis" : 0, "allPlans" : [ { "cursor" : "BasicCursor", "startKey" : { }, "endKey" : { } } ] } // -1 pro sestupné řazení, 1 pro vzestupné > db.bands.ensureIndex({genres: 1}) > db.bands.find({genres: "traditional"}).explain() { "cursor" : "BtreeCursor genres_1", "startKey" : { "genres" : "traditional" }, "endKey" : { "genres" : "traditional" }, "nscanned" : 3, "n" : 3, "millis" : 0, "allPlans" : [ { "cursor" : "BtreeCursor genres_1", "startKey" : { "genres" : "traditional" }, "endKey" : { "genres" : "traditional" } } ] }
Indexy lze vytvářet i nad více položkami najednou (složené indexy) či takové, které zajišťují jedinečnost dané položky (unikátní indexy). V jedné kolekci lze mít nejvýše 40 indexů, přičemž index nad položkou _id
existuje vždy a není možné ho odstranit. Více informací naleznete jako obvykle v dokumentaci.
Určitě jste zaznamenali slovo kurzor. Mongo na dotaz nevrací ihned všechny výsledky, jak by se mohlo zdát z příkladů výše, ale právě kurzor, nad kterým lze iterovat:
// lze používat i regulární výrazy > var c = db.bands.find({name: /^The.*/}) // teprve teď se skutečně začnou číst data > c.forEach( function(doc) { print(doc.name); } ) The Bothy Band The Chieftains The Pogues The Real McKenzies
Řazení? Stránkování? I to funguje tak, jak bychom mohli očekávat.
// počet nalezených výsledků > db.bands.find({name: /^The.*/}).count() 4 // -1 pro sestupné řazení, 1 pro vzestupné, jako u vytváření indexů > db.bands.find({name: /^The.*/}).sort({since: -1}).forEach( function(doc) { print(doc.since, doc.name) } ) 1992 The Real McKenzies 1982 The Pogues 1974 The Bothy Band 1962 The Chieftains // nejvýše 3 výsledky, a první 3 přeskočit > db.bands.find({name: /^The.*/}).sort({since: -1}).limit(3).skip(3).forEach( function(doc) { print(doc.since, doc.name) } ) 1962 The Chieftains
Viděli jsme, že možnosti dotazování jsou poměrně široké. Úpravy dokumentů se provádějí podobným stylem:
// nastavení hodnoty > db.bands.update({_id: ObjectId("4b76ef5780b71853fe6c89b6")}, {$set: {albums: ["New Tricks", "Close To The Bone", "Tall Tails", "Legacy", "Five"]}}) // přidání jedné hodnoty do pole > db.bands.update({_id: ObjectId("4b76ef5780b71853fe6c89b6")}, {$push: {albums: "The World's Room"}}) // přidání více hodnot do pole > db.bands.update({_id: ObjectId("4b76ef5780b71853fe6c89b6")}, {$pushAll: {albums: ["Fit?", "The Gab o Mey", "Play Live", "Four On The Floor"]}}) > db.bands.find({name: "Old Blind Dogs"}, {name: 1, albums: 1}) { "_id" : ObjectId("4b76ef5780b71853fe6c89b6"), "albums" : [ "New Tricks", "Close To The Bone", "Tall Tails", "Legacy", "Five", "The World's Room", "Fit?", "The Gab o Mey", "Play Live", "Four On The Floor" ], "name" : "Old Blind Dogs" }
Ve své základní podobě má funkce update
dva parametry: Dotaz, stejný jako ve funkci find
, a samotný předpis, co a jak upravit. Aktualizace jednoho dokumentu tímto způsobem je atomická. Jednotlivé položky můžeme i mazat:
> db.bands.update({_id: ObjectId("4b76ef5780b71853fe6c89b6")}, {$unset: {albums: 1}}) > db.bands.find({name: "Old Blind Dogs"}, {name: 1, albums: 1}) { "_id" : ObjectId("4b76ef5780b71853fe6c89b6"), "name" : "Old Blind Dogs" }
Operátorů je samozřejmě víc a jejich přehled najdete v dokumentaci.
Funkce update
má dva další, nepovinné parametry: upsert
a multi
. Slovo upsert je zřejmě kombinací slov „update“ a „insert“, což dává celkem jasnou představu o jeho funkci: Pokud dokument zadaný dotazem existuje, pak ho upraví, jinak ho z dat dotazu a updatu vytvoří:
// třetí parametr je true => upsert > db.bands.update({name: "Runrig"}, {$set: {since: 1973, genres: ["celtic", "folk", "rock"]}}, true) > db.bands.find({name: "Runrig"}) { "_id" : ObjectId("4b7860e762fecfdba45d39e8"), "genres" : [ "celtic", "folk", "rock" ], "name" : "Runrig", "since" : 1973 }
Trochu odbočíme od našeho příkladu: Jedním z použití, pro které je Mongo speciálně navrženo, jsou online statistiky. Upserty jsou zde jasnou volbou:
// $inc je přičtení zadaného čísla k položce > db.stats.update({page: "index"}, {$inc: {views: 1}}, true) > db.stats.find() { "_id" : ObjectId("4b78633662fecfdba45d39e9"), "page" : "index", "views" : 1 }
Proč se o tom zmiňuji: Mongo má ještě jednu zásadní vlastnost, o které nebyla řeč. Datové soubory jsou vždy mapovány do paměti (mmap), takže je předvedená inkrementace pekelně rychlá (přečtení jedné hodnoty z paměti a zápis na totéž místo). Pochopitelným důsledkem je, že pro větší databáze je nutné používat 64bitový systém, ale to už je dnes snad samozřejmost :-)
Zpět ke kvalitní hudbě. Poslední parametr (multi
) umožňuje najednou změnit více dokumentů (ovšem pozor, atomická je změna jen jednoho dokumentu, ne všech dohromady!).
// $in -- hodnota položky musí být rovna jedné z uvedených > db.bands.update({name: {$in: ["The Chieftains", "Planxty", "The Bothy Band"]}}, {$set: {stars: "6/5"}}) // upraven jen jeden dokument, ten, který databáze nalezla jako první > db.bands.find({name: {$in: ["The Chieftains", "Planxty", "The Bothy Band"]}}, {name: 1, stars: 1}) { "_id" : ObjectId("4b76f01480b71853fe6c89b8"), "name" : "Planxty", "stars" : "6/5" } { "_id" : ObjectId("4b76f01480b71853fe6c89b9"), "name" : "The Bothy Band" } { "_id" : ObjectId("4b76f01480b71853fe6c89ba"), "name" : "The Chieftains" } // čtvrtý parametr je true => multi > db.bands.update({name: {$in: ["The Chieftains", "Planxty", "The Bothy Band"]}}, {$set: {stars: "6/5"}}, false, true) // upraveny všechny dokumenty odpovídající dotazu > db.bands.find({name: {$in: ["The Chieftains", "Planxty", "The Bothy Band"]}}, {name: 1, stars: 1}) { "_id" : ObjectId("4b76f01480b71853fe6c89b8"), "name" : "Planxty", "stars" : "6/5" } { "_id" : ObjectId("4b76f01480b71853fe6c89b9"), "name" : "The Bothy Band", "stars" : "6/5" } { "_id" : ObjectId("4b76f01480b71853fe6c89ba"), "name" : "The Chieftains", "stars" : "6/5" }
Mazání dokumentů pak pro nás bude už procházka růžovým sadem: Funkce remove
přijímá jako parametr dotaz a smaže všechny nalezené dokumenty:
// bez dotazu = smazat všechno > db.stats.remove() > db.stats.count() 0 // punk'z dead > db.bands.remove({genres: "punk"}) > db.bands.find({genres: "punk"}).count() 0
Občas se samozřejmě setkáme s potřebou provázat mezi sebou dva dokumenty ze dvou různých kolekcí. Přestože v našem případě by bylo lepší ukládat alba přímo k jednotlivým kapelám, jak jsme viděli výše, ukážeme si reference právě na tomto příkladě. Mongo definuje konvenci, která slouží právě k tomuto účelu. Jedinou momentální výhodou jejího použití namísto nějaké konvence vlastní je podpora pro automatické dereferencování (na klientu!), takže žádné JOINy z SQL nečekejte.
> db.bands.find({name: "Old Blind Dogs"}, {_id: 1}) { "_id" : ObjectId("4b76ef5780b71853fe6c89b6") } > db.albums.insert({name: "New Tricks", band: new DBRef("bands", ObjectId("4b76ef5780b71853fe6c89b6"))}) ObjectId("4b7b1d6bb345126b62fb2230") // findOne vrátí právě jeden dokument > db.albums.findOne() { "_id" : ObjectId("4b7b1d6bb345126b62fb2230"), "name" : "New Tricks", "band" : { "$ref" : "bands", "$id" : ObjectId("4b76ef5780b71853fe6c89b6") } } // fetch načte referencovaný dokument > db.albums.findOne().band.fetch() { "_id" : ObjectId("4b76ef5780b71853fe6c89b6"), "genres" : [ "celtic", "folk", "world music" ], "name" : "Old Blind Dogs", "since" : 1990 } // a možná budeme chtít udělat vazbu obousměrnou > db.bands.update({_id: ObjectId("4b76ef5780b71853fe6c89b6")}, {$push: {albums: new DBRef("albums", ObjectId("4b7b1d6bb345126b62fb2230"))}}) > db.bands.findOne({_id: ObjectId("4b76ef5780b71853fe6c89b6")}) { "_id" : ObjectId("4b76ef5780b71853fe6c89b6"), "albums" : [ { "$ref" : "albums", "$id" : ObjectId("4b7b1d6bb345126b62fb2230") } ], "genres" : [ "celtic", "folk", "world music" ], "name" : "Old Blind Dogs", "since" : 1990 }
MongoDB sice patří do téže rodiny bezschémových databází jako CouchDB, ale zevnitř ani zvnějšku tyto dva produkty příliš podobné nejsou a rozhodně nejsou určené pro tatáž nasazení (vizte též srovnání). Tento úvodní článek pochopitelně není ani zdaleka vyčerpávající (nezmínili jsme například agregace, MapReduce pro dávkové zpracování dat, omezené kolekce, GridFS pro ukládání velkých souborů do MongoDB, provádění uživatelského kódu na straně serveru a řadu dalších věcí), přesto by měl dát čtenáři jistou představu o stylu práce s Mongem.
Vývoj MongoDB zašťiťuje firma 10gen, která je například spřízněna se serverem Business Insider, ovšem asi největším úspěchem je nasazení na serveru SourceForge.
Nástroje: Tisk bez diskuse
Tiskni Sdílej:
Ale že je samotná databáze pod AGPL mi vůbec nevadí, to se naprosté většiny lidí nikdy nedotkne.Vazne? S ohledem na fakt, ze to rozhrani neni standardizovane, tak IMHO kazdy program spada do kategorie odvozeneho dila -> musi byt taky AGPL.
debian/mongodb.upstart
, protože o upstart zatím fakt nestojím. Koukám, že tvůj balíček na něm závislost má. Dá se to nějak řešit? Možná bych měl zase prudit u zdroje