Set default sound level of FiiO K1 on plug

FiiO K1 became very silent recently when you plug it in on Ubuntu for some reason.

To fix it add an UDEV rule:

/etc/udev/rules.d/71-fiio-k1.rules

SUBSYSTEMS=="usb", ACTION=="add|change", ATTRS{manufacturer}=="FiiO", ATTRS{product}=="FiiO USB DAC K1", RUN+="/home/ajnasz/bin/init-fiio-k1.sh"

And create the shelscript which set the volume whenever you plug the DAC:

/home/ajnasz/bin/init-fiio-k1.sh

#!/bin/sh

CARD_ID="$(aplay -l | grep "FiiO.*K1" | cut -d ' ' -f 2 | tr -d ':')"

if [ -z "$CARD_ID" ];then
    return
fi

amixer -c "$CARD_ID" sset PCM 90%

ThinkPad T510 Xorg NVidia config

To enable screen backlight brightness adjustment, create the following file and restart the X server

DBus Error - xinitrc

I tried to use Spotify dbus interface to play/pause tracks, found a nice shell script but it did not work, but instead I got this error message:

Error org.freedesktop.DBus.Error.ServiceUnknown: The name org.mpris.MediaPlayer2.spotify was not provided by any .service files

Turned out that the issue is probably with my X11 session initialization.

Note: At the very least, ensure that the last if block in /etc/X11/xinit/xinitrc is present in your ~/.xinitrc file to ensure that the scripts in /etc/X11/xinit/xinitrc.d are sourced.

So, I tried to source the /etc/X11/Xsession before I start the window manager:

. /etc/X11/Xsession
exec i3

(i3 is my window manager)

Called startx but no luck, X did not start. The following error was in the ~/.xsession-errors:

dbus-update-activation-environment: warning: error sending to systemd: org.freedesktop.DBus.Error.InvalidArgs: Invalid environment assignments

I had no clue what would that mean, however after some echo based debugging in /etc/X11/Xsession I could confirm that the files from /etc/X11/Xsession.d were sourced and the last file which was sourced is /etc/X11/Xsession.d/99x11-common_start. That file is a one liner:

exec $STARTUP

I guessed that $STARTUP is undefined. Checked where it would be defined and I found /etc/X11/Xsession.d/50x11-common_determine-startup

If no X session startup program was passed to the Xsession script as an argument (e.g., by the display manager), or if that program was not executable, fall back to looking for a user's custom X session script, if allowed by the options file.

That means I can define it somehow, so I tried by setting it in my:

export STARTUP=i3
. /etc/X11/Xsession

Then everything looks good. Some magic happens in those files (probably in /etc/X11/Xsession.d/20dbus_xdg-runtime) what I didn't tried to figure out yet.

Hűtés csere

ThinkPad T510 Fan

Legutóbb írtam, hogy RAM-ot vettem a 8 éves notebookba, ami nem pont olyan, mint amilyet szerettem volna, de azért végül megtartottam. Abban a cikkben írtam azt is, hogy némi FAN ERROR-ral üzenetet is kaptam a csere után, ami végül elmúlt. Sajnos nem múlt el, hanem állandóvá vált. Így használhatatlan lett szegény gép. Az interneten írták, hogy nem egyedi a gond és a legjobb megoldás a csere. Szétnéztem ebay-en, majd rendeltem is egy vadonatújnak mondottat. Ez volt 2019. március 3-án, a érkezése március 19 és április 17 között várható. Másnap útnak indult Kínából, egy héttel később pedig elégedetten láttam, hogy már Magyarországon volt a csomag. Nagyjából fél nappal később, március 13-án pedig le is szállt vele repülő Ferihegyen. Abban is bíztam, hogy a Magyar Posta a nagy nemzeti ünnep előtt kiszállítja. Ez sajnos nem sikerült. Viszont legalább a megadott legkésőbbi érkezéshez képest csupán egy nap késéssel, április 18-án csütörtökön már itt is volt a postai értesítő, hogy csomagom jött külföldről. Remélem nem túl költséges a csomagok tárolása.

