Who did it better? Sometimes Good Guys Don’t Wear White

In 1966, the Standells released, “Sometimes  Good Guys Don’t Wear White“. The single made it to #43 on the Billboard Hot 100 chart. The album it came from, Dirty Water, was recorded in just two days and didn’t chart as well as the singles (it got to #52 on the Pop albums chart). The first song featured from it, the aptly named Dirty Water, got up to #11 on Billboard but not for long.

Here then is the original version of The Standells’ Sometimes Good Guys Don’t Wear White:

Flash forward two or three years and then this gem was recorded. Las Dilly Sisters were a sort of novelty act, two very young Mexican sisters, and recorded just one 45 single, that I can find at least, and this song was chosen as the B-Side. I really like this one.

Flying through time once again, we come to the end of the road for DC’s hardcore favorites, Minor Threat. Their final official recording while they were more-or-less still together was the nice little 45, Salad Days. This song was second on the b-side.

Well, there you go. Comment on which version you like best, eh. There are, of course, tons of other covers online you could check out too.

Cheers,

: )

 

Advertisement

Who Did it Better? Kick Out the Jams

Wayne Kramer and MC5 are known as a proto-punk band, since they performed in the late 1960’s / early 70’s and were young, angry and wrote fast loud songs for the most part.  According to one interview, “In the old days … they used to leap off the stage and blow a saxophone in people’s ears, or throw money at the audience – anything to get a reaction.” which sounds a lot like a lot of later punk and post-punk bands for sure.

This song was not exactly the angry anti-establishment tirade it seems though, according to Kramer:

People said “oh wow, ‘kick out the jams’ means break down restrictions” etc., and it made good copy, but when we wrote it we didn’t have that in mind. We first used the phrase when we were the house band at a ballroom in Detroit, and we played there every week with another band from the area. […] We got in the habit, being the sort of punks we are, of screaming at them to get off the stage, to kick out the jams, meaning stop jamming. We were saying it all the time and it became a sort of esoteric phrase. Now, I think people can get what they like out of it; that’s one of the good things about rock and roll.

So many covers of this song have been recorded or played live over the years, which makes sense considering its status and the respect later generations paid to MC5. Rockers of all sorts still regard this as an ideal: A perfect statement of rock and roll’s energy, frustration, whatever you want to call it.

I’ve chosen five covers here for us to compare today so let’s dig in.

Here’s the best video I could find with the original LP version  – 1969, MC5’s first (live) record:

Blue Oyster Cult liked the song so much they included it on their 1978 album, Some Enchanted Evening. This version is a bit more hard-rock, less punk. No MFers in the intro.

Here’s a cover by Bad Brains with Henry Rollins singing, from the ‘Pump Up The Volume’ 1990 movie soundtrack, since punkers should be able to do this better, right? I like that Henry brought the MFers in the intro back.

The Presidents of the United States of America included this somewhat modified  version of the song on their 1995 self-title debut LP. Nice and clean 90’s alternative rock sound here.

Rage Against the Machine released this version on their Renegades album in 2000. What can you say: They rage even though this is slower. Somehow, that makes it even heavier than the original. Love Tom Morello’s sounds in the solo with this one.

 

Well, this is a tough one. I love the song and love pretty much every cover I’ve heard.  If I have to choose, I suppose I’d go with the original, since, well, it was the first and defined the genre.

And as an added bonus, here’s some guy named Wayne teaching us how to play the song on guitar, for those interested:

Let us know what you think, which version you like best, in the comments.

🙂

 

Mary Margaret O’Hara – Miss America

This was, without a doubt, my favorite album of 1988.  Mary Margaret is the sister of Catherine O’Hara, a fantastic actress herself, but this is not one of those stories where someone famous got her sibling a record deal. Mary Margaret O’Hara has an amazing voice and real singing talent on her own. Plus these songs are jaw dropping amazing.

I’m sure you think you’ve heard a woman sing her heart out before but on this record, Ms. O’Hara is on another level. These timeless tunes send me to a very special happy place every time I hear them. We have poppy, bluesy, jazzy and spacey songs here, all completely perfect and none of them feel forced or fake in any way.

Can you tell I really like this album? That’s true and it makes the fact that she never made another full album that much harder to believe. She released a 4-song Christmas EP in 1991, and then contributed to the soundtrack for a 5-out-of-10 stars romantic comedy named ‘Apartment Hunting’ around 2000, and then… nothing.

