SyntaxHighlighter

Saturday, July 28, 2012

.pam_environment syntax in ubuntu

Some time ago, I started configuring my environment variables on ubuntu in /etc/profile. I thought I needed to set my environment variables in ~/.bashrc or ~/.bash_profile but when I open up terminals, they didn't seem to be used. So I resorted to the system-wide /etc/profile. I added things like JAVA_HOME, GROOVY_HOME, etc by adding lines like these at the bottom of the file:


export JAVA_HOME=/home/tim/apps/java/current
export GROOVY_HOME=/home/tim/apps/groovy/current

Each of those "current"  directories was a symlink to the version I was using. Then, I put bin directories in PATH variable.


export PATH=$PATH:$JAVA_HOME/bin:$GROOVY_HOME/bin


This all worked fine for me. But I learned that ubuntu was recommending user-specific environment variables in ~/.pam_environment. See https://help.ubuntu.com/community/EnvironmentVariables

I decided I'd move my configurations to the proper place and improve my linux knowledge. Being careful, however, I've only commented out the sections in /etc/profile so I could quickly revert them back, in case it somehow didn't work as expected.


On ubuntu help page mentioned above, it says:


~/.pam_environment - This file is specifically meant for setting a user's environment. It is not a script file, but rather consists of assignment expressions, one per line.

Assignment expressions. Ok. I'll just remove those "export" keywords.


JAVA_HOME=/home/tim/apps/java/current
GROOVY_HOME=/home/tim/apps/groovy/current
PATH=$PATH:$JAVA_HOME/bin:$GROOVY_HOME/bin

This should do it!


I rebooted the machine, got to the login screen, typed in my password, and then, typed in my password, typed in my password, and then.. errr... what is going on here?


It appeared to me that I wasn't able to login to my own account anymore! First, I was just annoyed. And then, I was afraid. How am I going to fix this, since I only have my own login to this machine, and the guest account can't do anything useful like changing someone else's configuration files.


Luckily, I was still able to use the browser logging in as "guest".
After googling around, I found I could boot into recovery mode by pressing left shift key while booting up. And then, googling around a bit more, I also discovered that the .pam_environment file has a special configuration format different to the usual /etc/profile.


What I had to follow was to use pam's configuration file style.
The whole thing was not quite clear to me, but reading man pages of pam and pam_env.conf gave me a hint as to how I should write my .pam_environment file.


So, after many reboots in recovery mode, dropping into command line as root, editing my .pam_environment to try different things, I came up with a working configuration:



JAVA_HOME=/home/tim/apps/java/current
GROOVY_HOME=/home/tim/apps/groovy/current
PATH DEFAULT=${PATH}:${JAVA_HOME}/bin:${GROOVY_HOME}/bin



I hope ubuntu help page will describe it a bit better in the future.

Tuesday, July 24, 2012

Calling closure in a Map

In groovy, you can store a piece of code in a Map (or even in List or whatever data structure you fancy), and call it when you want to.
It's a feature provided by closure.

For example, you can try this in groovy shell (groovysh):

groovy:000> a = [name: {return "Jack Sparrow"}]
===> {name=groovysh_evaluate$_run_closure1@26a0e990}
groovy:000> a.name
===> groovysh_evaluate$_run_closure1@26a0e990
groovy:000> a.name()
===> Jack Sparrow
groovy:000> a.name.call()
===> Jack Sparrow

But then, I was a bit confused by this:

groovy:000> a = [size: {return 5}]
===> {size=groovysh_evaluate$_run_closure1@707efa96}
groovy:000> a.size
===> groovysh_evaluate$_run_closure1@707efa96
groovy:000> a.size()
===> 1
groovy:000> a.size.call()
===> 5

Huh? Why is a.size() == 1?
I expected it to give 5.
But only when I added another closure, it dawned on me.

groovy:000> a = [age: {return 29}, size: {return 5}]
===> {age=groovysh_evaluate$_run_closure1@71257687, size=groovysh_evaluate$_run_closure2@5288d319}
groovy:000> a.age
===> groovysh_evaluate$_run_closure1@71257687
groovy:000> a.age()
===> 29
groovy:000> a.age.call()
===> 29
groovy:000> a.size
===> groovysh_evaluate$_run_closure2@5288d319
groovy:000> a.size()
===> 2
groovy:000> a.size.call()
===> 5

It might be very trivial to someone who's more familiar with groovy language, but it stumped me for a while.
Can you figure out what was going on?