Így, hogy már meg is jött a hűtő (réz borda és ventilátor), ma már időm is volt rá, hogy megpróbáljam beszerelni. Két youtube videó támogatásával nekiálltam kicsavarozni az összes csavart a laptop aljáról, és nagy nehezen a régi ventit is sikerült kivenni. Itt jegyezném meg, hogy a WD-40 nagyon jó csavarlazító és emellett remélem a gépben kárt nem tesz. Letisztítottam a korábbi hővezető pasztákat, aztán kicsi feszegetéssel betettem az új hútőt, ami csak minimálisan különbözik a régitől. Egy leszorító rögzítés van másképp megoldva meg a hátsó borda szélesebb egy fél centivel, de volt neki hely amit most rendesen kitölt. Összeszereltem félig, és bekapcsoltam, nem is sípolt, csak nem tudott bootolni, mivel az SSD-t kifelejtettem. Azt is bepattintottam, és láttam a még nem visszaszerelt billentyűzet helyén, hogy milyen szépen forog az a darab műanyag, ami ventilátor néven kerül forgalmazásba. Pár perc után minden megmaradt csavarnak megtaláltam a helyét, összepattintottam a műanyagokat és immáron az új hűtő halk surrogása felett írom ezt a rövid cikket. Tudom, hogy nagyon régi gép ez már, de 0.1-es load és 1.8G használt memória mellett szerintem elciripel még egy darabig.

Sikerült-e RAM-ot rendelni?

Hynix HMT351S6CFR8C-PB RAM

Karácsonykor bevásároltam udemy-n tanfolyamból, többek között vettem kubernetest is. Amikor 3 VM-et futtattam miatta, akkor kezdett a napokban 8-ik évébe lépő Lenovo T510-ben lévő 4GB memória kevés lenni. Mivel 8 év az mégis csak nyolc év, akku sem a régi, a ventillátor is hangos, meg különben is, de elkezdtem nézegetni, hogy ha új gépet szeretnék, akkor miből lehet válogatni a piacon. Kínálat az van, meg áruk is és akkor még bele sem gondoltam, hogy vajon miféle minőséget gyárthatnak mostanság. A notebook vásárlás helyett úgy gondoltam, először megpróbálok javítani a jelenlegin. Anno az SSD egy nagyon jó beruházás volt, gyakorlatilag amiatt nincs igazán szükségem a mai napig sem újabb számítóra, mi baj lehet? Gőzöm sincs a mai és a 8 évvel ezelőtti hardveriparról. Nem tudom, hogy mi-mivel kompatibilis, hogyan adja a legjobb teljesítményt, csupán halovány emlékeim vannak régebbről, hogy az azonos típusú RAM-ok jó hatással vannak egymásra. Azt nem tudom a mai napig sem, hogy mennyire kell azonosnak lenni.

Nem akartam kockáztatni, igyekeztem detto ugyanolyan RAM-ot beszerezni.

$ sudo dmidecode -t 17

[sudo] password for ajnasz:
# dmidecode 3.0
Getting SMBIOS data from sysfs.
SMBIOS 2.6 present.

Handle 0x002B, DMI type 17, 28 bytes
Memory Device
        Array Handle: 0x002A
        Error Information Handle: No Error
        Total Width: 64 bits
        Data Width: 64 bits
        Size: 4096 MB
        Form Factor: SODIMM
        Set: None
        Locator: DIMM 1
        Bank Locator: Bank 0/1
        Type: DDR3
        Type Detail: Synchronous
        Speed: 1334 MHz
        Manufacturer: 80AD
        Serial Number: 5A48C2FE
        Asset Tag: 1104
        Part Number: HMT351S6BFR8C-H9
        Rank: Unknown

Nem eltéveszthető a Part Number: HMT351S6BFR8C-H9 rész, erre kerestem, és Amazonon találtam.

Meg is rendeltem, hosszas utazás után ma érkezett meg, hazaérvén néhány csavartekerés után rögvest be is pattintottam a gép aljába. Indítás után Fan error hibaüzenet fogadott, ami arra ösztönzött, hogy még jobban szétszedjem és kicsit megtakarítsam a gépet. Kiporoltam a ventit, összeraktam és már majdnem egyből jó is lett, kicsit a seggére kellett verni, de elindult.

Boot után a free -h hihetetlen magas számokat produkált, tehát a RAM működik. Gyanútlanul kiadtam a fenti dmidecode -t 17 commandot, amikor is! (csak az új RAM adatait mutatom):

Handle 0x002C, DMI type 17, 28 bytes
Memory Device
        Array Handle: 0x002A
        Error Information Handle: No Error
        Total Width: 64 bits
        Data Width: 64 bits
        Size: 4096 MB
        Form Factor: SODIMM
        Set: None
        Locator: DIMM 2
        Bank Locator: Bank 2/3
        Type: DDR3
        Type Detail: Synchronous
        Speed: 1600 MHz
        Manufacturer: 80AD
        Serial Number: 41CA7A57
        Asset Tag: 1245
        Part Number: HMT351S6CFR8C-PB
        Rank: Unknown