51yr7zMlgoL._SS500

If this music is new to you, I am jealous of you. Enjoy hearing it for the first time!

If somehow Ms. O’Hara reads this, please make more music. We need to hear more of your beautiful voice.

Read about it – (4.5 stars out of 5)
https://www.allmusic.com/album/miss-america-mw0000205701

Buy it
https://www.discogs.com/Mary-Margaret-OHara-Miss-America/release/1631994

Or just listen to this YouTube playlist:

🙂

 

Who did it better? “How Is the Air Up There?”

Alright music fans, which version of this song do you like best?

The Changin’ Times  – How Is the Air Up There?  (1965)

 

The La De Das – How is the Air Up There?  (1966)

 

The Bangles – How is the Air Up there?  (1983)

 

Personally, I grew up in Southern California in the 80’s so the Bangles win for me. Then the original version from the Changin’ Times, then the La De da’s version.

: )

 

Grim Fandango and Homeworld – Remastered!

I am pretty old now, compared to most gamers. Two classic games from my ahem, second youth, are being re-issued as remastered versions so we can play them again with way better graphics. They’re probably both available for download somewhere on the net already but these releases will be both legal and completely awesome looking, compared to the originals. They were both always awesome games, amazing to play so adding better graphics should just make them that much better. Woot!

Grim-Fandango-Gameplay-1

Grim Fandango Remastered – http://www.gog.com/game/grim_fandango_remastered

Homeworld MohtershipHomeworld Remastered – http://store.steampowered.com/app/244160/

🙂

Python + Bottle + PythonAnywhere = easy online web app creation

What is it about weekends that they just really lend themselves to experimenting? I don’t know either. This weekend, I decided I’d spend some time with web development, so I assigned myself a task: create a simple web application which would have a single big text box at the top of the page which would allow the visitor to type whatever they want. Add a Submit button, allowing the user to save that text to a file on the server and then below that form, display all the text that has been entered so far. This is about as simple as a web form can get, I figured.

Capture

I wanted to use my slowly growing python skills for this little dynamic web page, so I looked around for a framework to use, since Python doesn’t create web page source natively. I looked at Tornado, Flask and a few others, but they all seemed like overkill for my very simple purposes. I  liked the way the Bottle home page made it sound very easy to create dynamic content so I decided to use it.

I knew my one page site / app was gong to need just one HTML page, and the only part of it that will be dynamic is the part which lists the previous entries, so I started by making a web page template. I used the very friendly, easy to learn Initializr to save time with this. Just visit the site, make a few choices on what type of page you’re looking for and then download the source files. You get an HTML 5 file, CSS and JavaScript all in one zip file. I spent about 30 more minutes removing the parts I didn’t want (including all of the JavaScript), adding the form, updating the colors in the CSS to liking and voila! Index.tpl (for a Bottle template) and supporting static files all set.

At this point, the site consists of these files:

index.tpl

css/main.css

css/normalize.min.css

Next, I created an empty file which would hold the text: ‘allthetext’. I created the app’s code file next too: ‘bottle_app.py’. It doesn’t matter what you name that file, for this little app, since it will be the only one.

With Bottle then, the first thing you do is map out the “routes” or URLs you want the app to know how to handle. My app is the only thing this web site will be for, so my default route is just “/”. In the code, I knew I would have to handle both a GET of the page, which would be called when you first visit that path, and also a POST, which will be called when you submit some text in the form.

The GET just needs to display the web page, so that was easy:

from bottle import Bottle, debug, get, request, route, run, static_file, template

app = Bottle()
@app.route('/', method='GET')
def index():
    return template('index')

 

The POST though has to get the value of the form field named ‘textarea’ and save that into the ‘allthetext’ file. Python makes this really simple but I have one complication. I want the newest entry to be at the top of the file, and then include the rest of the entries from there on down. Here’s the code that does this:

@app.route('/s', method='POST')
def submit_form():
    try:
        bigtext = request.forms.get('bigtext')
        if (bigtext):
            new_text = ''
            current_text = ''
            current_file = open('allthetext', 'r')
            current_text = current_file.read()
            current_file.close()

            new_file = open('allthetext', 'w')
            new_text = bigtext + '\n------------------------------------------------------------------------------------------------\n'
            new_text = new_text + current_text

            new_file.write(new_text)
            new_file.close()

            return template("index")
        else:
            return template('index')

 

