Categories
english Hacking Hardware Gadgets

RFID Door Opener…

recently i finished my latest project: a RFID Door Opener.

besides the fun while hacking it, it improves the security of the door. since it’s not an outside door but the door to the garage, it’s construction does not allow to mount a more secure lock. so locking it from the outside when leaving is a bit bothersome.
the new auto-lock feature, which allows the door to auto-lock itself, saves some time and works around the lazy user, who wouldn’t have locked it.
additionally it is now easy to add or remove the right of access by adding or deleting the RFID’s unique number from the system. revoking someone’s right of access is far more difficult with a normal key.

beside an arduino i’m using an easydriver to drive the stepper and a RFID reader from seeedstudio.

below you can see the schematics of the setup. the ends marked with Ard X go to a port of the arduino, which is the brain of the lock:

Schematics for the build
Schematics for the build

edit: since i’ve been asked what transistors i used for this build i’ll tell you:
as PNP transistor i’m using a ‘mje2955T‘, as NPN i’m using a ‘bc547‘.
it’s important to have the two different transistor types NPN and PNP.
NPN is the one closer to ‘Ard6’ and the PNP is the one switching the 12V power supply for the easydriver. the most important part is that the second transistor, switching the easy driver, will only work if you’re using a PNP transistor. these transistors switch when they get pulled to ground.

Code can be found on github and here:


/**
 * door lock application (c) 2011 Florian Klien
 * some code parts are borrowed from different authors ;) thx
 */

#include <NewSoftSerial.h>

#define rxPin 2
#define txPin 3

// door defs

#define DOOR_SENS  3 // analog
#define DRIVER_SWITCH 6
#define DOOR_SW 2 // analog

// motor defs
#define DIR_PIN 7
#define STEP_PIN 8
#define ledIN 5
#define ledOUT 11

NewSoftSerial rfid = NewSoftSerial( rxPin, txPin );

// The tag database consists of two parts. The first part is an array of
// tag values with each tag taking up 5 bytes. The second is a list of
// names with one name for each tag (ie: group of 5 bytes).
char* allowedTags[] = {
  "AABBCCDDEE",         // Tag 1
  "AABBCCDDEE",         // Tag 2
};

// List of names to associate with the matching tag IDs
char* tagName[] = {
  "User1",         // Tag 1
  "User2",         // Tag 2
};

// software version number:
char* software_version = "1.1";

// Check the number of tags defined
int numberOfTags = sizeof(allowedTags)/sizeof(allowedTags[0]);

int incomingByte = 0;    // To store incoming serial data

boolean locked = true;
int door_open = 0; // pseudo digital
boolean prev_status = false;
boolean auto_lock = true;
//unsigned long auto_lock_time = 0;
int auto_lock_delay = 5; // in seconds
int auto_lock_switch_time = 2; // in seconds
int status_led = 0;
unsigned long status_led_time = millis();
boolean status_led_on = false;

unsigned long status_breathe_time = millis();
int breathe_delay = 10;
boolean breathe_up = true;
int breathe_i = 15;

unsigned long last_successful_rfid_read = 0;
int rfid_success_timeout = 5000; // millis

float lock_speed = 1;

/**
 * Setup
 */
unsigned long time_door = millis();
unsigned long time_switch = millis();
long debounce = 500;

void setup() {
  pinMode(ledIN, OUTPUT);
  pinMode(ledOUT, OUTPUT);
  digitalWrite(ledIN, HIGH);
  digitalWrite(ledOUT, HIGH);
  delay(300);
  digitalWrite(ledIN, LOW);
  digitalWrite(ledOUT, LOW);

  pinMode(DRIVER_SWITCH, OUTPUT);
  digitalWrite(DRIVER_SWITCH, LOW);
  pinMode(DIR_PIN, OUTPUT);
  digitalWrite(DIR_PIN, LOW);
  pinMode(STEP_PIN, OUTPUT);
  digitalWrite(STEP_PIN, LOW);
  pinMode(DOOR_SENS,INPUT);
  pinMode(DOOR_SW,INPUT);
  
  Serial.begin(9600);   // Serial port for connection to host
  rfid.begin(9600);      // Serial port for connection to RFID module

  Serial.println("RFID reader starting up");
  delay(1000);
  Serial.println("done");
  Serial.print("Software Version no: ");
  Serial.println(software_version);
  Serial.print("door locked: ");
  Serial.print(locked, DEC);
  Serial.print("\n");
  Serial.print("door closed: ");
  Serial.print(!door_open, DEC);
  Serial.print("\n");
}