Jól észrevehető különbség a Part Number és a Speed.

Sikerült-e RAM-ot rendelni? Igen. Azt kaptam, amit rendeltem? Nem.

Viszont ez már így marad, az biztos.

pomodoro.sh

Letöltés

A pomodoro egy módszer, ami segíthet az idő hatékonyabb kihasználásában. Irodai, kreatív munka során különösen hasznos lehet. A lényege, hogy a munkákat rövidebb - alapvetően 25 perces - szakaszokra bontva végezzük el, minden szakasz között rövidebb szünetet tartva. A legfontosabb, hogy egy pomodoro menetben kizárólag az adott feladattal foglalkozzunk. A kizárólag főleg azt jelenti, hogy nem olvasunk e-mailt, mi nem kapcsolódik a témához, nem chat-elünk, nem szólunk senkihez, nem kalandozunk feladataink véget nem érő listájában, hanem csak azt az egy kiválasztott munkát végezzük. Ha letelt az idő, akkor pedig dobjunk el mindent, akkor muszáj bármi mást csinálni, e-mailezni, kismacskát simogatni is. A koncentrált munkavégzés miatt a produktivitásunk hatalmasat nőhet, a szünetek pedig megakadályozzák, hogy idő előtt elfáradjunk.

Mondanom sem kell, a legnehezebb figyelmen kívül hagyni azt a rengeteg ingert. Chat, e-mail, kolléga a másik asztalnál nem ismer viccet.

Természetesen a nagy analóg csörgőóra mellett rengeteg program létezik, ami mást sem tesz, mint visszaszámol, esetleg letilt értesítéseket. Éppen ilyen természetes, hogy nekem egyik sem felel meg tökéletesen és különben is egy igazán egyszerű igényről van szó, amit élvezet mégegyszer megoldani.