“S” is the Action of the form on the web page (index.tpl) so that’s the URL the code receives when you submit it. The logic here then just validates that we got some text from the form, and if so, opens the text file, writes the new text in there first, then writes all the older text back in as well. Close the file and re-load the page, and that’s about it. The HTML has python included which opens the file for reading, gets each line and adds that to the output, then closes the file.

I tested this locally and it works like a charm. One of the great things about Bottle is that it includes a development “server” out of the box, so you can test your work at your workstation even if you don’t have a web server running locally.

One thing that was not quite right at this pont was the static CSS files. The were not loading in the browser, as their paths were wrong. I added these Routes to the bottle_app.py script to solve that:

# Static Routes
@app.get('/<filename:re:.*\.js>')
def javascripts(filename):
    return static_file(filename, root='js/')

@app.get('/<filename:re:.*\.css>')
def stylesheets(filename):
    return static_file(filename, root='css/')

@app.route('/', method='GET')
def index():
    return template('index')

 

That just tells Bottle where to look for any file ending with “.css” or (in case I add this in later) “.js”.

So, just to summarize, this is what happens when you first visit the site:

  • the URL requested is handled by the Bottle server
  • it reads my .py code to see what is needed.
  • my .py file says that for the “/” route, just send the template file (index.tpl) back as the response
  • inside that index file, I have Python code reading the contents of the allthetext text file and adding that to the HTML of the page before the browser sees it.

When the user fills in the big text box on that page and clicks the Submit button, this is what happens:

  • Bottle handles the request again, processes the POST request for the “/s” URL
  • Opens the allthetext file, adds the new text from the form field to the top of the file, appends the old text and saves the file again
  • send the index.tpl template again as the response
  • and so the user’s page refreshes with their new text content added to the page beneath the form field and button.

Another really useful feature of Bottle is that when it gets a Form in the request, it knows how to collect up all that data and will also scrub it for things like cross-site-scripting attempts and clickjacking. As the developer of a very simple web app like mine, knowing that I don’t have worry about this is great. Of course, if I were creating an application for production use, I would still very carefully parse the data myself to make sure no bad requests are ever processed.

The last step in my little experiement then was just finding a way to run this online, so I could access the page from anywhere at any time. I found PythonAnywhere, a cloud service which makes it easy to host Python-based web apps like mine. They can support a lot of frameworks too including Flask and Bottle and many others, right out of the box. They very kindly offer a free service for small simple apps and also offer paid services at various levels for folks who need more performance or more CPU / Database time.

I created an account at PythonAnywhere, created my web app, uploaded my files, tweaked them a little for things like paths used and bang! It works. I would recommend this service to anyone else trying out any of the pieces of software I’ve described here. Their documentation is great and gets you up and running very quickly.

I am thinking now about what to do to my little web app next, like maybe adding date and time stamps to each entry and so on, but it was fun creating a really nice looking, HTML 5 compliant, web-based program that I can now run from anywhere, and it only took a few hours. Thanks for all the free stuff that made this all possible: Initializr, Python, Bottle and PythonAnywhere!

🙂

 

Arduino + 1602 LCD stopwatch project!

I recently bought an Arduino Mega2560 beginner’s starter kit from MicroCenter and was looking for an interesting project to use it in. I wanted to use the 1602 LCD from the kit in the project, somehow. It occurred to me that I need a portable stopwatch for my Rubik’s Cube solving so that’s what I made today. It uses the Arduino for counting seconds, minutes and hours and displays the count on a 1602 (16 columns by 2 rows) LCD display. There is a pause button, for when you want to go get some coffee and a reset button for when you lose your focus and need to start over (which happens to me a lot when my kids are around…).

Parts list:

1 x Arduino (Uno or Mega or whatever you have is OK, but there are 8 pins + 5v and GND used here)

1 x1602 LCD display (any as long as it uses Hitachi compatible pins)

1 x potentiometer (I used a 50K Ohm one that came with my kit but usually 10K Ohms is recommended)

1 x breadboard with lots of holes

2 x 4-pin momentary switches (the kind that you click once and then they reset)

2 x 10K Ohm resistors (don’t ask me the colors for the bands; I am partially colorblind. No, really.)

1 x N4001 Diode  (to protect the backlight pin on the LCD)

tons of jumper wires (about 20 actually)

The Arduino sketch:

    #include <LiquidCrystal.h>

    // initialize the library with the numbers of the interface pins
    LiquidCrystal lcd(12, 11, 5, 4, 3, 2);
    // input pins:
    int pausePin = 22;
    int resetPin = 8;
    // output pin:
    int ledPin = 13; 
    // time counting things:
    unsigned long previousMillis = 0;
    long interval = 1000;     // one second
    // pin state trackers:
    int resetButtonState = 0;
    int pauseButtonState = 0;
    int ledState = LOW;
    //de-bounce the switches:
    int lastPauseButtonState = LOW; 
    int lastResetButtonState = LOW; 
    long lastPauseDebounceTime = 0;
    long lastResetDebounceTime = 0;
    long debounceDelay = 50;
    // setup the pause tracker and stopwatch seconds counter
    boolean isPaused = false; 
    unsigned long secs = 0;
    unsigned long mins = 0;
    unsigned long hrs = 0;

    void setup() {
      lcd.begin(16, 2);    // the LCD has 16 columns and 2 rows 
      lcd.setCursor(1,0);    // put the cursor at the first column on the first row
      lcd.print("Stopwatch:");
      pinMode(ledPin, OUTPUT);  
      pinMode(resetPin, INPUT);
      pinMode(pausePin, INPUT);
      Serial.begin(9600);     // for logging on the PC
    }

    void loop() { 
      Serial.print(" - Hrs: "); 
      Serial.print(hrs);
      Serial.print(" - Mins: "); 
      Serial.print(mins);
      Serial.print("Secs: "); 
      Serial.print(secs);
      Serial.print(" --- pause state: ");
      Serial.print(pauseButtonState);
      Serial.print(" --- isPaused: ");
      Serial.print(isPaused);  
      Serial.print(" --- reset state: ");
      Serial.println(resetButtonState);

      pauseButtonState = digitalRead(pausePin);
      if (pauseButtonState != lastPauseButtonState) {
        lastPauseDebounceTime = millis();
      }

      if ((millis() - lastPauseDebounceTime) > debounceDelay) {
        if (pauseButtonState == HIGH) {
          Serial.println("*******************   PAUSE   ***************************");
          if (isPaused == true) {
            // start again
            isPaused = false;
          } else { 
            //pause now
            isPaused = true;     
          }
        }
      } 

      resetButtonState = digitalRead(resetPin);
      if (resetButtonState != lastResetButtonState) {
        lastResetDebounceTime = millis();
      }

      if ((millis() - lastResetDebounceTime) > debounceDelay) {
        if (resetButtonState == HIGH) {
          Serial.println("^^^^^^^^^^^^^^^^^^^^^^^^  RESET  ^^^^^^^^^^^^^^^^^^^^^^^^^^");
          secs = 0; 
          mins = 0;
          hrs = 0;
          isPaused = false;
          lcd.setCursor(1,1);
          lcd.print("               "); 
        }
      }

      if (isPaused == false) {
        unsigned long currentMillis = millis();  
        if(currentMillis - previousMillis >= interval) {
          secs = secs + 1;
          if (secs == 60) { 
            secs = 0;
            mins = mins + 1;
          }
          if (mins == 60) {
            mins = 0;
            hrs = hrs + 1;
          }       
          lcd.setCursor(1,1);
          String theTime = String(hrs);
          if (mins >= 10) {
            theTime = theTime + ":" + String(mins);
          } else {  
            theTime = theTime + ":0" + String(mins);
          }
          if (secs >= 10) {
            theTime = theTime + ":" + String(secs);
          } else {  
            theTime = theTime + ":0" + String(secs);
          }
          lcd.print(theTime);

          // flash the LED for status...
          previousMillis = currentMillis;   
          if (ledState == LOW) {
            ledState = HIGH;
          } else {
            ledState = LOW;
          }
          digitalWrite(ledPin, ledState);
        }
        delay(10);
        lastPauseButtonState = pauseButtonState;
        lastResetButtonState = resetButtonState; 
      }
    }

Initial hardware setup:

  1. Don’t connect the Arduino to power (your PC) until this is all setup.
  2. Drop the LCD onto the breadboard. I put mine over on the right side.
  3. Drop the potentiometer onto the breadboard away from the LCD.
  4. Jump wire from the 5V pin on the Arduino to the “+” rail on the breadboard.
  5. Jump wire from the GND pin on the Arduino to the “-” rail on the breadboard.
  6. Jump wire from the + rail to the left-most pin on the pot.
  7. Jump wire from the – rail to the right-most pin on the pot.
  8. Jump wire from pin “Vss” or #1 on the LCD to the – rail on the breadboard.
  9. Jump wire from pin “Vdd” or #2 on the LCD to the + rail on the breadboard.
  10. Jump wire from pin “V0” or #3 on the LCD to the middle pin on the pot.
  11. Diode from the + rail to pin “A” or #15 on the LCD.
  12. Jump wire from pin “K” or #16 to the – rail on the breadboard.
  13. Plug in the Arduino to your computer. (That’s right no sketch yet at this point)
  14. The LCD should be backlit if you did the “A” and “K” pins correctly.
  15. Adjust the pot’s knob. You should be able to turn it all the way down and then see a row of boxes on the LCD.
  16. If not, stop here and fix that. GIYF.
  17. Unplug the Arduino.
  18. Jump wire from pin “RW” on the LCD to GND on the breadboard.
  19. Jump wire from pin “RS” on the LCD to pin #12 on the Arduino.
  20. Jump wire from pin “E” on the LCD to pin #11 on the Arduino.
  21. Jump wire from pin #2 to pin “DB7” on the LCD.
  22. Jump wire from pin #3 to pin “DB6” on the LCD.
  23. Jump wire from pin #4 to pin “DB5” on the LCD.
  24. Jump wire from pin #5 to pin “DB4” on the LCD.
  25. Plug in the Arduino. Publish the sketch to your Arduino and watch it run.
  26. If everything is just right, you’ll see “Stopwatch:” and the time counting up on the LCD display.
  27. If not, pull your hair out and troubleshoot here before moving on. Curse a lot, it helps.

Add the switches to the board:

  1. Drop the two 4-pin switches onto the breadboard, spanning the middle valley.
  2. Name one of them reset and one pause in your mind. Keep them separate.
  3. Jump wire from the reset switch on the same side as the other components to pin #8 (or whatever you like) on the breadboard.
  4. Jump wire from the pause switch on the same side as the other components to pin #22 (or whatever you like) on the breadboard.
  5. Jump wire from the reset switch on the opposite side of the breadboard from the other components to the + rail on the breadboard.
  6. Jump wire from the pause switch on the opposite side of the breadboard from the other components to the + rail on the breadboard.
  7. 10K Ohm resistor from the other corner of the reset switch to another line on the opposite side of the breadboard from the other components.
  8. 10K Ohm resistor from the other corner of the pause switch to another line on the opposite side of the breadboard from the other components.
  9. Jump wire from from the reset switch’s resistor’s second line to GND on the breadboard.
  10. Jump wire from from the pause switch’s resistor’s second line to GND on the breadboard.
  11. Whew, that’s it!

LCD Stopwatch - Fritzing diagram

Close-up on the Arduino

Close-up on the pot and switchesClose-up on the LCD and wires

Run it!

With all of that wiring, connecting the LCD to the pot and he Arduino, and the switches wired to their own digital pins on the Arduino and to power and ground, you’re all set.

When you plug the Arduino back in, load the program if it’s not already there and voila! (Hopefully) The screen will show the stopwatch and start counting up. When you press the Reset button, the time should reset to zero. When you press the Pause button, the time should freeze, until you press Pause again.

Troubleshooting

I had trouble with the lCD not displaying the text I sent it at first. I found it was very lossly connecting to the breadboard, so I had to jiggle it and remove/replace it a few times until I could feel it going into the holes on the board properly and then it worked from there. If you have trouble, try this and then check online for other solutions. I rely on the kind folks at http://reddit.com/r/arduino for lots of help and you can too.

Conclusion

If you try this out, let me know how it goes in the comments. Let me know too if you see any really significant improvements I could make. Thanks for reading this far.

🙂

Playing around with Jabber on the Rasberry Pi