/**
 * Loop
 * non-blocking version of each function!
 */
void loop() {
  readRFID();
  doorSensor();
  doorSwitch();
  statusLed();
}

void doorSensor(){
  // pseudo digital
  door_open = analogRead(DOOR_SENS);
  if(millis() - time_door > debounce){
    if (door_open <= 500 && prev_status == false){
      Serial.println("Door: opened");
      prev_status = true;
      locked = false;
    }else  if(door_open > 500 && prev_status == true){
      Serial.println("Door: closed");
      prev_status = false;
      if(auto_lock){
        Serial.println("locking door automatically...");
        delay(auto_lock_delay*1000); 
        lock();
        locked = true;
      }
    }
    time_door = millis();  
  }
}

void doorSwitch(){
  int dstimer = 0;
  int door_switch = analogRead(DOOR_SW); // pseudo digital
  if(millis() - time_switch > debounce && door_switch >= 300){
    while (analogRead(DOOR_SW) >= 300) {
      delay(100);
      dstimer++;
    }
    Serial.println(door_switch,DEC);
    Serial.println(dstimer,DEC);
    if (dstimer < auto_lock_switch_time*10) { //button has been pressed less than 2 seconds = 1000/100
        if (locked == false){
          Serial.println("door locked");
          locked = true;
          lock();
        }else if(locked == true){
          Serial.println("door unlocked");
          locked = false;
          unlock();
      }
    }else {
      // auto_unlock off/on
      if(auto_lock == true){
        Serial.println("auto_lock off");
        auto_lock = false;
      }else{
        Serial.println("auto_lock on");
        auto_lock = true;
      }
      analogWrite(ledIN, 0); // resetting output
    
    }
    time_switch = millis();  
  }
}

// breathing status led on the inside
void statusBreathe(){
  if( (status_breathe_time + breathe_delay) < millis() ){
    analogWrite(ledIN, breathe_i/1.5);
    status_breathe_time = millis();
    if (breathe_up == true){
      if (breathe_i > 150) {
        breathe_delay = 4;
      }
      if ((breathe_i > 125) && (breathe_i < 151)) {
        breathe_delay = 5;
      }
      if (( breathe_i > 100) && (breathe_i < 126)) {
        breathe_delay = 7;
      }
      if (( breathe_i > 75) && (breathe_i < 101)) {
        breathe_delay = 10;
      }
      if (( breathe_i > 50) && (breathe_i < 76)) {
        breathe_delay = 14;
      }
      if (( breathe_i > 25) && (breathe_i < 51)) {
        breathe_delay = 18;
      }
      if (( breathe_i > 1) && (breathe_i < 26)) {
        breathe_delay = 19;
      }
      breathe_i += 1;
      if( breathe_i >= 255 ){
        breathe_up = false;
      }
    }else{
      if (breathe_i > 150) {
        breathe_delay = 4;
      }
      if ((breathe_i > 125) && (breathe_i < 151)) {
        breathe_delay = 5;
      }
      if (( breathe_i > 100) && (breathe_i < 126)) {
        breathe_delay = 7;
      }
      if (( breathe_i > 75) && (breathe_i < 101)) {
        breathe_delay = 10;
      }
      if (( breathe_i > 50) && (breathe_i < 76)) {
        breathe_delay = 14;
      }
      if (( breathe_i > 25) && (breathe_i < 51)) {
        breathe_delay = 18;
      }
      if (( breathe_i > 1) && (breathe_i < 26)) {
        breathe_delay = 19;
      }
      breathe_i -= 1;
      if( breathe_i <= 15 ){
        breathe_up = true;
        breathe_delay = 970/2;
      }
    }
  }
}