Volt nekem erre már IRC/Slack botom (https://github.com/Ajnasz/hubot-pomodoro). Ezekről gyakran lemaradtam, mert az irc másik tmux sessionben fut és hangot sem ad. Hónapokkal ezelőtt kezdtem el csinálni az egyszerűbb megoldást, amit shellben meg tudok pattintani.

sleep 10 && aplay /usr/share/sounds/purple/alert.wav

10 másodperc után egy hangfile lejátszása. A tmux is kiemeli az ablakot, mert változás történt.

sleep $(echo "1 * 5" | bc) && aplay /usr/share/sounds/purple/alert.wav

Itt azt próbáltam, hogy milyen volna bc paranccsal lehetne kiszámolni, hogy meddig tartson a sleep.

sleep $((25 * 60)) && aplay /usr/share/sounds/purple/alert.wav

Látszik, hogy mennyivel szebb, ha a bc sincs bevonva.

i=0;while [ $i -lt $((1 * 1)) ];do sleep 1; i=$(($i + 1)); echo $i;done; aplay /usr/share/sounds/purple/alert.wav

Na itt már látszik, hogy nem fog egy sorban megállni. Csak illene tudni, hogy mennyi idő telt el. Egymás alá írogatja a másodperceket.

i=0;while [ $i -lt $((1 * 5)) ];do sleep 1; i=$(($i + 1)); echo -n "\r$i";done;echo; aplay /usr/share/sounds/purple/alert.wav

Itt pedig az utolsó egy soros, ahol helyben frissítjük a kiírt eltelt időt.

Nagyjából ezután került be minden egy file-ba és lett a neve pomodoro.sh.

Ha pedig már shell script, akkor igyekeztem mindent konfigurálhatóvá tenni. A minden kimerült abban, hogy lehessen másik hangot játszani és tetszőleges időt beállítani. Aztán, ahogy az evolúció megkívánta, a desktop notification is megjelent, hogy már nagyon biztosan lássam, ha véget ért a pomodoro session. Ezzel kapcsolatos, hogy notification daemonként a dunstot használom, majd a notify-send-et is lecseréltem dunstify-ra.

Ez volt a pont, ami után már hetekig hozzá sem kellett nyúlni a scripthez, nem volt ötletem, amitől még fantasztikusabb lehetne. Viszont felismertem, hogy Slack üzenetek még mindig csak jönnek és jönnek, én meg nézem őket, ami a fókusz elvesztésével jár. Nosza, több se kellett, feltúrtam a Slack API dokumentációját, hogy megtudjam, miképpen lehetne scriptből roppant elfoglalttá beállítani magam. Így történt az, hogy a ha van SLACK_TOKEN, akkor a pomodoro menet kezdetekor beállítja a "ne zavarj" státuszt és még egy pofás paradicsom ikont is elhelyez a nevünk mellé, a session végén ezeket természetesen törli. Sőt, újabban nem csak törli, hanem visszaállítja a korábban beállítottra.

Korábban egy gistbe küldtem a változásokat, de a projekt érettsége nyomán úgy döntöttem, hogy egy saját Github repositoryt is megérdemel.

Raspberry Pi külső drive kernel panic

Ha a Raspberryn a root partíció egy külső (USB) drive-on van, akkor ügyelni kell arra, hogy a /boot/cmdline.txt-ben, a root értéke PARTUUID formában legyen megadva, nem pedig /dev/sda1 vagy hasonló módon. Ellenkező esetben Kernel Panic - not syncing: VFS: Unable to mount root fs on unknown-block... és ehhez hasonló hibaüzenet várhat, legkésőbb akkor, amikor egy reboot paranccsal szeretnéd újraindítani a gépet.

A PARTUUID a sudo blkid paranccsal kinyerhető, valami ehhez hasonló lesz az output:

József Attila Telegram bot

távirat

Múlt héten volt a költészet napja és a városszerte kiragasztott versek sokasága sugallta, hogy mennyire jó lenne többet olvasni belőlük.

Mindeközben már régóta szerettem volna valamivel izgalmasabbá tenni a Telegramot és ez egy kiváló alkalom egy robot megszületéséhez.

Így mekről beszereztem a József Attila összest, kicsit megsimogattam, megformáltam, hogy használható legyen a chaten is, majd elkészült Jószef Attila bot, aki kérésre egy véletlenszerű verset küld neked.

Kicsit tovább gondolva jó ötletnek tűnt, ha nem csak kérésre, hanem akár naponta, kérés nélkül küldene egy-egy verset, csak azért, hogy társuljon valami izgalmas, szép a reggeli metrózáshoz. Ezért készült el a József Attila versek Telegram csatorna is. Ehhez csatlakozva naponta egy verset kapsz majd.

HashBackup DigitalOcean Spaces-be

Korábban a Raspberry PI-n keresztül töltöttem le a mentéseket, viszont a minap az SD kártya teljesen tönkrement így gyorsan kellett találnom valami megoldást a mentésre, mert ennél már csak az lenne rosszabb, ha mindenem eltűnne.

Szeretném ha valami rendes helyen lennének a mentések és nem az én összetákolt tárolóimon. Gondoltam, milyen jó lenne erre a DigitalOcean Spaces nevű object storage megoldása.

Ezelőtt duplicity-t használtam, azt hittem, hogy nem lehet egy ördöglakat megoldani, hogy mostantól máshová küldje a mentést. Igazából nem az, de akkor hirtelen képtelen voltam megoldani (azóta már sikerült).

Ezért elkezdtem egy olyan mentési megoldás után kutatni, amit hamar össze tudok rakni. A nagy sietségben a hashackup-ot választottam.

Shell script argument parsing

Ha shell scriptet írunk, gyakran előfordulhat, hogy szeretnénk azt paraméterezhetővé tenni.

Az alapvető módszerem, hogy vannak bizonyos beállítások definiálva a fájl elején egy alapértelmezett értékkel, majd a paraméterektől függően változtatom ezek értékét:

#!/bin/sh

IS_FLAG_A_SET=0

for i in $@; do
    case $i in
        '-a')
            IS_FLAG_A_SET=1
            ;;
        *)
            echo "Usage $0 -a"
            exit 1
    esac
done

if [ $IS_FLAG_A_SET -eq 1 ]; then
    echo "FLAG A"
fi

Lekezeltük még azt is, hogy össze-vissza paramétereket ne engedjünk csak úgy bepasszintani, ekkor ugynis kiírjuk a használati utasítást és hibakóddal kilépünk.

Izgalmasabb a helyzet, ha nem csak flagekkel dolgozunk, hanem szeretnénk valamilyen értéket átadni egy opcióval, mondjuk egy fájlnevet. Ha tudjuk, hogy csak egy ilyen lehet az adott programban, akkor megtehetjük, hogy azt mondjuk, hogy legyen az utolsó paraméter mindig a fájlnév.

FILE_NAME=${@:-1}
echo $FILE_NAME

De ez egyáltalán nem dinamikus, hiszen bármikor lehet olyan igény, hogy több értéket szeretnénk átadni. Ekkor például elkezdhetünk úgy dolgozni a paramétereken, hogy ha például -f kapcsolót kaptuk, akkor a következő paraméter a fájlnévnek vesszük: