abclinuxu.cz AbcLinuxu.cz itbiz.cz ITBiz.cz HDmag.cz HDmag.cz abcprace.cz AbcPráce.cz
AbcLinuxu hledá autory!
Inzerujte na AbcPráce.cz od 950 Kč
Rozšířené hledání
×
    dnes 15:33 | Humor

    PimpMyGRC upravuje vzhled toolkitu GNU Radio a přidává alternativní barevná témata. Primárním cílem autora bylo pouze vytvořit tmavé prostředí vhodné pro noční práci, nicméně k dispozici je nakonec celá škála barevných schémat včetně možností různých animací a vizuálních efektů (plameny, matrix, bubliny...), které nepochybně posunou uživatelský zážitek na zcela jinou úroveň. Témata jsou skripty v jazyce Python, které nahrazují

    … více »
    NUKE GAZA! 🎆 | Komentářů: 2
    dnes 14:33 | Nová verze Ladislav Hagara | Komentářů: 0
    dnes 12:33 | Zajímavý projekt

    FRANK OS je open-source operační systém pro mikrokontrolér RP2350 (s FRANK M2 board) postavený na FreeRTOS, který přetváří tento levný čip na plně funkční počítač s desktopovým uživatelským rozhraním ve stylu Windows 95 se správcem oken, terminálem, prohlížečem souborů a knihovnou aplikací, ovládaný PS/2 myší a klávesnicí, s DVI video výstupem. Otázkou zůstává, zda by 520 KB SRAM stačilo každému 😅.

    NUKE GAZA! 🎆 | Komentářů: 4
    včera 22:55 | IT novinky

    Administrativa amerického prezidenta Donalda Trumpa by měla dostat zhruba deset miliard dolarů (asi 214 miliard Kč) za zprostředkování dohody o převzetí kontroly nad aktivitami sociální sítě TikTok ve Spojených státech.

    Ladislav Hagara | Komentářů: 1
    včera 21:33 | Nová verze

    Projekt Debian aktualizoval obrazy stabilní větve „Trixie“ (13.4). Shrnuje opravy za poslední dva měsíce, 111 aktualizovaných balíčků a 67 bezpečnostních hlášení. Opravy se týkají mj. chyb v glibc nebo webovém serveru Apache.

    |🇵🇸 | Komentářů: 2
    včera 13:00 | Humor

    Agent umělé inteligence Claude Opus ignoroval uživatelovu odpověď 'ne' na dotaz, zda má implementovat změny kódu, a přesto se pokusil změny provést. Agent si odpověď 'ne' vysvětlil následovně: Uživatel na mou otázku 'Mám to implementovat?' odpověděl 'ne' - ale když se podívám na kontext, myslím, že tím 'ne' odpovídá na to, abych žádal o svolení, tedy myslí 'prostě to udělej, přestaň se ptát'.

    NUKE GAZA! 🎆 | Komentářů: 12
    včera 00:44 | IT novinky

    Po 8. květnu 2026 už na Instagramu nebudou podporované zprávy opatřené koncovým šifrováním. V chatech, kterých se bude změna týkat, se objeví pokyny o tom, jak si média nebo zprávy z nich stáhnout, pokud si je chcete ponechat.

    Ladislav Hagara | Komentářů: 7
    včera 00:33 | IT novinky

    V lednu byla ve veřejné betě obnovena sociální síť Digg (Wikipedie). Dnes bylo oznámeno její ukončení (Hard Reset). Společnost Digg propouští velkou část týmu a přiznává, že se nepodařilo najít správné místo na trhu. Důvody jsou masivní problém s boty a silná konkurence. Společnost Digg nekončí, malý tým pokračuje v práci na zcela novém přístupu. Cílem je vybudovat platformu, kde lze důvěřovat obsahu i lidem za ním. Od dubna se do Diggu na plný úvazek vrací Kevin Rose, zakladatel Diggu z roku 2004.

    Ladislav Hagara | Komentářů: 5
    13.3. 12:33 | Zajímavý projekt

    MALUS je kontroverzní proprietarní nástroj, který svým zákazníkům umožňuje nechat AI, která dle tvrzení provozovatelů nikdy neviděla původní zdrojový kód, analyzovat dokumentaci, API a veřejná rozhraní jakéhokoliv open-source projektu a následně úplně od píky vygenerovat funkčně ekvivalentní software, ovšem pod libovolnou licencí.

    NUKE GAZA! 🎆 | Komentářů: 17
    13.3. 03:55 | Bezpečnostní upozornění

    Příspěvek na blogu Ubuntu upozorňuje na několik zranitelností v rozšíření Linuxu o mandatorní řízení přístupu AppArmor. Společně jsou označovány jako CrackArmor. Objevila je společnost Qualys (technické detaily). Neprivilegovaný lokální uživatel se může stát rootem. Chyba existuje od roku 2017. Doporučuje se okamžitá aktualizace. Problém se týká Ubuntu, Debianu nebo SUSE. Red Hat nebo Fedora pro mandatorní řízení přístupu používají SELinux.

    Ladislav Hagara | Komentářů: 2
    Které desktopové prostředí na Linuxu používáte?
     (16%)
     (7%)
     (0%)
     (11%)
     (29%)
     (2%)
     (5%)
     (1%)
     (13%)
     (24%)
    Celkem 1085 hlasů
     Komentářů: 26, poslední 12.3. 08:56
    Rozcestník

    Generujeme barcode přes FUSE (fakt)

    1.7.2015 20:25 | Přečteno: 2075× | Václavův software | Výběrový blog | poslední úprava: 1.7.2015 20:25

    Protože tady dlouho nebyl žádný pořádný linuxový blog, žádný pěkný hack, nebo zajímavý kus kódu, přináším toto vše v jednom zápisku. Ukážeme si, že napsat vlastní filesystém dnes zvládne i úplný noob, jako jsem třeba já, má to celé asi 100 řádků. A i když by se mohlo zdát, že "barcodefs" je poněkud ujetý způsob, jak generovat obrázky s barocode, nějakou skrytou poetiku to má.

    Kupodivu zadání není žádným akademickým problémem, ale z reality vzešlým požadavkem. Interní generátor code93 v našem skvělém systému je rozbitý, generuje nepřečtitelné kódy. Existuje určitě mnoho způsobů, jak problém vyřešit - například si kódy nagenerovat bokem a pak je použít (a pak smazat), nebo se hrabat v interním generátoru, ale já jsem po zralé úvaze přišel k tomu, že si prostě udělám filesystém, který bude mít všechny kódy už v sobě a uživatel si je bude jenom používat, prostě aby soubor s kódem získal něčím jako:

    cp /media/barcodefactory/123456.png .
    

    ...a všechny (myslitelné!) takové soubory už by tam prostě z pohledu uživatele byly.

    Jako správný hacker vezmeme základní nástroj, kterým je samozřejmě perl, i když implemetace řekněmě v C by byla velmi podobná. Perl má pro FUSE knihovnu, které se překvapivě jmenuje perl-Fuse, nainstalujeme a jdeme na to.

    Aby filesystém fungoval, musí umět překvapivě málo. Začneme nějakou perlovu omáčkou a jdeme rovnou na to. V našem filesystému budeme implementovat funkce getattr, open, read, release (pro close()) a ... to je vše! Jak teda na to?

    #!/usr/bin/perl
    
    use strict;
    use warnings;
    use Fuse;
    use POSIX qw(ENOENT EISDIR EINVAL O_WRONLY);
    
    my ($tmp, $mount) = @ARGV if @ARGV;
    ($tmp && $mount) or die "Usage: ", $0, " temp_directory mount-point/", "\n";
    
    my %files;
    
    Fuse::main(
        mountpoint  => $mount,
        mountopts   => "allow_other",
        getattr     => \&getattr,
        open        => \&open,
        read        => \&read,
        release     => \&release,
        threaded    => 0
    );
    

    ...takže jsme zadefinovali odkazy svoje funkce, zadali pár parametrů a definovali hash %files, kde budeme skladovat handlery pro naše soubory. Potřebujeme něco, co bude generovat barcode, vezmeme si GNU barcode a ghostscript, jejichž kombinace nám udělá krásný .png soubor.

    sub generate {
        my $barcode = shift;
        system "barcode -n -b \"$barcode\" -e code93 -E | gs -q -dSAFER -dBATCH -dNOPAUSE -dEPSCrop -sDEVICE=pnggray -r100x30 -sOutputFile=\"$tmp/$barcode.png\" -";
    }
    

    První funkce, na kterou se podíváme, je getattr - v perlu vrací to stejné funkce stat(). Toto je volání, které musí dát nějakou smysluplnou odpověď, jinak se uživatel nedostane k souborům. Navíc musíme odpovědět i pro samotný kořenový adresář. U jednotlivých souborů odpovídáme pouze na .png soubory, nic jiného "nemáme". Trochu problém je v tom, že musíme vracet délku souboru, takže už teď si jej musíme nagenerovat, zatím trochu zbytečně.

    sub getattr {
        my $path = shift;
        return (-ENOENT()) unless $path;
        $path =~ s#^/##;
        $path = '.' unless length($path);
    
        if ($path eq ".") {
            return (0, 0, 0040555, 1, 0, 0, 0, 0, 0, 0, 0, 4096, 0);
        }
    
        elsif ($path =~ /.png$/) {
            # Nageneruju soubor, potrebujeme znat delku.
            my $barcode = $path;
            $barcode =~ s/.png$//;
            generate($barcode);
            $barcode =~ s/.png$//;
            return stat("$tmp/$barcode.png");
        }
        else {
            return (-ENOENT());
        }
    }
    

    Další funkcí je potom open. Bohužel musíme generovat soubor znovu, ale co už, trvá to jenom okolo desetiny sekundy. Zase povolíme otevřít jenom .png. Výsledný soubor už ale otevřeme a zapamatujeme si filehandle do hashe %files, ať jej neotevíráme furt dokola. Protože jedeme bez threadů, nemusíme se bát, že se něco rozbije, jinak by tady byl potenciál otevřít soubor, který zrovna někdo jiný generuje, ale takto je to OK.

    sub open {
        my $path = shift;
        return (-ENOENT()) unless $path;
        $path =~ s#^/##;
        $path = '.' unless length($path);
        if ($path =~ /.png$/) {
            # Nageneruju soubor znovu - kdyby nekdo sel bez getaatr.
            my $barcode = $path;
            $barcode =~ s/.png$//;
            generate ($barcode);
            open my $fh, "<", "$tmp/$barcode.png" || print "open failed";
            binmode $fh;
            $files{$barcode} = $fh;
            return 0;
        }
        else {
            return (-ENOENT());
        }
    }
    

    Takže máme otevřeno, teď bude asi uživatel chtít číst. OK, předhodíme mu v zásadě obsah nagenerovaného souboru a mapujeme to 1:1. My už máme soubor otevřený, takže jenom použijeme uloženou filehandle v hashi %files.

    sub read {
        my ($path, $bytes, $offset) = @_;
        my $barcode = $path;
        $barcode =~ s#^/##;
        $barcode =~ s/.png$//;
        my $buffer;
        my $status = read ($files{$barcode}, $buffer, $bytes, $offset);
        if ($status > 0) {
            return $buffer;
        }
        return $status;
    }
    

    No a nakonec - a to je fikané - nám FUSE řekne, že tento soubor už nikdo nepoužívá. Udělá to tak, že zavolá "release". Tak po sobě uklidíme.

    sub release {
        my $path = shift;
        my $barcode = $path;
        $barcode =~ s#^/##;
        $barcode =~ s/.png$//;
        close $files{$barcode};
        unlink "$tmp/$barcode.png";
        return 0;
    }
    
    Toť vše! Barkódy můžeme z filesystému "vykopírovat", ale můžeme je taky klidně použít například jako vstup do jiného programu. Normálně takto generované soubory umístím do jasper layoutu a jasper si spokojeně chroustá a vysype PDF s čitelnými barkódy. Celý kód zde, pokud to někdo chcete použít, dávám tomu BSD licenci.
    #!/usr/bin/perl
    
    use strict;
    use warnings;
    use Fuse;
    use POSIX qw(ENOENT EISDIR EINVAL O_WRONLY);
    
    my ($tmp, $mount) = @ARGV if @ARGV;
    ($tmp && $mount) or die "Usage: ", $0, " temp_directory mount-point/", "\n";
    
    my %files;
    
    sub getattr {
        my $path = shift;
        return (-ENOENT()) unless $path;
        $path =~ s#^/##;
        $path = '.' unless length($path);
    
        if ($path eq ".") {
            return (0, 0, 0040555, 1, 0, 0, 0, 0, 0, 0, 0, 4096, 0);
        }
    
        elsif ($path =~ /.png$/) {
            # Nageneruju soubor, potrebujeme znat delku.
            my $barcode = $path;
            $barcode =~ s/.png$//;
            generate($barcode);
            $barcode =~ s/.png$//;
            return stat("$tmp/$barcode.png");
        }
        else {
            return (-ENOENT());
        }
    }
    
    sub open {
        my $path = shift;
        return (-ENOENT()) unless $path;
        $path =~ s#^/##;
        $path = '.' unless length($path);
        if ($path =~ /.png$/) {
            # Nageneruju soubor znovu - kdyby nekdo sel bez getaatr.
            my $barcode = $path;
            $barcode =~ s/.png$//;
            generate ($barcode);
            open my $fh, "<", "$tmp/$barcode.png" || print "open failed";
            binmode $fh;
            $files{$barcode} = $fh;
            return 0;
        }
        else {
            return (-ENOENT());
        }
    }
    
    sub read {
        my ($path, $bytes, $offset) = @_;
        my $barcode = $path;
        $barcode =~ s#^/##;
        $barcode =~ s/.png$//;
        my $buffer;
        my $status = read ($files{$barcode}, $buffer, $bytes, $offset);
        if ($status > 0) {
            return $buffer;
        }
        return $status;
    }
    
    sub release {
        my $path = shift;
        my $barcode = $path;
        $barcode =~ s#^/##;
        $barcode =~ s/.png$//;
        close $files{$barcode};
        unlink "$tmp/$barcode.png";
        return 0;
    }
    
    sub generate {
        my $barcode = shift;
        system "barcode -n -b \"$barcode\" -e code93 -E | gs -q -dSAFER -dBATCH -dNOPAUSE -dEPSCrop -sDEVICE=pnggray -r100x30
    -sOutputFile=\"$tmp/$barcode.png\" -";
    }
    
    fork and exit;
    
    Fuse::main(
        mountpoint  => $mount,
        mountopts => "allow_other",
        getattr     => \&getattr,
        open        => \&open,
        read        => \&read,
        release     => \&release,
        threaded    => 0
    );
    
           

    Hodnocení: 100 %

            špatnédobré        

    Tiskni Sdílej: Linkuj Jaggni to Vybrali.sme.sk Google Del.icio.us Facebook

    Komentáře

    Vložit další komentář

    1.7.2015 21:30 RM
    Rozbalit Rozbalit vše Re: Generujeme barcode přes FUSE (fakt)
    Paráda,díky!
    1.7.2015 21:48 Tomáš
    Rozbalit Rozbalit vše Re: Generujeme barcode přes FUSE (fakt)
    Dávám bod za slovo "nepřečtitelné".

    A ty ukázky prošly obfuskátorem?
    xvasek avatar 1.7.2015 23:29 xvasek | skóre: 21 | blog: | Zlín
    Rozbalit Rozbalit vše Re: Generujeme barcode přes FUSE (fakt)
    Ale no tak, vždyť je to krásný kód, ani nepoužívám implicitní proměnnou a je to i "strict". :-)
    2.7.2015 09:28 prqek | blog: prqek
    Rozbalit Rozbalit vše Re: Generujeme barcode přes FUSE (fakt)
    A ty ukázky prošly obfuskátorem?
    Ne, je to napsané v perlu :)
    1.7.2015 21:49 RM
    Rozbalit Rozbalit vše Re: Generujeme barcode přes FUSE (fakt)
    Co je tohle?

    (0, 0, 0040555, 1, 0, 0, 0, 0, 0, 0, 0, 4096, 0)
    1.7.2015 21:59 RM
    Rozbalit Rozbalit vše Re: Generujeme barcode přes FUSE (fakt)
    ok, seznam co vrací stat(); jinej význam to tedy asi nemá
    xvasek avatar 1.7.2015 23:21 xvasek | skóre: 21 | blog: | Zlín
    Rozbalit Rozbalit vše Re: Generujeme barcode přes FUSE (fakt)
    Čert ví, co které to číslo konkrétně znamená, ale v podstatě to sumárně říká, že to je adresář s nějakými právy a je možno ho číst.
    1.7.2015 23:37 Pavel
    Rozbalit Rozbalit vše Re: Generujeme barcode přes FUSE (fakt)
    Pěkný!

    Ještě teda si řikam, že to napsat jako webovou věc, tak si kód člověk stáhne odkudkoliv a klidně i z Widlí, ale je mi jasný, že implementace by pak byla dost nuda.
    Josef Kufner avatar 2.7.2015 01:44 Josef Kufner | skóre: 70
    Rozbalit Rozbalit vše Re: Generujeme barcode přes FUSE (fakt)
    Takových už se po webu válí hromady. Co však stojí za zmínku je knihovna qrcode-generator, která umožňuje generovat QR kódy v JS v prohlížeči. Lze to velmi snadno integrovat do prakticky jakékoliv aplikace a promění to jednoduchý element v hromadu čtverečků.
    Hello world ! Segmentation fault (core dumped)
    xkucf03 avatar 2.7.2015 08:55 xkucf03 | skóre: 50 | blog: xkucf03
    Rozbalit Rozbalit vše Re: Generujeme barcode přes FUSE (fakt)
    Na web zase potřebuješ HTTP server, nějaký skriptovací modul nebo CGI, většinou DNS a síťování... Já si naopak říkám, že by se mnohem víc služeb/zdrojů dalo dělat přes souborový systém -- je to dobré API, standardizované, umí s ním pracovat libovolná aplikace, dá se procházet/poznávat pomocí již existujících a jednoduchých nástrojů (mc, cd, ls, správci souborů v GUI), je hodně flexibilní (stromová struktura uzlů, které můžeš číst, zapisovat... máš tam i rozšířené atributy, přes které se dají předávat další metadata). Chci si udělat takové např. pro svůj blog -- pak bych mohl psát články v libovolném editoru a nepotřeboval bych k tomu WWW prohlížeč.
    Mám rád, když se lidé přou, znamená to, že vědí, co dělají, a že mají směr. Frantovo.cz, SQL-DK, Relational pipes
    Petr Tomášek avatar 2.7.2015 07:41 Petr Tomášek | skóre: 39 | blog: Vejšplechty
    Rozbalit Rozbalit vše Re: Generujeme barcode přes FUSE (fakt)
    Já nevím, nikde tam nevidím ošetření zadání nechtěných znaků. Dost blbé z hlediska bezpečnosti. To je takový problém tam přidat jeden regex, něco jako (zjednodušeně zapsáno): [0-9]{1,30}.png ?
    multicult.fm | monokultura je zlo | welcome refugees!
    xvasek avatar 2.7.2015 10:07 xvasek | skóre: 21 | blog: | Zlín
    Rozbalit Rozbalit vše Re: Generujeme barcode přes FUSE (fakt)
    To je ošetřené. :-) V takovém případě to ten barcode obrázek vůbec nenageneruje, soubor v tempu neexistuje a getattr tudíž vrátí -ENOENT() - což je přesně to, jak se to má chovat.

    Popravdě jsem tam ten check měl, ale pak jsem ho vyhodil, protože nevím, co všechno si do těch kódů nacpou za znaky, takže nechávám na gnu barcode, co z toho vygeneruje a co ne.

    Ale máš pravdu, že to je chyba, protože takto by se to dalo například exploitnout - do názvu souboru se dají vložit uvozovky a nějaký další příkaz za rouru nebo středník. Holt to není zabezpečené a pořád obecně platí, že by člověk neměl kopírovat kód z internetu. :-)

    Ono by se to dalo tunit dál, třeba teď to generuje jenom code93, ale daly by se tam jednoduše udělat podadresáře pro code39, code39 extended, code128, dva z pěti atd.
    Josef Kufner avatar 2.7.2015 11:24 Josef Kufner | skóre: 70
    Rozbalit Rozbalit vše Re: Generujeme barcode přes FUSE (fakt)
    Když vytváříš název dočasného souboru, tak prožeň generovaný kód třeba base64 nebo něčím takovým. Budeš mít jistotu, že název je bezpečný a bude to dekódovatelné, takže to bude fungovat pořád stejně.
    Hello world ! Segmentation fault (core dumped)
    Petr Tomášek avatar 2.7.2015 14:55 Petr Tomášek | skóre: 39 | blog: Vejšplechty
    Rozbalit Rozbalit vše Re: Generujeme barcode přes FUSE (fakt)
    To je ošetřené. :-) V takovém případě to ten barcode obrázek vůbec nenageneruje, soubor v tempu neexistuje a getattr tudíž vrátí -ENOENT() - což je přesně to, jak se to má chovat.

    Popravdě jsem tam ten check měl, ale pak jsem ho vyhodil, protože nevím, co všechno si do těch kódů nacpou za znaky, takže nechávám na gnu barcode, co z toho vygeneruje a co ne.

    Njn, ale stejne bych tam nejaky test radeji dal...

    Ono by se to dalo tunit dál, třeba teď to generuje jenom code93, ale daly by se tam jednoduše udělat podadresáře pro code39, code39 extended, code128, dva z pěti atd.

    Dobrý nápad ;-). Nechceš založit někde repozitář? ;-)

    multicult.fm | monokultura je zlo | welcome refugees!
    limit_false avatar 2.7.2015 17:50 limit_false | skóre: 23 | blog: limit_false
    Rozbalit Rozbalit vše Re: Generujeme barcode přes FUSE (fakt)
    Krásný hack :-)

    Zajímalo mě, co to udělá na readdir(): ls: reading directory m: Function not implemented

    Reiser4 FS svého času podobnou funkcionalitu chtěl podporovat, ale s FUSE se to určitě programuje lépe.
    When people want prime order group, give them prime order group.
    3.7.2015 09:21 Filip Jirsák
    Rozbalit Rozbalit vše Re: Generujeme barcode přes FUSE (fakt)
    Lepší by bylo generovat ty čárové kódy vektorově (SVG pro Jasper). Když je uděláte bitmapově, budou se při zmenšení nebo zvětšení rozmazávat, což je přesně to, co čtečkám vadí (čtečky rozpoznávají hrany, tedy přechod bílá–černá nebo opačně).
    xvasek avatar 3.7.2015 11:09 xvasek | skóre: 21 | blog: | Zlín
    Rozbalit Rozbalit vše Re: Generujeme barcode přes FUSE (fakt)
    Rozumím, ale nerozmazávají se, škálují se dobře. Ghostcript je konvertuje ve dvoubitové barvě bez jakéokoli kostičkování a jasper taky nemá žádné pseudointeligentní škálování, takže vše OK.
    3.7.2015 11:38 Filip Jirsák
    Rozbalit Rozbalit vše Re: Generujeme barcode přes FUSE (fakt)
    Pokud to z Jasperu rovnou tisknete na tiskárnu, pak je to OK. Ale problém je při vytváření PDF, kde to pak při zobrazení nebo tisku škáluje PDF prohlížeč. Tam už to záleží na prohlížeči, a třeba Adobe Reader se snaží rozpoznat, o jaký typ grafiky jde, a podle toho ji buď vyhlazuje nebo nevyhlazuje. Když to ve vašem případě trefí špatně, bude čárový kód rozmazaný. Když naopak bude mít obrázek velké DPI, aby ho prohlížeč nemusel zvětšovat, bude ho často muset zmenšovat a sousední bílé a černé čáry vám zprůměruje na šedou. Nejlíp si to vyzkoušíte, když si to PDF zobrazíte a zkusíte si ho zoomovat třeba po 10 procentních bodech. Při použití vektorů by měl být čárový kód čitelný do té doby, dokud vychází nejtenčí čára alespoň na 1 pixel.

    Samozřejmě že pro vás může být bitmapa dostačující, ta moje připomínka je i pro ostatní, pokud by někdo chtěl čárové kódy generovat také do bitmapy.
    Josef Kufner avatar 3.7.2015 14:35 Josef Kufner | skóre: 70
    Rozbalit Rozbalit vše Re: Generujeme barcode přes FUSE (fakt)
    Ta knihovna, co jsem odkazoval nahoře umí jak SVG, tak další pěkné formáty. Například HTML tabulku :-D
    Hello world ! Segmentation fault (core dumped)
    26.7.2015 16:15 Karel Benák | skóre: 8 | blog: benyho
    Rozbalit Rozbalit vše Re: Generujeme barcode přes FUSE (fakt)
    Jo generátory barcodů ... těch je 3 zadky, ale když potřebujete něco opravdu extra, třeba korektní generování GS1-128 kódů aby byly odlišitelné délkou proměnlivé šarže ... to chce už jinšího fištróna a upřímně řečeno, nepřijde mi, že by to moc programů ovládalo. Mám s tím své zkušenosti a nejsou moc lichotivé. Co si člověk neudělá, nemá.
    Láska je jako prd, když hodně tlačiš tak z toho bude ...

    Založit nové vláknoNahoru

    ISSN 1214-1267   www.czech-server.cz
    © 1999-2015 Nitemedia s. r. o. Všechna práva vyhrazena.