void statusLed(){
  if(auto_lock == false){
    status_led = 150;
  }else{
    // set this to > 0 if you want the status led to blink in default mode
    status_led = 0; 
    if(status_led == 0){
      statusBreathe();
    }
  }
  if(millis() - status_led_time >= status_led && status_led != 0){
    status_led_on = !status_led_on;
    digitalWrite(ledIN,status_led_on);
    status_led_time = millis();
  }
}

void readRFID(){
  byte i         = 0;
  byte val       = 0;
  byte checksum  = 0;
  byte bytesRead = 0;
  byte tempByte  = 0;
  byte tagBytes[6];    // "Unique" tags are only 5 bytes but we need an extra byte for the checksum
  char tagValue[10];

  if(rfid.available()>0){
    if((val = rfid.read()) == 2) {        // Check for header
    bytesRead = 0;
    while (bytesRead < 12) {            // Read 10 digit code + 2 digit checksum
      val = rfid.read();
      Serial.print(val,BYTE);
      // Append the first 10 bytes (0 to 9) to the raw tag value
      if (bytesRead < 10)
      {
        tagValue[bytesRead] = val;
      }

      // Check if this is a header or stop byte before the 10 digit reading is complete
      if((val == 0x0D)||(val == 0x0A)||(val == 0x03)||(val == 0x02)) {
        break;                          // Stop reading
      }

      // Ascii/Hex conversion:
      if ((val >= '0') && (val <= '9')) {
        val = val - '0';
      }
      else if ((val >= 'A') && (val <= 'F')) {
        val = 10 + val - 'A';
      }

      // Every two hex-digits, add a byte to the code:
      if (bytesRead & 1 == 1) {
        // Make space for this hex-digit by shifting the previous digit 4 bits to the left
        tagBytes[bytesRead >> 1] = (val | (tempByte << 4));

        if (bytesRead >> 1 == 5) {                // If we're at the checksum byte,
          checksum ^= tagBytes[bytesRead >> 1];   // Calculate the checksum... (XOR)
        };
      } else {
        tempByte = val;                           // Store the first hex digit first
      };
  
      bytesRead++;                                // Ready to read next digit
    }
  

    // Send the result to the host connected via USB
    if (bytesRead == 12) {                        // 12 digit read is complete
      tagValue[10] = '\0';                        // Null-terminate the string

      Serial.print("Tag read: ");
      for (i=0; i<5; i++) {
        // Add a leading 0 to pad out values below 16
        if (tagBytes[i] < 16) {
          Serial.print("0");
        }
        Serial.print(tagBytes[i], HEX);
      }
      Serial.println();

      Serial.print("Checksum: ");
      Serial.print(tagBytes[5], HEX);
      Serial.println(tagBytes[5] == checksum ? " -- passed." : " -- error.");

      // Show the raw tag value
      //Serial.print("VALUE: ");
      //Serial.println(tagValue);
      Serial.print("door_open: ");
      Serial.println(door_open,DEC);
      // Search the tag database for this particular tag
      int tagId = findTag( tagValue );

      // Only fire the strike plate if this tag was found in the database
      if( tagId > 0 )
      {
        Serial.print("Authorized tag ID ");
        Serial.print(tagId);
        if(door_open > 500 && (last_successful_rfid_read + rfid_success_timeout) < millis() ){
          Serial.print(": unlocking for ");
          Serial.println(tagName[tagId - 1]);   // Get the name for this tag from the database
          unlock();
          last_successful_rfid_read = millis();
          delay(2000);
        }
      } else {
        Serial.println("Tag not authorized");
        //failSound();
        for (int i=0;i<7;i++){ // FIXXME nonblocking version?
          digitalWrite(ledOUT, HIGH);
          digitalWrite(ledIN, HIGH);
          delay(100);
          digitalWrite(ledOUT, LOW);
          digitalWrite(ledIN, LOW);
          delay(80);
        }
      }
      Serial.println();     // Blank separator line in output
    }

    bytesRead = 0;
  }
  }   
}

/**
 * Fire the relay to activate the strike plate for the configured
 * number of seconds.
 */
void unlock() {
  digitalWrite(ledOUT, HIGH);
  digitalWrite(ledIN, HIGH);
  delay(100);
  // if your stepper is powerful enough you can use full speed
  rotateDeg(-800, 0.6);
  digitalWrite(ledIN, LOW);  
  digitalWrite(ledOUT, LOW);
  locked = false;
}

void lock(){
  digitalWrite(ledOUT, HIGH);
  digitalWrite(ledIN, HIGH);
  delay(100);
  rotateDeg(800, 1);
  digitalWrite(ledIN, LOW);  
  digitalWrite(ledOUT, LOW);
  locked = true;
}

void rotate(int steps, float speed){
  
  // power driver
  digitalWrite(DRIVER_SWITCH,HIGH);
  delay(200);
  //rotate a specific number of microsteps (8 microsteps per step) - (negitive for reverse movement)
  //speed is any number from .01 -> 1 with 1 being fastest - Slower is stronger
  int dir = (steps > 0)? HIGH:LOW;
  steps = abs(steps);

  digitalWrite(DIR_PIN,dir); 

  float usDelay = (1/speed) * 250;

  for(int i=0; i < steps; i++){
    digitalWrite(STEP_PIN, HIGH);
    delayMicroseconds(usDelay); 

    digitalWrite(STEP_PIN, LOW);
    delayMicroseconds(usDelay);
  }

  // unpower driver
  delay(200);
  digitalWrite(DRIVER_SWITCH,LOW);
} 

void rotateDeg(float deg, float speed){
  // power driver
  digitalWrite(DRIVER_SWITCH,HIGH);
  delay(200);
  //rotate a specific number of degrees (negative for reverse movement)
  //speed is any number from .01 -> 1 with 1 being fastest - Slower is stronger
  int dir = (deg > 0)? HIGH:LOW;
  digitalWrite(DIR_PIN,dir); 

  int steps = abs(deg)*(1/0.225);
  float usDelay = (1/speed) * 250;

  for(int i=0; i < steps; i++){
    digitalWrite(STEP_PIN, HIGH);
    delayMicroseconds(usDelay); 

    digitalWrite(STEP_PIN, LOW);
    delayMicroseconds(usDelay);
  }
  // unpower driver
  delay(200);
  digitalWrite(DRIVER_SWITCH,LOW);
}

/**
 * Search for a specific tag in the database
 */
int findTag( char tagValue[10] ) {
  for (int thisCard = 0; thisCard < numberOfTags; thisCard++) {
    // Check if the tag value matches this row in the tag database
    if(strcmp(tagValue, allowedTags[thisCard]) == 0)
    {
      // The row in the database starts at 0, so add 1 to the result so
      // that the card ID starts from 1 instead (0 represents "no match")
      return(thisCard + 1);
    }
  }
  // If we don't find the tag return a tag ID of 0 to show there was no match
  return(0);
}
Categories
english Hacking Hardware Gadgets

Non-blocking breathing led for arduino…

i needed a version of the ‘breathing led’ for a project which would not block everything else. button presses and the actual application should run without delay. after all it would just be cosmetics 😉

the result looks like this:

here is my non-blocking code for a breathing led with arduino:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
[sourcecode lang="c"]
/*
&quot;Breathing non-blocking sleep LED.&quot;
Florian Klien 2011
blog.flo.cx
based on work from Jeremy Saglimbeni (thecustomgeek.com)
*/
#define LED 5 // any PWM led will do
unsigned long status_breathe_time = millis();
int breathe_delay = 10;
boolean breathe_up = true;
int breathe_i = 15;

void setup() { // bring the LED up nicely from being off
  for(i = 0 ; i &lt;= 15; i+=1)
  {
    analogWrite(11, i);
    delay(5);
  }
}

void loop()
{
  nonBlockingBreathe();
  otherImportantNonBlockingStuff();
}