Hi all, this is just a post about my experience today trying something new on my Pi. I have been looking for something my son could do with his Pi, as it’s been sitting on a shelf gathering dust for some time. I thought maybe it would be easy to set it up as a chat / Jabber / XMPP server so he and his buddies could chat whenever they like and so he would learn some about computer networking and Linux services and so on. The Pi is already running Raspbian as the OS so this seemed straight forward.

Here are the steps we followed:

sudo apt-get update 
sudo apt-get upgrade

(Just to get everything up to date in Raspbian land)

sudo apt-get install prosody

Prosody is the XMPP / Jabber server I chose to use, after reading about it and ejabberd online. They both offer lots of options but Prosody won out for its seemingly simple installation. I made that choice pretty quickly though, so I really don’t know ejabberd much.

cd /etc/prosody
sudo cp prosody.cfg.lua ./prosody.cfg.BAK
sudo nano prosody.cfg.lua

Now I needed to make a backup of the config file (LUA really) for Prosodyand then edit it. The server’s web site is very clear about most of this, so reference that if you need help:

https://prosody.im/doc/configure

First, you need to set up a virtual host name. This concerned me at first, as all of the examples given looked like real domain names, like example.com, and I didn’t want to pay for any service like that (and I don’t own one already). I ended up just using my Pi’s name on the network, “firstpi” and that works fine as long as the clients are all inside the house.

Next, I enabled registration as its called, as that seemed the easiest way to get “accounts” setup on my new Chat server. Note to self — don’t forget to disable registration before opening the server to the full internet, or any oddball out there could also register an account for themselves!

sudo service prosody restart

… to make those config changes take effect.

Sure enough, now I have the Pi and Prosody listening on my LAN. I needed a Jabber client to connect to it with. I ended up using Gajim, since my laptops all run Windows and that worked out OK. It has a nice installer and runs OK for the most part. I have seen about six “An error just occurred…” messages in just a few minutes of playing with it, but it’s an Open Source project, so that’s OK.

In Gajim, I just added a couple accounts for me and my son. Creating the accounts in Gajim was easy (since I had the server setup to allow registration via the client) and before long, I was able to open a group chat room for my multiple accounts and chat. I named the accounts like, “dad@mypi” and “theboy@mypi” and Gajim found the server on its own.

I installed yaxim on my Android tablet next and again, setting up and connecting was easy there. It is limited to only one account in use at a time though, so that might be a concern for some.

https://play.google.com/store/apps/details?id=org.yaxim.androidclient&hl=en

Now we can chat at each other from our Windows laptops and Android tablets with ease. Next step will be setting this up so we can chat from outside the house as well.

The advantages of doing this are: we set it all up ourselves, it’s a private chat channel, we learned a little about networking along the way.

🙂

Cron stopped sending emails; dead.letter file was a clue

Quick post:

Today I realized that a few days ago I stopped receiving emails from the Raspberry Pi running the latest version of the Raspbian OS. I have Cron setup to run a couple of scripts on a daily or weekly basis and by default cron emails the results of those scripts to my personal email address. I realized those emails stopped coming a few days ago but couldn’t remember making any changes that should have caused this. Here are the steps I took to troubleshoot this:

1. Check crontab -l to list the jobs associated with my Pi login. They looked the same as always.

2. Run one of the scripts manually from the terminal. Yep, I get the expected result there.

3. Re-configure cron with crontab -l so the job will run in about two minutes. Wait three minutes… No output. Still no email. Hmmmm…

4. Notice there is a file in my Home directory named dead.letter that I never saw there before. Read up on what that is…

5. Look for logs in the /var/log. Notice that mail.log there has these entries:

May  5 13:13:08 firstpi sSMTP[10443]: Creating SSL connection to host
May  5 13:13:08 firstpi sSMTP[10443]: SSL connection using RSA_ARCFOUR_SHA1
May  5 13:13:08 firstpi sSMTP[10443]: Authorization failed (535 5.7.8 http://support.google.com/mail/bin/answer.py?answer=14257 om5sm30207817igb.16 – gsmtp)

6. Remember that I changed the password on this email account a few days ago!

7. Edit the ssmtp.conf file which stores the credentials for this email account and update the password in there. Save the file, exit, breathe a sigh of relief.

Success!

🙂