Inhaltsverzeichnis
shack open/close monitor
das open/close-monitoring basiert auf der annahme, dass die shack-tuere nur unverschlossen ist, wenn der shack geoeffnet ist.
abstract
der server erhaelt einen cron-job der alle 2 minuten ein cgi-script auf einem webserver aufruft, das wiederum eine von zwei flag-dateien schreibt. welche flag-datei geschrieben wird, ist abhaengig davon, ob der status „open“ oder „closed“ uebermittelt wird. die jeweils nicht geschriebene flag-datei wird dabei geloescht. auf dem webserver wird jede minute geprueft, ob diese datei aelter als 3 minuten ist, in diesem falle wird sie geloescht.
das cgi-script gibt bei aufruf ohne parameter ein pixel als gif zurueck, das abhaengig von der existenz der flagdateien gruen (vorhanden, offen), rot (vorhanden, closed) oder grau (fehlen) ist.
dieses pixel wird vergroessert aus der passenden wiki-seite heraus aufgerufen
weiter kann der status als text auf deutsch und english abgefragt werden
die anzeige ist somit auf 2 minuten genau
trafficabschaetzung
ein simpler http-request der form
GET /sopen/******************************** HTTP/1.0\r\nHost: shackspace.de\r\n\r\n
ergibt:
- 899 Byte/Rq.
- 647.280 KiB/d
- 20.065680 MiB/m
fuer wurschtkaes 20mb im monat hat man den status auf 2 minuten genau.
messung: exemplarischer traffic mit tcpdump beobachtet, 32 '*' in der uri
details & implementierung
Dieser Teil ist obsolet und benötigt einen Neuschrieb
in der tuerzarge ist in die oeffnung fuer den schliessriegel ein reedkontakt mit expoxidharz eingeklebt. in den schliessriegel ist ein magnet eingearbeitet. vom reedkontakt fuehrt eine zweidrahtleitung zur naechsten cat5-dose im lager. das andere ende der leitung ist mit einem cisco-consolekabel ueber einen usb-rs232-adaper mit einem rechner verbunden. benutzt werden dabei nur zwei adern, sie verbinden RTS und DSR (an sich: beliebige statusleitungen, ausgenommen RI, weil nicht durch das ciscokabel gefuehrt), wenn die tuer abgeschlossen ist. wenn die tuere offen ist, bleiben die leitungen unverbunden.
das programm /usr/local/app/shackdoor/shackdoor
setzt RTS und prueft daraufhin DSR. je nach stand des schalters ergibt sich das
eine oder andere ergebnis. da aufgrund der etwa 40-50 m leitungslaenge und des bei geoeffneter tuere hohen abschlusswiderstands
die signalqualitaet schlecht ist, laesst sich die offene tuer nicht immer eindeutig erkennen (und seltsamer weise seit januar 2015
auch der geschlossene kontakt nicht mehr). also findet die abfrage einfach 20 mal statt. das ging schneller, als an
der verkabelung weitere schaltmittel anzubringen.
- shackdoor.c
#include <stdio.h> #include <stdlib.h> #include <sys/ioctl.h> #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> #include <unistd.h> #include <signal.h> #define STIME 3000000 void SetRTS(int fd) { int currstat; ioctl(fd, TIOCMGET, &currstat); currstat |= TIOCM_RTS; ioctl(fd, TIOCMSET, &currstat); } void ResetRTS(int fd) { int currstat; ioctl(fd, TIOCMGET, &currstat); currstat &= ~TIOCM_RTS; ioctl(fd, TIOCMSET, &currstat); } void SetDTR(int fd) { int currstat; ioctl(fd, TIOCMGET, &currstat); currstat |= TIOCM_DTR; ioctl(fd, TIOCMSET, &currstat); } void ResetDTR(int fd) { int currstat; ioctl(fd, TIOCMGET, &currstat); currstat &= ~TIOCM_DTR; ioctl(fd, TIOCMSET, &currstat); } void ClearSerPins(int fd) { int currstat = 0; ioctl(fd, TIOCMSET, &currstat); } int GetRI(int fd) { int currstat; ioctl(fd, TIOCMGET, &currstat); return((currstat & TIOCM_RNG)?1:0); } int GetCD(int fd) { int currstat; ioctl(fd, TIOCMGET, &currstat); return((currstat & TIOCM_CAR)?1:0); } int GetDSR(int fd) { int currstat; ioctl(fd, TIOCMGET, &currstat); return((currstat & TIOCM_DSR)?1:0); } int GetCTS(int fd) { int currstat; ioctl(fd, TIOCMGET, &currstat); return((currstat & TIOCM_CTS)?1:0); } int GetSerStat(int fd) { int currstat; ioctl(fd, TIOCMGET, &currstat); return(currstat); } int main (int argc, char** argv) { int i, stat; int fd = 0; if ( argc != 2 ) { printf("usage: %s device\n",argv[0]); exit(1); } if (( fd = open(argv[1], O_RDWR | O_NDELAY )) < 0 ) { printf("can't open device '%s'\n",argv[1]); exit(2); } SetRTS(fd); stat = GetSerStat(fd); if ( stat & TIOCM_DSR) puts("DSR "); printf("\n"); /* debug for (;;) { usleep(100000); stat = GetSerStat(fd); if ( stat & TIOCM_RNG) puts("RI "); if ( stat & TIOCM_CAR) puts("CAR "); if ( stat & TIOCM_DSR) puts("DSR "); if ( stat & TIOCM_CTS) puts("CTS "); printf("\n"); } */ close(fd); return(0); }
die tuerstatus-kvm prueft jede minute per cronjob den zustand des tuerkontakts:
- /etc/cron.d/sopen-shack
0-59/1 * * * * nobody /usr/local/app/shackdoor/sopen-client 2>&1 | logger
sopen-client fragt den zustand der tuere ab und setzt einen http-request an shackspace.de ab, der den zustand uebermittelt:
- /usr/local/app/shackdoor/sopen-client
#!/bin/bash set -u #old# closed loop (=closed door) is more reliable to detect. #old# so, if there are $CHECKS checks in favour to the door #old# being closed, we might assume that this is the case. # as of 2015-01 things got fishy # so we just look who got the most votes # DEV=/dev/ttyUSB0 if [ ! -e "$DEV" ]; then logger -t "sopen-client" "'$DEV' missing: $( ls -ld /dev )" exit fi CHECKS=20 QUOTA=$(( $CHECKS / 2 )) FLAG=0 for i in $( seq 1 $CHECKS ); do if [ x`/usr/local/app/shackdoor/shackdoor /dev/ttyUSB0` = xDSR ] ; then FLAG=$(( $FLAG + 1 )) fi done echo FLAG=$FLAG if [ "$FLAG" -lt "$QUOTA" ] ; then echo -e -n "GET /sopen/open/************ HTTP/1.0\r\nHost: shackspace.de\r\n\r\n" | nc shackspace.de 80 else echo -e -n "GET /sopen/closed/************ HTTP/1.0\r\nHost: shackspace.de\r\n\r\n" | nc shackspace.de 80 fi
trouble
die erkennung der dsr-leitung funktioniert nicht in jedem falle zuverlaessig, insbesondere der zustand „tuere offen“, bei dem die Leiterschleife geoeffnet ist, ist verrauscht, weil hochohmig. das sieht dann so aus:
die leitungslaenge macht das sicher nicht besser. daher wird der zustand mehrfach abgefragt und eine mehrheitsentscheidung getroffen.
das usb-rs232-dingsy faellt manchmal logisch vom host und braucht dann einen powercycle (aus- und wieder einstecken). deshalb prueft sopen-client extra, ob /dev/ttyUSB0 da ist.
auf shackspace.de
sopen
ist ein CGI und wurde als /var/www/shackspace.de/htdocs/sopen
abgelegt:
- sopen
#!/usr/bin/perl -Tw # Thu Nov 29 17:52:13 CET 2012, chris # Wed Jan 7 11:35:36 CET 2015, chris # - added three-state display # # - Doku hier: # - http://shackspace.de/wiki/doku.php?id=shack_open_close_monitor use strict; my $uri_c = "/sopen/closed/**********"; my $uri_o = "/sopen/open/**********"; my $uri_text_de = "/sopen/text/de"; my $uri_text_en = "/sopen/text/en"; my $uri_img_png = "/sopen/img/png"; my $uri_img_gif = "/sopen/img/gif"; my $uri_imgtxt_png = "/sopen/imgtxt/en/png"; my $uri_imgtxt_gif = "/sopen/imgtxt/en/gif"; my $statfilename = "/var/run/sopen/stat"; my $trackfilenameon = "/var/run/sopen/track-on"; my $trackfilenameoff = "/var/run/sopen/track-off"; # strip trailing ?... my $request_uri = (split(/\?/,$ENV{REQUEST_URI}))[0]; # korrekte uri setzt flag file $request_uri eq $uri_o and do { open F,">".$statfilename or die "can't open '$statfilename': $!"; print F time ," OPEN\n"; close F; -e $trackfilenameon or do { open F,">".$trackfilenameon or die "can't open '$trackfilenameon': $!"; print F time ,"\n"; close F; }; -e $trackfilenameoff and do { unlink $trackfilenameoff; }; print "Content-Type: text/plain\n\n1\n"; exit; }; # else if $request_uri eq $uri_c and do { open F,">".$statfilename or die "can't open '$statfilename': $!"; print F time ," CLOSED\n"; close F; -e $trackfilenameon and do { unlink $trackfilenameon; }; -e $trackfilenameoff or do { open F,">".$trackfilenameoff or die "can't open '$trackfilenameoff': $!"; print F time ,"\n"; close F; }; print "Content-Type: text/plain\n\n1\n"; exit; }; # else if $request_uri eq $uri_text_de and do { print "Content-Type: text/plain\n\n"; -f $trackfilenameon and do { print "offen\n"; } or -f $trackfilenameoff and do { print "geschlossen\n"; } or do { print "keine Daten\n"; }; exit; }; # else if $request_uri eq $uri_text_en and do { print "Content-Type: text/plain\n\n"; -f $trackfilenameon and do { print "open\n"; } or -f $trackfilenameoff and do { print "closed\n"; } or do { print "no data\n"; }; exit; }; # else if $request_uri eq $uri_img_png and do { print "Content-Type: image/png\n\n"; -f $trackfilenameon and do { # pixel gruen print "\211\120\116\107\15\12\32\12\0\0\0\15\111\110\104\122\0\0\0\1\0", "\0\0\1\1\3\0\0\0\45\333\126\312\0\0\0\4\147\101\115\101\0\0\261", "\217\13\374\141\5\0\0\0\3\120\114\124\105\0\377\0\64\136\300", "\250\0\0\0\70\164\105\130\164\123\157\146\164\167\141\162\145", "\0\130\126\40\126\145\162\163\151\157\156\40\63\56\61\60\141\40", "\40\122\145\166\72\40\61\62\57\62\71\57\71\64\40\50\120\116\107", "\40\160\141\164\143\150\40\61\56\62\51\335\25\56\111\0\0\0\12", "\111\104\101\124\170\234\143\140\0\0\0\2\0\1\110\257\244\161", "\0\0\0\7\164\111\115\105\7\333\13\27\7\64\73\1\135\24\55\0\0\0", "\0\111\105\116\104\256\102\140\202"; } or -f $trackfilenameoff and do { # pixel rot print "\211\120\116\107\15\12\32\12\0\0\0\15\111\110\104\122\0\0\0\1\0", "\0\0\1\1\3\0\0\0\45\333\126\312\0\0\0\4\147\101\115\101\0\0\261", "\217\13\374\141\5\0\0\0\3\120\114\124\105\377\0\0\31\342\11\67\0", "\0\0\70\164\105\130\164\123\157\146\164\167\141\162\145\0\130\126", "\40\126\145\162\163\151\157\156\40\63\56\61\60\141\40\40\122\145", "\166\72\40\61\62\57\62\71\57\71\64\40\50\120\116\107\40\160\141", "\164\143\150\40\61\56\62\51\335\25\56\111\0\0\0\12\111\104\101", "\124\170\234\143\140\0\0\0\2\0\1\110\257\244\161\0\0\0\7\164\111", "\115\105\7\333\13\27\7\65\17\71\362\321\331\0\0\0\0\111\105\116", "\104\256\102\140\202"; } or do { # pixel grey print "\211\120\116\107\15\12\32\12\0\0\0\15\111\110\104\122\0\0\0\1\0", "\0\0\1\1\3\0\0\0\45\333\126\312\0\0\0\4\147\101\115\101\0\0\261", "\217\13\374\141\5\0\0\0\3\120\114\124\105\177\177\177\220\312\33", "\43\0\0\0\70\164\105\130\164\123\157\146\164\167\141\162\145\0\130", "\126\40\126\145\162\163\151\157\156\40\63\56\61\60\141\40\40\122", "\145\166\72\40\61\62\57\62\71\57\71\64\40\50\120\116\107\40\160\141", "\164\143\150\40\61\56\62\51\335\25\56\111\0\0\0\12\111\104\101\124", "\10\231\143\140\0\0\0\2\0\1\364\161\144\246\0\0\0\7\164\111\115\105", "\7\337\1\7\13\12\35\32\267\176\341\0\0\0\0\111\105\116\104\256\102", "\140\202"; }; exit; }; # else if $request_uri eq $uri_imgtxt_png and do { print "Content-Type: image/png\n\n"; -f $trackfilenameon and do { # image text gruen print "\211\120\116\107\15\12\32\12\0\0\0\15\111\110", "\104\122\0\0\0\57\0\0\0\25\10\3\0\0", "\0\341\30\107\125\0\0\0\140\120\114\124\105\77", "\277\140\7\367\13\0\376\0\5\371\11\14\362\22", "\23\353\35\27\347\44\25\351\40\31\345\46\33\343", "\52\47\327\74\41\335\63\45\331\70\50\326\76\54", "\322\103\61\315\113\71\305\127\73\303\132\102\274\145", "\110\266\157\116\260\167\122\254\176\127\247\206\130\246", "\206\133\243\214\136\240\220\141\235\225\144\232\231\150", "\226\237\157\217\252\154\222\245\160\217\253\370\234\167", "\162\0\0\0\355\111\104\101\124\170\332\325\222\315", "\162\303\40\14\204\15\210\26\203\377\222\206\70\266", "\43\351\375\337\62\22\156\153\174\354\55\325\1\326", "\322\67\73\153\15\215\375\133\65\357\312\207\344\313", "\355\14\230\135\200\201\242\335\376\155\134\305\307\47", "\63\56\101\324\225\21\171\351\144\42\2\361\41\2", "\107\105\356\363\301\367\230\3\164\13\5\235\246\64", "\314\350\105\304\276\357\223\166\264\157\347\203\167\124", "\264\331\344\152\110\224\347\161\27\5\241\165\3\153", "\363\301\47\214\145\62\261\331\61\340\111\154\103\10", "\255\121\376\223\362\311\277\347\217\302\17\45\206\163", "\160\343\126\362\113\251\221\70\14\174\261\371\376\313", "\7\36\12\177\45\47\274\374\57\136\24\3\51\247", "\371\165\13\135\305\333\165\325\235\171\272\51\26\243", "\246\250\363\313\361\240\265\332\117\240\155\214\23\255", "\120\143\70\152\175\167\340\311\25\157\175\106\246\57", "\70\331\112\56\106\372\351\264\47\136\267\371\136\357", "\355\237\362\57\123\170\25\154\160\141\302\111\0\0", "\0\0\111\105\116\104\256\102\140\202", ; #perl -e '$/="";$c=0;for (split //,<>) { printf "\\%o",ord($_); ++$c%14 or print "\n"}' ~chris/open.png } or -f $trackfilenameoff and do { # image text rot print "\211\120\116\107\15\12\32\12\0\0\0\15\111\110", "\104\122\0\0\0\100\0\0\0\25\10\3\0\0", "\0\301\364\30\250\0\0\0\132\120\114\124\105\377", "\0\0\377\13\13\377\24\24\377\31\31\377\41\41", "\377\53\53\377\63\63\377\73\73\377\104\104\377\112", "\112\377\124\124\377\133\133\377\145\145\377\153\153\377", "\161\161\377\203\203\377\213\213\377\220\220\377\232\232", "\377\243\243\377\263\263\377\274\274\377\307\307\377\313", "\313\377\324\324\377\334\334\377\344\344\377\353\353\377", "\363\363\377\374\374\46\155\212\6\0\0\1\44\111", "\104\101\124\170\332\355\122\111\266\204\40\14\4\34", "\160\150\301\131\132\250\373\137\363\47\212\266\372\372", "\0\177\321\265\200\172\41\103\205\104\210\37\116\344", "\145\116\247\114\116\203\56\263\73\223\11\103\155\104", "\311\107\170\355\1\274\265\150\161\65\70\115\54\173", "\107\326\201\261\104\342\232\133\74\226\132\327\156\242", "\267\303\60\25\111\351\2\251\32\103\231\126\23\51", "\153\121\23\112\116\120\125\315\210\341\23\57\375\302", "\222\244\24\135\330\15\141\346\53\361\124\317\271\350", "\165\44\217\304\242\70\23\24\250\357\116\45\52\261", "\73\51\321\143\66\331\376\246\11\112\304\76\45\354", "\245\3\35\131\33\156\206\27\50\324\322\167\354\55", "\60\50\163\224\31\372\213\202\346\256\100\343\265\335", "\75\266\337\316\73\156\270\103\232\46\251\74\234\324", "\105\201\130\75\217\57\127\347\24\326\267\342\1\204", "\221\222\61\163\263\370\114\150\47\75\162\161\221\340", "\155\323\123\231\16\206\40\111\302\152\52\33\326\204", "\76\330\333\272\105\313\205\115\143\114\263\21\143\27", "\66\135\326\150\2\174\257\342\254\111\116\76\2\141", "\140\135\31\61\14\362\330\3\354\304\117\325\143\225", "\244\172\356\246\372\302\176\370\237\370\3\237\327\25", "\353\351\254\364\365\0\0\0\0\111\105\116\104\256", "\102\140\202", ; } or do { # image text grey print "\211\120\116\107\15\12\32\12\0\0\0\15\111\110", "\104\122\0\0\0\72\0\0\0\25\10\3\0\0", "\0\40\237\15\371\0\0\0\4\147\101\115\101\0", "\0\261\217\13\374\141\5\0\0\0\146\120\114\124", "\105\172\172\172\176\176\176\203\203\203\204\204\204\210", "\210\210\215\215\215\223\223\223\227\227\227\230\230\230", "\235\235\235\240\240\240\245\245\245\251\251\251\255\255", "\255\260\260\260\266\266\266\271\271\271\274\274\274\303", "\303\303\305\305\305\314\314\314\316\316\316\324\324\324", "\327\327\327\331\331\331\337\337\337\340\340\340\344\344", "\344\351\351\351\355\355\355\362\362\362\364\364\364\373", "\373\373\377\377\377\250\34\47\121\0\0\0\70\164", "\105\130\164\123\157\146\164\167\141\162\145\0\130\126", "\40\126\145\162\163\151\157\156\40\63\56\61\60\141", "\40\40\122\145\166\72\40\61\62\57\62\71\57\71", "\64\40\50\120\116\107\40\160\141\164\143\150\40\61", "\56\62\51\335\25\56\111\0\0\0\360\111\104\101", "\124\70\215\355\121\313\162\302\60\14\124\36\206\72", "\61\204\330\111\234\30\374\320\377\377\144\245\200\11", "\320\366\322\351\364\304\36\74\132\215\326\273\262\1", "\336\310\160\366\117\244\107\54\177\57\55\276\16\134", "\106\23\103\107\105\241\3\272\346\332\54\207\20\65", "\113\305\30\222\255\100\42\242\7\250\47\237\346\152", "\223\46\55\7\224\0\66\235\224\345\202\260\244\256", "\165\110\122\65\311\66\116\120\366\270\27\0\207\111", "\66\161\313\162\161\144\210\75\354\120\21\73\73\356", "\355\271\256\322\155\310\370\307\135\7\377\20\230\216", "\250\241\105\301\143\221\173\12\353\333\256\107\117\121", "\103\336\65\263\27\351\352\132\144\127\312\135\122\64", "\201\247\142\275\356\300\256\167\366\44\65\353\256\255", "\305\353\73\55\101\311\231\166\255\320\210\46\4\276", "\254\357\152\142\273\46\276\112\65\71\232\200\347\374", "\302\143\214\206\3\53\237\226\201\207\373\350\77\230", "\315\303\26\370\107\174\363\221\157\374\33\76\1\352", "\14\23\160\224\41\237\53\0\0\0\7\164\111\115", "\105\7\337\1\7\13\67\50\46\354\362\174\0\0", "\0\0\111\105\116\104\256\102\140\202", ; # perl -e '$/="";$c=0;for (split //,<>) { printf "\\%o",ord($_); ++$c%14 or print "\",\n\t\t\t\""}' nodata.png }; exit; }; # else if $request_uri eq $uri_imgtxt_gif and do { print "Content-Type: image/gif\n\n"; -f $trackfilenameon and do { # image text gruen print "\107\111\106\70\71\141\57\0\25\0\365\46\0\77", "\277\140\7\367\13\0\376\0\5\371\11\11\365\16", "\14\362\22\23\353\35\27\347\44\25\351\40\31\345", "\46\33\343\52\47\327\74\41\335\63\45\331\70\50", "\326\76\57\317\110\54\322\103\61\315\113\66\310\123", "\71\305\127\73\303\132\117\257\172\102\274\145\106\270", "\153\110\266\157\116\260\167\122\254\176\127\247\206\125", "\251\202\130\246\206\133\243\214\136\240\220\141\235\225", "\144\232\231\150\226\237\157\217\252\154\222\245\156\220", "\250\160\217\253\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\41\371\4\0\0", "\0\0\0\54\0\0\0\0\57\0\25\0\0\6", "\376\100\201\100\40\20\10\4\2\201\100\40\20\10", "\4\2\201\100\40\20\10\4\2\201\100\40\20\10", "\4\2\201\100\40\20\10\4\2\201\100\40\20\10", "\4\2\201\100\40\20\10\4\2\201\100\40\20\10", "\200\2\201\100\40\20\10\4\2\201\100\40\20\10", "\4\2\201\100\40\20\10\4\2\201\100\40\20\10", "\4\2\201\100\40\20\10\4\2\201\100\40\20\10", "\4\2\201\100\40\20\10\4\2\201\100\40\20\0", "\5\2\201\100\40\20\10\4\2\201\100\40\20\10", "\4\2\201\100\40\20\10\4\2\201\100\40\20\10", "\4\2\201\100\40\20\10\4\2\201\100\40\20\10", "\4\2\201\100\40\20\10\4\2\201\100\40\0\12", "\4\2\201\100\40\20\10\4\11\206\101\40\20\10", "\4\203\100\41\40\20\10\4\202\101\41\120\10\24", "\2\2\301\40\40\20\10\4\202\300\100\40\20\10", "\4\2\201\100\40\130\210\114\246\321\47\1\24\10", "\4\26\323\150\144\372\64\4\2\200\151\64\32\215", "\74\2\1\140\24\21\10\4\202\115\107\40\20\10", "\4\2\201\376\100\340\30\151\22\205\306\207\224\20", "\10\0\43\6\3\322\31\31\4\200\321\2\350\160", "\70\30\2\1\140\104\112\10\4\202\116\107\40\20", "\10\4\2\201\140\100\352\10\4\2\101\40\324\21", "\10\0\44\201\100\140\60\105\4\0\222\100\40\20", "\10\4\2\0\11\24\52\10\4\232\16\120\40\20", "\10\4\2\201\200\61\132\10\4\2\201\140\142\12", "\10\0\44\201\100\120\60\115\4\200\121\42\221\120", "\4\4\2\0\11\101\322\10\4\235\216\100\40\20", "\10\4\2\201\303\164\20\10\4\2\240\0\62\62", "\10\0\243\301\240\220\61\51\4\0\223\311\144\32", "\55\4\2\0\111\0\61\121\4\232\215\100\40\20", "\10\4\2\101\302\4\21\10\4\2\201\205\64\20", "\0\106\243\221\151\104\21\10\0\100\122\241\120\50", "\14\4\2\300\110\40\260\230\32\232\215\100\40\20", "\10\4\2\201\0\4\12\10\4\2\3\51\43\20", "\0\110\213\205\42\40\20\10\0\44\201\100\40\20", "\10\4\0\222\100\40\360\220\200\240\216\100\40\20", "\10\4\2\201\40\101\376\12\105\26\23\22\250\40", "\20\0\110\2\201\100\40\20\10\0\243\110\44\22", "\211\10\4\0\222\100\40\50\210\114\35\201\100\40", "\20\10\4\2\201\300\240\31\1\115\44\114\101\40", "\20\0\110\2\201\100\40\20\10\0\243\221\311\64", "\42\11\4\0\222\100\40\20\50\114\35\201\100\40", "\20\10\4\2\201\100\40\20\4\4\2\201\100\40", "\20\10\4\2\201\100\40\0\12\4\2\201\100\40", "\20\10\4\2\201\100\40\20\10\4\2\201\100\40", "\20\10\4\2\201\100\40\20\10\4\2\201\100\40", "\20\10\4\2\201\100\40\20\10\4\2\201\100\40", "\20\10\4\2\201\100\0\24\10\4\2\201\100\40", "\20\10\4\2\201\100\40\20\10\4\2\201\100\40", "\20\10\4\2\201\100\40\20\10\4\2\201\100\40", "\20\10\4\2\201\100\40\20\10\4\2\201\100\40", "\20\10\4\2\201\0\50\20\10\4\2\201\100\40", "\20\10\4\2\201\100\40\20\10\4\2\201\100\40", "\20\10\4\2\201\100\40\20\10\4\2\201\100\40", "\20\10\4\2\201\100\40\20\10\165\4\2\201\100", "\40\20\10\4\2\1\120\40\20\10\4\2\201\100", "\40\20\10\4\2\201\100\40\20\10\4\2\201\100", "\40\20\10\4\2\201\100\40\20\10\4\2\201\100", "\40\20\10\4\2\201\100\40\20\10\4\2\201\100", "\40\20\10\4\2\240\100\40\20\10\4\2\201\100", "\40\20\10\4\2\201\100\40\20\10\4\2\201\100", "\40\20\10\4\2\201\100\40\20\10\4\2\201\100", "\40\20\10\4\2\201\100\40\20\10\4\2\201\40", "\10\0\73", ; # perl -e '$/="";$c=0;for (split //,<>) { printf "\\%o",ord($_); ++$c%14 or print "\",\n\t\t\t\""}' ~chris/open.gif } or -f $trackfilenameoff and do { # image text rot print "\107\111\106\70\71\141\100\0\25\0\364\35\0\377", "\0\0\377\13\13\377\24\24\377\31\31\377\41\41", "\377\53\53\377\63\63\377\73\73\377\104\104\377\112", "\112\377\124\124\377\133\133\377\145\145\377\153\153\377", "\161\161\377\203\203\377\213\213\377\220\220\377\232\232", "\377\243\243\377\263\263\377\274\274\377\307\307\377\313", "\313\377\324\324\377\334\334\377\344\344\377\353\353\377", "\363\363\377\374\374\0\0\0\0\0\0\41\371\4", "\0\0\0\0\0\54\0\0\0\0\100\0\25\0", "\0\5\376\40\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\200\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\2\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\10\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\40\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\200\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\2\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\10\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\40\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\200\221\30", "\0\0\200\301\0\0\0\0\0\0\0\34\111\1", "\0\0\0\0\300\221\24\0\0\0\301\0\16\303", "\60\10\0\20\14\203\20\0\0\0\0\0\0\0", "\0\0\0\300\262\376\165\35\330\145\7\0\165\0", "\0\0\0\0\54\133\327\165\330\1\0\100\221\165", "\135\207\35\40\0\104\135\327\165\335\5\0\121\327", "\165\35\326\0\0\0\0\0\0\0\0\260\200\335", "\265\34\13\126\1\121\7\0\0\0\0\113\127\41", "\103\202\161\6\0\120\134\2\22\112\65\0\0\324", "\55\313\262\44\0\20\165\212\322\120\335\4\0\0", "\0\0\0\10\0\301\166\5\0\0\4\1\20\161", "\0\0\0\100\300\131\0\0\0\303\166\1\40\200", "\141\0\0\0\0\0\104\35\0\0\0\0\0\121", "\7\0\0\360\164\10\0\200\0\0\0\0\200\164", "\13\0\0\0\0\0\121\7\0\0\0\44\235\2", "\0\0\0\2\117\47\0\122\147\71\5\0\0\100", "\324\35\307\161\10\0\0\165\0\0\0\1\330\75", "\0\0\0\0\0\0\113\167\0\0\0\0\0\0", "\161\0\0\0\300\322\35\40\0\0\0\300\164\5", "\0\74\133\327\125\3\0\100\135\327\165\235\2\0", "\21\7\200\0\0\0\234\4\0\0\0\0\0\200", "\164\15\0\0\0\0\0\121\376\7\0\0\0\2", "\107\307\0\0\0\0\122\27\0\0\0\30\121\67", "\1\100\324\21\304\100\4\0\10\104\35\0\0\200", "\320\75\0\0\0\0\0\0\200\266\15\0\0\30", "\2\0\165\40\0\0\0\0\150\231\0\0\100\301", "\121\0\160\10\0\0\140\26\0\100\35\0\200\0", "\0\0\0\324\1\0\0\110\235\1\0\0\0\0", "\0\0\40\335\366\64\122\7\116\100\324\71\216\343", "\4\300\321\151\216\362\160\332\0\4\333\366\54\120", "\7\1\0\30\165\116\343\70\15\0\104\235\343\74", "\127\7\1\0\0\0\0\0\0\0\140\124\140\327", "\155\222\0\104\135\327\165\335\0\0\6\325\165\334", "\64\0\0\120\120\135\327\201\123\0\0\121\327\165", "\135\327\1\100\324\165\335\126\51\0\0\0\0\0", "\0\0\0\2\0\0\4\2\0\0\0\0\0\0", "\0\0\0\0\0\200\0\0\0\0\0\0\10\0", "\2\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\40\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\376\0\0", "\0\0\0\200\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\2", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\10\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\40\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\200\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\2\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\10\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\40\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\200\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\2\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\10\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\33\0\0\0\0\0\0\0\40\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\100\10", "\0\73", ; # perl -e '$/="";$c=0;for (split //,<>) { printf "\\%o",ord($_); ++$c%14 or print "\",\n\t\t\t\""}' ~chris/closed.gif } or do { # image text grey print "\107\111\106\70\71\141\72\0\25\0\366\120\0\172", "\172\172\173\173\173\176\176\176\177\177\177\200\200\200", "\201\201\201\203\203\203\204\204\204\207\207\207\210\210", "\210\213\213\213\215\215\215\216\216\216\223\223\223\226", "\226\226\227\227\227\230\230\230\233\233\233\235\235\235", "\237\237\237\240\240\240\243\243\243\245\245\245\247\247", "\247\250\250\250\251\251\251\253\253\253\255\255\255\256", "\256\256\260\260\260\263\263\263\265\265\265\266\266\266", "\270\270\270\271\271\271\272\272\272\274\274\274\275\275", "\275\276\276\276\303\303\303\304\304\304\305\305\305\306", "\306\306\307\307\307\310\310\310\313\313\313\314\314\314", "\315\315\315\316\316\316\320\320\320\321\321\321\324\324", "\324\326\326\326\327\327\327\331\331\331\334\334\334\337", "\337\337\340\340\340\341\341\341\342\342\342\343\343\343", "\344\344\344\351\351\351\354\354\354\355\355\355\356\356", "\356\357\357\357\361\361\361\362\362\362\363\363\363\364", "\364\364\365\365\365\366\366\366\367\367\367\370\370\370", "\371\371\371\373\373\373\374\374\374\375\375\375\376\376", "\376\377\377\377\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\0\0\0\0\0\0\0\0\0", "\0\0\0\0\0\41\371\4\0\0\0\0\0\54", "\0\0\0\0\72\0\25\0\0\7\376\200\0\202", "\203\204\205\206\207\210\211\212\213\214\215\216\217\220", "\221\222\223\224\225\226\212\71\65\227\233\211\231\212", "\35\120\6\234\222\236\211\240\2\221\76\60\51\106", "\104\40\203\2\47\104\120\71\24\206\6\56\104\106", "\47\245\13\60\104\114\65\7\0\22\120\306\100\202", "\11\63\100\114\66\303\207\76\114\47\22\56\120\22", "\202\65\114\42\31\65\326\205\70\114\40\26\71\120", "\232\0\31\63\22\26\106\63\0\6\44\120\17\13", "\202\33\351\24\106\346\206\76\71\202\2\120\44\0", "\32\100\311\60\250\7\277\101\17\6\12\72\20\314", "\120\12\144\0\100\211\62\344\2\242\76\30\203\166", "\1\260\0\145\236\40\126\204\62\100\111\60\250\124", "\7\40\306\240\20\21\164\152\320\311\224\53\241\141", "\24\244\121\40\101\0\2\14\22\112\170\315\35\76", "\0\13\240\210\100\5\22\300\206\120\202\202\16\5", "\120\364\142\306\24\330\264\131\350\146\213\20\16\42", "\31\44\330\50\7\340\0\224\24\13\50\20\211\231", "\220\4\210\4\136\123\64\270\147\4\221\252\214\61", "\47\372\245\230\325\243\52\41\3\60\214\30\111\121", "\52\3\63\34\56\332\12\42\141\4\10\204\163\314", "\154\344\32\165\10\25\343\307\220\43\113\236\114\271", "\262\345\313\230\55\5\2\0\73", ; }; exit; }; # else print "Content-Type: image/gif\n\n"; -f $trackfilenameon and do { # pixel gruen print "GIF87a\001\0\001\0\200\0\0\0\377\0\0\0\0,\0\0\0\0\001\0\001\0\0\002\002D\001\0;"; exit; } or -f $trackfilenameoff and do { # pixel rot print "GIF87a\001\0\001\0\200\0\0\377\0\0\0\0\0,\0\0\0\0\001\0\001\0\0\002\002D\001\0;"; exit; }; # else # pixel grau print "GIF87a\001\0\001\0\200\0\0\177\177\177\0\0\0,\0\0\0\0\001\0\001\0\0\002\002D\001\0;";
dieser cronjob wirft /var/run/sopen/stat
, /var/run/sopen/track-on
und /var/run/sopen/track-off
weg, wenn /var/run/sopen/stat
aelter als 3 minuten ist:
/etc/cron.d/sopen
:
- /etc/cron.d/sopen
* * * * * www-data /usr/bin/perl -e '$s = "/var/run/sopen/stat"; [stat $s]->[9] + 180 < time and do { unlink $s, "/var/run/sopen/track-on", "/var/run/sopen/track-off" };'
run-verzeichnis angelegt:
mkdir /var/run/sopen chown www-data:www-data /var/run/sopen
der webserver muss /sopen
als cgi starten. ausserdem muss sopen
als sopen.gif
aufrufbar sein,
da dokuwiki sonst anstelle des bildes einen link darstellt.
/etc/apache2/sites-available/20-shackspace.de.conf
ScriptAlias /sopen /var/www/shackspace.de/htdocs/sopen ScriptAlias /sopen.gif /var/www/shackspace.de/htdocs/sopen
apache
ueber neue config informieren:
/etc/init.d/apache2 reload
im wiki wird es so eingebunden:
{{http://shackspace.de/sopen.gif?20x10}}
oder so:
{{http://shackspace.de/sopen?.gif?20x10}}
und so sieht's dann aus:
anwendung und ausgabeformate
format | desc. | url | sample |
---|---|---|---|
gif | pixel | http://shackspace.de/sopen/img/gif | |
png | pixel | http://shackspace.de/sopen/img/png | |
gif | grafische anzeige mit eingerendertem text | http://shackspace.de/sopen/imgtxt/en/gif | |
png | grafische anzeige mit eingerendertem text | http://shackspace.de/sopen/imgtxt/en/png | |
ascii[0] | deutscher text | http://shackspace.de/sopen/text/de | offen |
ascii[0] | english text | http://shackspace.de/sopen/text/en | closed |
[0] als content-type wird „text/plain“ uebermittelt
Kabelweg
Die Leitung verlaeuft von der Zuhaltung der Eingangstuere am Tuerrahmen nach oben, ueber den Tuerrahmen, an die Decke, durch die Wand zum Lager, im Lager links, oben unter der Decke ans Fenster, dort nach unten bis auf Hoehe des Fensterbankkanals. Hier ist ein db9-Stecker[0] angeloetet, in den ein hellblaues Cisco-Console-Kabel eingesteckt ist, dessen RJ45-Stecker in eine Netzwerkdose im Fensterbankkanal eingesteckt ist. Im RZ kommt die Leitung auf einem Patchfeld wieder raus, es folgen ein LAN-Kabel, ein weiteres hellblaues Cisco-Console-Kabel sowie ein USB/RS232-Adapter, der schliesslich im linken Rack in den KVM-Host (Name?): eingesteckt ist.
Pix:
Debugging
[0] Der db9-Stecker waere der erste Pruefpunkt fuer die Messung des Tuerkontakts:
Abziehen, Widerstand an den Leitungen messen
- Tuer auf: Widerstand unendlich
- Tuer zu: Widerstand nahe null
Fuer die Funktion Richtung Server:
- db9-Stecker wieder einstecken und (nur!) die beiden Leitungen kurzschliessen. Der Tuerstatus sollte innerhalb 3 Minuten reagieren.
Auf dem KVM-Host (Name?):
- ist der USB/RS232-Adapter ueberhaupt am USB sichtbar?
- ist das USB-Device dort vorhanden?
- ist das USB-Device der KVM zugaenglich gemacht?
Auf der KVM-Instanz „tuerstatus“:
- ist
/dev/ttyUSB0
vorhanden - hat
/dev/ttyUSB0
die Device-IDs „188, 0“ - ist
/dev/ttyUSB0
world-readable while sleep 1 ; do /usr/local/app/shackdoor/shackdoor /dev/ttyUSB0; done
liefert bei geschlossener Tuere „DSR“, bei offener Tuere Leerzeilen- ausgelesene Werte der letzten 2 Wochen:
( zcat syslog.{14,13,12,11,10,9,8,7,6,5,4,3,2}.gz ; cat syslog{.1,} ) | awk '-F[ =]+' '$6 == "FLAG" { print $1,$2,$3,$7 }' | less
(idee und drehbuch: chris)
abgeleitete anwendungen
shackspace Opening Statistics
apache --------------------------------------+ | | | writes log | V | /var/www/shackspace.de/logs/access.log* | | | | | /etc/cron.d/sopen | is read by | | | | | calls 1/10m | | V V | calls /usr/local/bin/sopen-rrd ----+ | | | | | writes data | calls | | | | +--> rrdtool <------+ | | | | writes rrd | V | /var/cache/sopen/sopen.rrd | | | +-------------+ | | | | /var/www/shackspace.de/htdocs/sopen-show-42h.png <--+ | | | | +--------+ | | | | | | /var/www/shackspace.de/htdocs/sopen-show-23d.png <--+ | | | | | +--------+ | | | | | | /var/www/shackspace.de/htdocs/sopen-show-42w.png <--+ | | | | +--------+ | | calls | V | /var/www/shackspace.de/htdocs/sopen-show | | | is read by | calls | V +--------> rrdtool | | writes image data V png / stdout / cgi
Weekly, hourly and half-hourly breakdowns
apache | | writes log V /var/www/shackspace.de/logs/access.log* | | /etc/cron.d/sopen | is read by | | | calls 1/4h | V V /usr/local/bin/sopen-list ---+ | | | writes data | calls V | /var/cache/sopen/sopen.dat | | | | is read by | | | +--> gnuplot <------+ | | writes images V /var/www/shackspace.de/htdocs/sopen.d/bywday.png /var/www/shackspace.de/htdocs/sopen.d/byhour.png /var/www/shackspace.de/htdocs/sopen.d/byhalfhour.png
shackspace offen? (1x1 pixel abfragen)
Autor: Felix
Ein Kommandozeilen-Skript welches den Status des Shacks ( open/close ) anzeigt. Gut auch um persönliche Statistiken zu bauen. Dependencies sind bash, wget, awk, md5sum ( alles core-utils ). Ziel war es die Dependencies und den Footprint so gering wie möglich zu halten.
- shack_offen.sh
#!/bin/bash # author: makefu # fetch the status from the website CURR=` wget "http://shackspace.de/sopen.gif" -O - 2>/dev/null | md5sum | awk '{print $1}'` # reference md5sum of a green 1x1 pixel REF="d8ced8c4a470ea0098a2b0d7b19c06b2" [ "$CURR" == "$REF" ] && echo -e "shack is \e[1;32mOPEN\e[0m" && exit 0 [ "$CURR" != "$REF" ] && echo -e "shack is \e[1;31mCLOSED\e[0m" && exit 1
Shackspace offen? (String abfragen)
Autor: chf
is_shack_open.sh:
- is_shack_open.sh
#!/bin/bash status=$( echo -e -n "GET /sopen/text/en HTTP/1.0\r\nHost: shackspace.de\r\n\r\n" | nc shackspace.de 80 2>&1 | egrep -c "^open$" ) if [ $status -eq 1 ]; then echo -e "Yuhuu, doors are \e[1;32mopen\e[0m! ;-)" else echo -e "Damn, it's \e[1;31mclosed\e[0m." exit 1 fi
und um dann gleich informiert zu werden:
while [ 1 ]; do is_shack_open.sh && zenity --info --text="Shack is open" ; sleep 1m; done
!offen im IRC
Autor: cmile
Im IRC einfach !open schreiben und auf die Antwort von Krebsbeam warten.
- open.py
# jsb/plugs/common/c-lang.py # -*- coding: utf-8 -*- ## jsb imports from jsb.utils.exception import handle_exception from jsb.lib.commands import cmnds from jsb.lib.examples import examples ## basic imports import re, urllib import random ## defines ## isopen command def handle_isopen(bot, ievent): mysock = urllib.urlopen("http://shackspace.de/sopen/text/de") ievent.reply(mysock.read()) cmnds.add('isopen', handle_isopen, ['OPER', 'USER', 'GUEST']) cmnds.add('offen', handle_isopen, ['OPER', 'USER', 'GUEST'])