void nonBlockingBreathe(){
  if( (status_breathe_time + breathe_delay) &lt; millis() ){
    analogWrite(LED, breathe_i);
    status_breathe_time = millis();
    if (breathe_up == true){
      if (breathe_i &gt; 150) {
        breathe_delay = 4;
      }
      if ((breathe_i &gt; 125) &amp;&amp; (breathe_i &lt; 151)) {
        breathe_delay = 5;
      }
      if (( breathe_i &gt; 100) &amp;&amp; (breathe_i &lt; 126)) {
        breathe_delay = 7;
      }
      if (( breathe_i &gt; 75) &amp;&amp; (breathe_i &lt; 101)) {
        breathe_delay = 10;
      }
      if (( breathe_i &gt; 50) &amp;&amp; (breathe_i &lt; 76)) {
        breathe_delay = 14;
      }
      if (( breathe_i &gt; 25) &amp;&amp; (breathe_i &lt; 51)) {
        breathe_delay = 18;
      }
      if (( breathe_i &gt; 1) &amp;&amp; (breathe_i &lt; 26)) {
        breathe_delay = 19;
      }
      breathe_i += 1;
      if( breathe_i &gt;= 255 ){
        breathe_up = false;
      }
    }else{
      if (breathe_i &gt; 150) {
        breathe_delay = 4;
      }
      if ((breathe_i &gt; 125) &amp;&amp; (breathe_i &lt; 151)) {
        breathe_delay = 5;
      }
      if (( breathe_i &gt; 100) &amp;&amp; (breathe_i &lt; 126)) {
        breathe_delay = 7;
      }
      if (( breathe_i &gt; 75) &amp;&amp; (breathe_i &lt; 101)) {
        breathe_delay = 10;
      }
      if (( breathe_i &gt; 50) &amp;&amp; (breathe_i &lt; 76)) {
        breathe_delay = 14;
      }
      if (( breathe_i &gt; 25) &amp;&amp; (breathe_i &lt; 51)) {
        breathe_delay = 18;
      }
      if (( breathe_i &gt; 1) &amp;&amp; (breathe_i &lt; 26)) {
        breathe_delay = 19;
      }
      breathe_i -= 1;
      if( breathe_i &lt;= 15 ){
        breathe_up = true;
        breathe_delay = 970;
      }
    }
  }
}
[/sourcecode]

original code came from thecustomgeek.com

Categories
english OnlineLife Net Webapps

how to catch 336 twitter bots in 12 hours…

yesterday our API (the API of qr.cx) returned rubbish for about 12 hours. i apologize for that, this will not happen again. we are working on a reimplementation which should be far more reliable.

however the thing had an upside. we were able to expose twitter bots who published this rubbish without checking. in total we found 336 twitter bots who did so. they included

1
<br /><b>Notice</b>: Undefined variable: [...] in <b>/[...]/qr.cx/htdocs/api/index.php</b>[...]"

in their tweets. a human being would not do that. firstly the API is made for automated use, so why would one use that on a regular basis; secondly the error is apparent to a human user. one would not publish a tweet with the full nonsense. the bots did.

so now we can search twitter for this perfidious string and see which account is a bot. this is good, this could help twitter™ to identify malicious users/bots and protect their normal human users.

but it also helps us, the urlshortener, to safeguard the system. we can identify spam links. we can search the twitter bot’s stream for links it has shortened before. those links are most likely links to spam or fraudulent pages. disabling those would be no harm.

i’m looking forward to implementing these security features. it will definitely require a little more thinking to setup a nice safe system.

Categories
english Motoriges

Airpower 11…

this saturday we went to see the “Airpower 11“. apparently the biggest air show in europe, and it really was huge.

it started on friday, with about 140.000 people. on saturday there were about 160.000. let’s just say: it was really crowded!

i did take some photos:
Royal Saudi HawksLockheed P-38L LightningChance Vought F4U-4 CorsairNorth American B-25J MitchellLockheed C-130 HerculesEurofighter TyphoonPatrouille SuisseMiG -29AFrecce Tricolori
the full selection can be found in my picasa album.

Categories
data english

Big Bang Theory Relationship Diagram…

not 10 days ago meisterluk posted a graph of relationship data he collected watching The Big Bang Theory .

i was not quite happy with his visualization. so i redid it, here it is:

Big Bang Theory Relationship Diagram
Big Bang Theory Relationship Diagram

i hope it is self explanatory. in case it is not,

Categories
OnlineLife Net Webapps

UPC schläft in der witzkiste…

es muss leider wirklich von schlafen die rede sein. so wenig wie sich da tut, die können nur schlafen.

wir haben seit langem einen internetanschluss bei inode, einer UPC tochter. der grund zum wechsel von der telekom war die möglichkeit 2 telefonleitungen (telefon, fax) auf einen anschluss zu nehmen und mittels durchwahl auch das faxgerät getrennt vom telefon zu erreichen.
das paket das wir genommen haben hatte satte 20 mbit down und etwa 2 mbit upstream. nur dass die leitung von anfang an diese leistung nicht bringen konnte. angeblich liegt zwischen ein und zwei (die angaben sind widersprüchlich) kilometer kupfer zwischen uns und dem nächsten schaltkasten oder dem wählamt an dem das verstärkt wird. wie dem auch sei; seit anfang an haben wir nur 6 der 20 (30%) der leistungsbandbreite. es wurde uns zwar vorgeschlagen auf ein günstigers paket umzusteigen (kleinere bandbreite), da wären aber nicht die gleichen telefoniepreise ins festnetz vorhanden gewesen. naja. das war vor knapp fünf jahren, im oktober/november 2006.

seither lief unsere leitung ganz passabel. vor einem jahr fing es an dass das netz mitunter seeeeeehr langsam wurde und ein neustart des modems das problem wieder kurzzeitig gelöst hat. vor 3 monaten war es dann soweit dass das alle 48 stunden immer schlimmer wurde. es waren die paketverluste auf bis zu 33% angewachsen. ein anruf bei der hotline hat das bestätigt und wir haben das modem getauscht. bis dahin alles ok.

ich habe das modem selbst im UPC shop in graz abgeholt, da der postweg etwa 5 tage länger gedauert hätte, dabei liess ich mir auch gleich die zugangsdaten zur konfiguration nochmal ausdrucken. nach dem aufbau des neuen modems fand ich nun aber keine logindaten für unsere zweite telefonnummer – die faxnummer. die gabs nicht mehr. ein anruf bei der hotline hat das bestätigt. die nummer sieht er nicht – und er kann sie auch nicht mehr hinzufügen. na ganz toll. alles was wir machen könnten wäre eine email an die UPC zentrale in wien zu schreiben und uns dort zu beschweren. das haben wir noch nicht getan.
das ist meiner meinung nach eine einseitige vertragsänderung! das kann sich die UPC nur erlauben weil sie die stärkeren sind und man von ihnen abhängig ist. das darf es nicht geben, sei es durch schlecht geschultes personal, die einfach anschlüsse löschen, durch schlamperei oder einfach durch vorsatz (den ich hier aber nicht unterstellen will).

zu diesem ärger kommt jetzt noch eine email die uns heute erreicht hat:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
Sehr geehrter Herr X!  

In den letzten Jahren hat UPC seine Netzinfrastruktur laufend ausgebaut, um schnellstes Internet durch den Einsatz modernster Hard- und Software anbieten zu können. UPC stellt sich als einer der führenden Anbieter von Telekommunikationslösungen in Österreich dem Anspruch, seinen Kunden auch in Zukunft eine versorgungssichere Dienstleistung in höchster Qualität zur Verfügung zu stellen. Als Folge ersuchen wir um Verständnis, dass nun ein Internet-Service-Entgelt in Höhe von EUR 15,- pro Jahr und Vertrag mit einem Internet-Produkt eingeführt wird. Die Verrechnung erfolgt im Voraus - erstmalig ab August 2011 - und wird danach jährlich jeweils im August eingehoben. In diesem Zusammenhang freuen wir uns Ihnen abermals weitere, neue Service-Verbesserungen mitteilen zu können, welche in Kürze verfügbar sind:

-- Steigerung Ihres E-Mail-Speicherplatzes von bisher 200 MB auf 500 MB pro E-Mail-Adresse
-- Erhöhung Ihres Webspace von bisher 10 bzw. 30 MB auf 250 MB
-- Im Herbst: erneute Steigerung Ihres E-Mail-Speicherplatzes auf riesige 5.000 MB

Für allfällige Rückfragen stehen wir Ihnen gerne online unter to http://irs.cmt.upc.biz/emessageirs/servlet/XXXXXX zur Verfügung.
Bitte beachten Sie Ihr Recht zur kostenlosen Vertragsbeendigung, das bis zum Inkrafttreten der Änderung (31.07.2011) schriftlich ausgeübt werden kann.
Wir wünschen Ihnen weiterhin gute Unterhaltung mit Ihren UPC-Diensten. Mit freundlichen Grüßen

Ihr UPC-Team

Bitte antworten Sie nicht auf dieses Mail. Falls Sie Fragen zu UPC oder unseren Produkten haben, finden Sie mehr unter www.upc.at.
Impressum
Dieses Mailing ist ein Service der UPC Telekabel Wien GmbH im Auftrag aller UPC Gesellschaften in Österreich.
Wolfganggasse 58-60, 1120 Wien. Firmenbuchnummer: 84116a, Handelsgericht Wien. Unternehmensgegenstand: TV-, Digital TV-, Breitband Internet- und Telefon-Dienstleistungen, in den Gebieten Wien, Graz, Klagenfurt, Wiener Neustadt, Region Baden sowie im Raum Purkersdorf und Reichenau an der Rax.

dieses mail teilt einem mit dass sie der meinung sind zu wenig geld für ihr tolles service zu bekommen. dem möchte ich widersprechen!
in erster linie zahlen wir für ein internetpaket bei UPC weil wir einen anschluss haben wollen. der umstand dass ein telefon mitkommt ist nett, aber teilweise auch unpraktisch, weil es halt ‘nur’ VOIP ist und über die gleiche leitung geschliffen wird. dieser anschluss hat seit 5 jahren (!) die unveränderte bandbreite von 30% des eingekauften wertes! was soll das?
und dann kommen die her und schreiben einem wie toll sie ihre infrastruktur aufbauen und erweitern? entschuldigung aber neue platten im rechenzentrum in den server zu stecken und die mailquotas raufzuschrauben ist standard. das ist keine infrastrukturverbesserung!
infrastrukturverbesserung ist es neue leitungen zu legen und/oder die bandbreiten zu erhöhen. wieso gibt es in graz nur vereinzelt fiber to the home? und das nicht mal von UPC?

des weiteren interessiert mich der webspace nicht den ihr bei jedem anschluss mitverschenkt. behaltet den bitte und gebt mir den anschluss um ein paar euro weniger.
der zeitpunkt des einhebens der pauschale im august kann auch kein zufall sein. entweder hoffen sie auf geduldige gemüter, dank urlaubsgeld, oder auf ein simples übersehen des betrages, weil alle gerade auf urlaub sind.

wenn jemand einen guten kleinen provider kennt der in graz eine brauchbare bandbreite (≥8Mbit) zusammenbringt, lasst mich das bitter per kommentar wissen. ich bin reif für einen wechsel. allerdings fürchte ich dass es UPC da wieder mangels alternativen bleiben wird. 🙁

Categories
data english

Bday Pie Chart…

having had my birthday recently i was curious about the channels via which people would wish me a happy birthday. so i counted.
and here is the resulting pie chart – what’s better than a pie chart for a birthday wish statistic?

Birthday wishes pie chart
Birthday wishes pie chart

it is not surprising that facebook has the biggest piece of the pie. people seem to like facebook, i certainly have a bigger contact list than on skype plus facebook knows my birthday and people get reminded. that sums up to a high count.
suprising for me was the second place: the telephone. it beat skype by 1.4% to the second place. it has no auto reminder and it’s the only medium that is synchronous.
third place goes to skype (just text messages) with 11.9%. fourth to SMS.
the last place is shared by twitter and e-mail. twitter is not that much of a surprise to me. on the other hand e-mail suprised me. a few years ago this statistic would have looked totally different without facebook, and e-mail would not be in the last place.

what’s your birthday wishes statistic like?