tag:blogger.com,1999:blog-23835418002241955962009-06-26T13:52:57.673-04:00Jason Noble's Technical AdventuresJason Noblehttp://www.blogger.com/profile/12779892190227704860noreply@blogger.comBlogger17125tag:blogger.com,1999:blog-2383541800224195596.post-35564049026393616982009-06-26T10:24:00.002-04:002009-06-26T10:27:27.366-04:00Installing Ruby 1.9 on Mac OS X 10.5Tommie over at <span style="font-style:italic;">The blog with no name</span> has written up a how-to on getting Ruby 1.9 up and running in Mac OS X 10.5.7.
Check it out: <a href="http://www.tommycampbell.net/2009/06/26/ruby-1-9-on-mac-os-x">http://www.tommycampbell.net/2009/06/26/ruby-1-9-on-mac-os-x</a><div class="blogger-post-footer"><img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/2383541800224195596-3556404902639361698?l=jasonnoble.org'/></div>Jason Noblehttp://www.blogger.com/profile/12779892190227704860noreply@blogger.com0tag:blogger.com,1999:blog-2383541800224195596.post-27925481997969506522009-06-04T14:32:00.002-04:002009-06-04T14:41:39.089-04:00How to optionally load gemsI ran into an interesting problem today at work. I wanted to optionally require a ruby gem. For example, I wanted to include the <a href="http://metric-fu.rubyforge.org/">metric_fu</A> rake tasks, but if a user doesn't have metric_fu installed, it's no big deal.
Here's how I implemented it. I added a method to the Kernel space (where require lives) called desire. I desire this gem to be installed, but I should just see a warning if it's not installed, instead of an error.
Here's the desire method (I put this in RAILS_ROOT/Rakefile).
<pre>
module Kernel
def desire(library)
require library
return true
rescue LoadError
STDERR.puts "warning: #{library} gem desired but not found."
return false
end
end
</pre>
And here's how I optionally include the metric_fu gem.
<pre>
desire 'metric_fu'
</pre>
If a user has the gem installed, rake -T will show the tasks.
<pre>
jasonn:~/sources/my_files [master] $ rake -T
(in ~/sources/my_files)
rake metrics:all
[...]
jasonn:~/sources/my_files [master] $
</pre>
If the user does not have the gem installed, rake -T will show a simple warning and continue.
<pre>
jasonn:~/sources/my_files [master] $ rake -T
(in ~/sources/my_files)
warning: metric_fu gem desired but not found.
[...]
jasonn:~/sources/my_files [master] $
</pre>
I don't remember where I first heard the desire idea, but it works pretty well.<div class="blogger-post-footer"><img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/2383541800224195596-2792548199796950652?l=jasonnoble.org'/></div>Jason Noblehttp://www.blogger.com/profile/12779892190227704860noreply@blogger.com1tag:blogger.com,1999:blog-2383541800224195596.post-28056964791557953592009-06-03T02:35:00.004-04:002009-06-03T02:41:33.930-04:00Recover files from git after deletion is committedAs I work with git, I have occasionally deleted files that I later end up needing. You can restore the files, but it's not an intuitive process.
First, I looked at the output of git log to find the commit that I deleted the files with.
Then I ran:
<pre>
git diff --diff-filter=D --name-only HEAD@{'7 days ago'} | \
grep -v ^doc | perl -ne 'chomp;
print "git show e2ae03f14ce81a3e24ec6cbe2e73767f4c6fed1b -- $_ > $_\n";' | sh
</pre>
This restored all the files I deleted that did not start with doc.
These files show up as diff files, so you may have to remove the "- " at the beginning of the lines and commit messages. But hey, it's better than not having the files you need.<div class="blogger-post-footer"><img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/2383541800224195596-2805696479155795359?l=jasonnoble.org'/></div>Jason Noblehttp://www.blogger.com/profile/12779892190227704860noreply@blogger.com0tag:blogger.com,1999:blog-2383541800224195596.post-56860336128759539902009-04-24T15:32:00.003-04:002009-05-03T15:10:35.905-04:00RDoc preview in TextmateWhile working on writing some RDOC documentation within Textmate, <a href=http://reinh.com>Rein Henrichs</a> and I ran into a problem. Textmate will give you previews if you use markdown, but not RDOC. Below is how to do it.<P>
Create /usr/bin/rdoc2html:<P>
<pre>
#!/usr/bin/env ruby -rrdoc/markup/simple_markup -rrdoc/markup/simple_markup/to_html
input = ARGF ? ARGF.read : STDIN.read
print SM::SimpleMarkup.new.convert(input, SM::ToHtml.new)
</pre>
Make the file executable:<P>
<pre>
sudo chmod +x /usr/bin/rdoc2html
</pre>
In textmate, hit Cntrl-Option-Command-P. In the bottom left corner, click Show options. Click the Pipe text through checkbox and enter "/usr/bin/rdoc2html" in the text box.<div class="blogger-post-footer"><img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/2383541800224195596-5686033612875953990?l=jasonnoble.org'/></div>Jason Noblehttp://www.blogger.com/profile/12779892190227704860noreply@blogger.com0tag:blogger.com,1999:blog-2383541800224195596.post-43469756183690540662009-04-21T19:18:00.004-04:002009-05-03T15:09:02.557-04:00GitHub set author/committerI've been working with Git lately and more specifically <a href=http://github.com>GitHub</A>. I noticed something odd one day. Where most commits have one author listed, these commits had both a author and a committer listed.<P>
More specifically, the author field was the two members of the pair and the committer name was the workstation they were working on.<P>
Example:<P>
<a onblur="try {parent.deselectBloggerImageGracefully();} catch(e) {}" href="http://jasonnoble.org/uploaded_images/Picture-1-779797.png"><img style="display:block; margin:0px auto 10px; text-align:center;cursor:pointer; cursor:hand;width: 320px; height: 53px;" src="http://jasonnoble.org/uploaded_images/Picture-1-779792.png" border="0" alt="" /></a>
<P>
After playing around with it for a while, I figured out how to do it. Within a git project clone, run the following commands.
<P>
1) git config user.name = 'Workstation Perlwizard'<BR>
2) git config user.email = 'devs2@perlwizard.org'<BR>
3) export GIT_AUTHOR_NAME='Bob Smith and Betty Jones'<BR>
4) export GIT_AUTHOR_EMAIL='devs@perlwizard.org'<p>
Both of the email addresses need to be addresses Github doesn't know about (i.e. not used to sign up for an account). <P>
I could see using this while pairing with someone so you both get name credit for your commits.<P><div class="blogger-post-footer"><img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/2383541800224195596-4346975618369054066?l=jasonnoble.org'/></div>Jason Noblehttp://www.blogger.com/profile/12779892190227704860noreply@blogger.com1tag:blogger.com,1999:blog-2383541800224195596.post-47124119349470625352008-09-20T20:17:00.003-04:002008-09-20T20:28:31.724-04:00Making TextMate recognize .html.erb filesI'm using <a href=http://macromates.com/>TextMate</A> to edit some Ruby on Rails code. I noticed that when I edit the view pages (i.e. index.html.erb) it does not do syntax highlighting. It turns out this is because the Ruby on Rails TextMate bundle thinks that ruby view pages are .rhtml, which is the old Ruby on Rails format.<P>
You can fix this by editing the Ruby on Rails language bundle.<P>
In TextMate, select Bundles -> Bundle Editor -> Edit Languages.<P>
This will open a new Bundle Editor Window. Click the arrow next to Ruby on Rails and highlight HTML (Rails). In the code that appears on the right, change the following
<pre>
fileTypes = ( 'rhtml');
</pre>
to
<pre>
fileTypes = ('rhtml', 'html.erb');
</pre>
Click the red X in the upper left corner to close the Bundle Editor Window.<P>
Close any existing .html.erb files, when you open them again, they will have ruby syntax highlighting.<P><div class="blogger-post-footer"><img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/2383541800224195596-4712411934947062535?l=jasonnoble.org'/></div>Jason Noblehttp://www.blogger.com/profile/12779892190227704860noreply@blogger.com0tag:blogger.com,1999:blog-2383541800224195596.post-7681364507759044102008-09-14T11:40:00.001-04:002008-09-14T11:42:33.086-04:00Adding an alias to the Tcsh ShellIf you run a command frequently, you can create what is called an alias. The alias is usually a shorter form of the original command.
To make these aliases permament, you should edit .tcshrc in your home directory.
<pre>
11:32:07 /Users/jasonn $ vi ~/.tcshrc
</pre>
Add a line of text of the format "alias YYY 'ZZZ'" where YYY is the command you want to type in, and ZZZ is the command you want to be run.
Examples:
<pre>
# Show long listing
alias ll 'ls -al'
# Use VI Improved instead of Vi
alias vi 'vim'
# Change into your web directory
alias web 'cd /usr/local/apache/vhosts/jasonnoble.org/htdocs'
</pre>
This alias ability allows you to type less at your shell prompt. If you find yourself typing the same long command line over and over, try adding an alias to your shell that will save you some time.<div class="blogger-post-footer"><img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/2383541800224195596-768136450775904410?l=jasonnoble.org'/></div>Jason Noblehttp://www.blogger.com/profile/12779892190227704860noreply@blogger.com0tag:blogger.com,1999:blog-2383541800224195596.post-53525435954535388442008-09-14T11:28:00.004-04:002008-09-14T11:39:32.948-04:00Adding an alias to the Bash ShellIf you run a command frequently, you can create what is called an alias. The alias is usually a shorter form of the original command.
To make these aliases permament, you should edit .bashrc in your home directory.
<pre>
11:32:07 /Users/jasonn $ vi ~/.bashrc
</pre>
Add a line of text of the format "alias YYY='ZZZ'" where YYY is the command you want to type in, and ZZZ is the command you want to be run.
Examples:
<pre>
# Show long listing
alias ll='ls -al'
# Use VI Improved instead of Vi
alias vi='vim'
# Change into your web directory
alias web='cd /usr/local/apache/vhosts/jasonnoble.org/htdocs'
</pre>
This alias ability allows you to type less at your shell prompt. If you find yourself typing the same long command line over and over, try adding an alias to your shell that will save you some time.<div class="blogger-post-footer"><img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/2383541800224195596-5352543595453538844?l=jasonnoble.org'/></div>Jason Noblehttp://www.blogger.com/profile/12779892190227704860noreply@blogger.com0tag:blogger.com,1999:blog-2383541800224195596.post-47160581139231064432008-09-10T09:21:00.002-04:002008-09-10T09:21:58.306-04:00Free Online BooksA co-worker pointed me to a bunch of free online <a href=http://www.linuxtopia.org/online_books/index.html>books</a>. Looks pretty cool, I'll have to check a couple of them out.<div class="blogger-post-footer"><img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/2383541800224195596-4716058113923106443?l=jasonnoble.org'/></div>Jason Noblehttp://www.blogger.com/profile/12779892190227704860noreply@blogger.com0tag:blogger.com,1999:blog-2383541800224195596.post-44716938724448853962008-09-03T13:46:00.001-04:002008-09-03T13:48:02.448-04:00Google Chrome first experienceI just installed Google <a href=http://www.google.com/chrome>Chrome</a>, a new web browser by Google.
Below is my first experience:
<IMG SRC=/images/google_chrome.jpg><div class="blogger-post-footer"><img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/2383541800224195596-4471693872444885396?l=jasonnoble.org'/></div>Jason Noblehttp://www.blogger.com/profile/12779892190227704860noreply@blogger.com0tag:blogger.com,1999:blog-2383541800224195596.post-30885025543149563422008-09-02T19:23:00.015-04:002008-09-02T21:13:51.797-04:00Allow rails users to login via vsftpdSo you've <a href=http://jasonnoble.org/2008/09/installing-ruby-on-rails-in-freebsd.html>installed</a> Ruby on Rails with Postgres, and now you want to allow your users to login to your server via FTP.
First we need to install our FTP server. (In the config window, I left the defaults)
<pre>
freebsd-vm# cd /usr/ports/ftp/vsftpd
freebsd-vm# make install
cd /usr/ports/ftp/vsftpd && make config;
[...]
ports included in the Ports Collection. Please type 'make deinstall'
to deinstall the port if this is a concern.
For more information, and contact details about the security
status of this software, see the following webpage:
http://vsftpd.beasts.org/
freebsd-vm#
</pre>
Edit the vsftpd conf file.
<pre>
#vi /usr/local/etc/vsftpd.conf
# Remove all lines and replace with the following
background=YES
listen=YES
anonymous_enable=NO
local_enable=YES
virtual_use_local_privs=YES
write_enable=YES
connect_from_port_20=YES
secure_chroot_dir=/usr/local/share/vsftpd/empty
pam_service_name=vsftpd
guest_enable=YES
user_sub_token=$USER
local_root=/usr/local/www/apache22/data/$USER
chroot_local_user=YES
hide_ids=YES
ftpd_banner=Welcome to FTP server
file_open_mode=0770
local_umask=0000
anon_mkdir_write_enable=NO
guest_username=vsftpd
user_config_dir=/etc/vsftpd/users
</pre>
Create the vsftpd startup script.
<pre>
# vi /usr/local/etc/rc.d/vsftpd
#!/bin/sh
#
# $FreeBSD: ports/ftp/vsftpd/files/vsftpd.sh.in,v 1.7 2006/02/20 20:47:01 dougb Exp $
#
# PROVIDE: vsftpd
# REQUIRE: DAEMON
# Add the following line to /etc/rc.conf to enable `vsftpd':
#
# vsftpd_enable="YES"
# vsftpd_flags="/some/path/conf.file" # Not required
#
. "/etc/rc.subr"
name="vsftpd"
rcvar=`set_rcvar`
load_rc_config "$name"
: ${vsftpd_enable:="NO"}
: ${vsftpd_flags:=""}
command="/usr/local/libexec/$name"
required_files="/usr/local/etc/$name.conf"
start_precmd="vsftpd_check"
vsftpd_check()
{
if grep -q "^ftp[ ]" /etc/inetd.conf ${required_files}
then
err 1 "ftp is already activated in /etc/inetd.conf"
fi
if ! egrep -q -i -E "^listen.*=.*YES$" ${required_files}
then
err 1 "vsftpd script need "listen=YES" on config file"
fi
if ! egrep -q -i -E "^background.*=.*YES$" ${required_files}
then
err 1 "vsftpd script need "background=YES" on config file"
fi
}
run_rc_command "$1"
</pre>
We also need to install the Postgres PAM module.
<pre>
freebsd-vm# cd /usr/ports/security/pam-pgsql/
freebsd-vm# make install
=> libpam-pgsql-0.6.3.tar.bz2 doesn't seem to exist in /usr/ports/distfiles/.
=> Attempting to fetch from http://nchc.dl.sourceforge.net/sourceforge/pam-pgsql/.
[...]
to use this module. Note, that unlike most other ports, this port
installs a file into /usr/lib directly (/usr/lib/pam_pgsql.so),
because PAM requires that.
===> Registering installation for pam-pgsql-0.6.3_1
freebsd-vm#
</pre>
We need to tell vsftpd to use pam-pgsql for authentication.
<pre>
freebsd-vm# vi /etc/pam.d/vsftpd
auth required pam_pgsql.so config_file=/etc/pam_pgsql_vsftpd.conf
account required pam_pgsql.so config_file=/etc/pam_pgsql_vsftpd.conf
freebsd-vm# vi /etc/pam_pgsql_vsftpd.conf
debug
pw_type = md5
connect = hostaddr=127.0.0.1 port=5432 dbname=<font color="red">sample_development</font> \
user=<font color="red">sample_user</font> password='<font color="red">password</font>' connect_timeout=15
auth_query = select <font color="yellow">password</font> from <font color="yellow">users</font> where <font color="yellow">username</font> = %u
acct_query = select <font color="white">ftp_disabled</font> as acc_expired, \
0 as acc_new_pwreq, (password ISNULL or password = '') as user_password \
from <font color="white">users</font> where username = %u
freebsd-vm#
</pre>
(Long lines above with \ in them should be one long continuous line in your config file)<P>
The configuration above assumes three things:
<UL>
<li>Your rails database name is <font color="red">sample_development</font> and can be connected with the username <font color="red">sample_user</font> and the password <font color="red">password</font> (change red items as needed)</li>
<li>Your rails users are stored in the <font color="yellow">users</font> table. The password field is <font color="yellow">password</font> and the username field is <font color="yellow">username</font> (change yellow items as needed)</li>
<li>Your rails users are stored in the <font color="white">users</font> table. The <font color="white">ftp_disabled</font> field holds whether they should be allowed access (false) or denied (true). (Change white items as needed)</li>
</ul>
We need to create the user for vsftpd to run as.
<pre>
freebsd-vm# adduser
Username: vsftpd
Full name: FTP User
Uid (Leave empty for default):
Login group [vsftpd]:
Login group is vsftpd. Invite vsftpd into other groups? []:
Login class [default]:
Shell (sh csh tcsh nologin) [sh]: nologin
Home directory [/home/vsftpd]:
Use password-based authentication? [yes]: no
Lock out the account after creation? [no]:
Username : vsftpd
Password : <disabled>
Full Name : FTP User
Uid : 1002
Class :
Groups : vsftpd
Home : /home/vsftpd
Shell : /usr/sbin/nologin
Locked : no
OK? (yes/no): yes
adduser: INFO: Successfully added (vsftpd) to the user database.
Add another user? (yes/no): no
Goodbye!
freebsd-vm#
</pre>
Configure vsftpd to start on boot and manually start vsftpd once.
<pre>
freebsd-vm# echo vsftpd_enable="YES" >> /etc/rc.conf
freebsd-vm# chmod 755 /usr/local/etc/rc.d/vsftpd
freebsd-vm# /usr/local/etc/rc.d/vsftpd start
Starting vsftpd.
freebsd-vm#
</pre>
Now we need to create a couple of users to test this with. Below we'll create a user bob that can login and a user bob2 that can not login.
<pre>
freebsd-vm$ psql -U sample_rails_app sample_rails_app_development
Welcome to psql 8.3.3, the PostgreSQL interactive terminal.
Type: \copyright for distribution terms
\h for help with SQL commands
\? for help with psql commands
\g or terminate with semicolon to execute query
\q to quit
sample_rails_app_development=> select * from users;
id | username | password | homedir | ftp_disabled | created_at | updated_at
----+----------+----------+---------+--------------+------------+------------
(0 rows)
sample_rails_app_development=> insert into users (username,password,homedir,ftp_disabled) values ('bob',md5('bob'),'/home/bob',false);
INSERT 0 1
sample_rails_app_development=> insert into users (username,password,homedir,ftp_disabled) values ('bob2',md5('bob2'),'/home/bob2',true);
INSERT 0 1
sample_rails_app_development=> select * from users;
id | username | password | homedir | ftp_disabled | created_at | updated_at
----+----------+----------------------------------+------------+--------------+------------+------------
2 | bob | 9f9d51bc70ef21ca5c14f307980a29d8 | /home/bob | f | |
3 | bob2 | 436187b1cb437f7ddb11952542e351a4 | /home/bob2 | t | |
(2 rows)
sample_rails_app_development=> \q
freebsd-vm$
</pre>
Next, we need to create bob's virtual home directory (as root).
<pre>
freebsd-vm# mkdir -p /usr/local/www/apache22/data/bob
freebsd-vm# chown vsftpd:vsftpd /usr/local/www/apache22/data/bob
</pre>
Now we can login as bob (and not bob2).
<pre>
freebsd-vm$ ftp localhost
Trying 127.0.0.1...
Connected to localhost.
220 Welcome to FTP server
Name (localhost:jasonn): bob
331 Please specify the password.
Password:
230 Login successful.
Remote system type is UNIX.
Using binary mode to transfer files.
ftp> pwd
Remote directory: /
ftp> quit
221 Goodbye.
freebsd-vm$ ftp localhost
Trying 127.0.0.1...
Connected to localhost.
220 Welcome to FTP server
Name (localhost:jasonn): bob2
331 Please specify the password.
Password:
530 Login incorrect.
ftp: Login failed.
ftp> quit
221 Goodbye.
freebsd-vm$
</pre>
Now we need a script that will setup the FTP user's home directory. (Run this in your Rails project's home directory)
<pre>
freebsd-vm# echo "User.write_ftp_configs" >> write_ftp_configs
freebsd-vm#
</pre>
Change the User model in your Rails application
<pre>
freebsd-vm$ vi app/models/user.rb
class User < ActiveRecord::Base
require 'ftools'
def self.write_ftp_configs
User.find(:all, :conditions => ["ftp_disabled is false"]).each do |user|
unless File.directory?(user.homedir)
# If their home directory doesn't exist, create it
# and chown the directory to vsftpd (1002)
File.makedirs user.homedir
File.chown(1002, 1002, user.homedir)
end
if File.directory?("/etc/vsftpd/users")
unless Dir.entries("/etc/vsftpd/users").include?(user.homedir)
File.open("/etc/vsftpd/users/#{user.username}", 'w') {|f|
f.write("local_root=#{user.homedir}\n");
}
end
end
end
end
end
</pre>
Now you just need to setup a cronjob to run as root. This script should cd to your rails root directory and run "script/runner write_ftp_configs". <P>
<font color="red">WARNING</font>: There may be security implications with this script. If a user has their homedir set to something "sensitive" this script will create that directory for them. Make sure you trust anyone that can add users via your Rails application. You could fix this by adding a validation on the homedir field so that it's only a subdirectory of a given directory (i.e. $RAILS_ROOT/upload) or something.<div class="blogger-post-footer"><img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/2383541800224195596-3088502554314956342?l=jasonnoble.org'/></div>Jason Noblehttp://www.blogger.com/profile/12779892190227704860noreply@blogger.com0tag:blogger.com,1999:blog-2383541800224195596.post-26786142086496456862008-09-01T23:09:00.008-04:002008-09-02T19:19:49.280-04:00Installing Ruby on Rails in FreeBSD with PostgresAfter setting up a minimal FreeBSD <a href=http://jasonnoble.org/2008/09/setting-up-freebsd-vm-with-parallels.html>install</a>, I want to get Ruby on Rails up and running.
Let's see what's available in the ports tree.
<pre>
freebsd-vm# make search key=gem | grep Path: | grep ruby
Path: /usr/ports/audio/rubygem-mp3info
Path: /usr/ports/benchmarks/rubygem-railsbench
Path: /usr/ports/databases/ruby-dbd_pg
Path: /usr/ports/databases/ruby-rdbc1
Path: /usr/ports/databases/rubygem-activerecord
[...]
Path: /usr/ports/devel/ruby-gemfinder
Path: /usr/ports/devel/ruby-gems
Path: /usr/ports/devel/rubygem-activesupport
[...]
Path: /usr/ports/www/rubygem-scrubyt
Path: /usr/ports/www/rubygem-taggable
freebsd-vm#
</pre>
The ruby-gems port looks promising.
<pre>
freebsd-vm# cd devel/ruby-gems
freebsd-vm# cat pkg-descr
a package management framework for the Ruby programming language
An application or library is packaged into a gem, which is
a single installation unit.
RubyGems entirely manages its own filesystem space, rather
than installing files into the "usual" places. This enables
greater functionality and reliability.
Using RubyGems, you can:
- download and install Ruby libraries easily
- not worry about libraries A and B depending on
different versions of library C
- easily remove libraries you no longer use
- have power and control over your Ruby platform!
WWW: http://docs.rubygems.org/
freebsd-vm#
</pre>
Let's install it. The install requires ruby to be installed, so it installs that for us (if it's not currently installed). I enabled RDOC and DEBUG and disabled IPV6.
(This may take a while, go grab some coffee)
<pre>
freebsd-vm# make install
=> rubygems-1.2.0.tgz doesn't seem to exist in /usr/ports/distfiles/ruby.
=> Attempting to fetch from http://rubyforge.rubyuser.de/rubygems/.
rubygems-1.2.0.tgz 100% of 241 kB 138 kBps
===> Extracting for ruby18-gems-1.2.0_1
[...]
If `gem` was installed by a previous RubyGems installation, you may need
to remove it by hand.
===> Registering installation for ruby18-gems-1.2.0_1
freebsd-vm#
</pre>
Now we need to install rails.
<pre>
freebsd-vm# gem install rails
Successfully installed rake-0.8.1
Successfully installed activesupport-2.1.0
Successfully installed activerecord-2.1.0
Successfully installed actionpack-2.1.0
[...]
Installing RDoc documentation for actionpack-2.1.0...
Installing RDoc documentation for actionmailer-2.1.0...
Installing RDoc documentation for activeresource-2.1.0...
freebsd-vm#
</pre>
We need to install postgresql-server.
<pre>
freebsd-vm# make search key=postgres | grep Path: | grep server
Path: /usr/ports/databases/aolserver-nspostgres
Path: /usr/ports/databases/erserver
Path: /usr/ports/databases/postgresql73-server
Path: /usr/ports/databases/postgresql74-server
Path: /usr/ports/databases/postgresql80-server
Path: /usr/ports/databases/postgresql81-server
Path: /usr/ports/databases/postgresql82-server
Path: /usr/ports/databases/postgresql83-server
Path: /usr/ports/finance/tinyerp-server
Path: /usr/ports/net/sipxcommserverlib
Path: /usr/ports/net-im/iserverd
freebsd-vm#
</pre>
Let's install version 8.3. In the options that came up, I selected Build with PAM support and left the other defaults.
<pre>
freebsd-vm# cd databases/postgresql83-server/
freebsd-vm# make install
cd /usr/ports/databases/postgresql83-server && make config;
[...]
For more information, and contact details about the security
status of this software, see the following webpage:
http://www.postgresql.org/
freebsd-vm#
</pre>
We need to set postgres to run at startup.
<pre>
freebsd-vm# echo 'postgresql_enable="YES"' >> /etc/rc.conf
freebsd-vm#
</pre>
Initialize the database.
<pre>
freebsd-vm# /usr/local/etc/rc.d/postgresql initdb
The files belonging to this database system will be owned by user "pgsql".
This user must also own the server process.
The database cluster will be initialized with locale C.
The default text search configuration will be set to "english".
creating directory /usr/local/pgsql/data ... ok
creating subdirectories ... ok
selecting default max_connections ... 40
selecting default shared_buffers/max_fsm_pages ... 28MB/179200
creating configuration files ... ok
creating template1 database in /usr/local/pgsql/data/base/1 ... ok
initializing pg_authid ... ok
initializing dependencies ... ok
creating system views ... ok
loading system objects' descriptions ... ok
creating conversions ... ok
creating dictionaries ... ok
setting privileges on built-in objects ... ok
creating information schema ... ok
vacuuming database template1 ... ok
copying template1 to template0 ... ok
copying template1 to postgres ... ok
WARNING: enabling "trust" authentication for local connections
You can change this by editing pg_hba.conf or using the -A option the
next time you run initdb.
Success. You can now start the database server using:
/usr/local/bin/postgres -D /usr/local/pgsql/data
or
/usr/local/bin/pg_ctl -D /usr/local/pgsql/data -l logfile start
freebsd-vm#
</pre>
Start up the database.
<pre>
freebsd-vm# /usr/local/etc/rc.d/postgresql start
freebsd-vm#
</pre>
Verify that postgres is working.
<pre>
freebsd-vm# createdb test -U pgsql
freebsd-vm# dropdb test -U pgsql
freebsd-vm#
</pre>
There is a postgres GEM. Let's install it.
<pre>
freebsd-vm# gem install postgres
Building native extensions. This could take a while...
Successfully installed postgres-0.7.9.2008.01.28
1 gem installed
Installing ri documentation for postgres-0.7.9.2008.01.28...
Installing RDoc documentation for postgres-0.7.9.2008.01.28...
freebsd-vm#
</pre>
Now we can create a rails application (I switched to a non-root user).
<pre>
freebsd-vm$ rails -d postgresql sample_rails_app
create
create app/controllers
create app/helpers
create app/models
create app/views/layouts
create config/environments
[...]
create log/server.log
create log/production.log
create log/development.log
create log/test.log
freebsd-vm$
</pre>
Setup the postgres user for your application
<pre>
freebsd-vm$ createuser --username pgsql
Enter name of role to add: sample_rails_app
Shall the new role be a superuser? (y/n) n
Shall the new role be allowed to create databases? (y/n) y
Shall the new role be allowed to create more new roles? (y/n) n
freebsd-vm$ cd sample_rails_app/
freebsd-vm$ rake db:create
(in /usr/home/jasonn/sample_rails_app)
freebsd-vm$
</pre>
Create a User scaffold
<pre>
freebsd-vm$ script/generate scaffold User username:string password:string homedir:string ftp_disabled:boolean
exists app/models/
exists app/controllers/
exists app/helpers/
create app/views/users
exists app/views/layouts/
exists test/functional/
exists test/unit/
exists public/stylesheets/
create app/views/users/index.html.erb
create app/views/users/show.html.erb
create app/views/users/new.html.erb
create app/views/users/edit.html.erb
create app/views/layouts/users.html.erb
create public/stylesheets/scaffold.css
create app/controllers/users_controller.rb
create test/functional/users_controller_test.rb
create app/helpers/users_helper.rb
route map.resources :users
dependency model
exists app/models/
exists test/unit/
exists test/fixtures/
create app/models/user.rb
create test/unit/user_test.rb
create test/fixtures/users.yml
create db/migrate
create db/migrate/20080902231538_create_users.rb
freebsd-vm$ rake db:migrate
(in /usr/home/jasonn/sample_rails_app)
== 20080902231538 CreateUsers: migrating ======================================
-- create_table(:users)
NOTICE: CREATE TABLE will create implicit sequence "users_id_seq" for serial column "users.id"
NOTICE: CREATE TABLE / PRIMARY KEY will create implicit index "users_pkey" for table "users"
-> 0.0645s
== 20080902231538 CreateUsers: migrated (0.0665s) =============================
freebsd-vm$
</pre>
Start up the server and visit http://localhost:3000/users (you may need to substitute your IP address)
<pre>
freebsd-vm$ script/server
=> Booting WEBrick...
=> Rails 2.1.0 application started on http://0.0.0.0:3000
=> Ctrl-C to shutdown server; call with --help for options
[...]
</pre>
Create some users, your Rails app is up and running.<div class="blogger-post-footer"><img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/2383541800224195596-2678614208649645686?l=jasonnoble.org'/></div>Jason Noblehttp://www.blogger.com/profile/12779892190227704860noreply@blogger.com0tag:blogger.com,1999:blog-2383541800224195596.post-24015384014199041512008-09-01T22:41:00.003-04:002008-09-01T23:00:28.640-04:00Updating the ports tree in FreeBSDAfter setting up a minimal FreeBSD <a href=http://jasonnoble.org/2008/09/setting-up-freebsd-vm-with-parallels.html>install</a>, I want to update the ports tree to the latest available.
FreeBSD provides a program to do this. It's called cvsup (cvs update). To find the program, I search the ports tree.
<pre>
freebsd# cd /usr/ports
freebsd# make search key=cvsup | grep Path:
Path: /usr/ports/chinese/auto-tw-l10n
Path: /usr/ports/lang/ezm3
Path: /usr/ports/misc/instant-server
Path: /usr/ports/net/csup
Path: /usr/ports/net/cvsup
Path: /usr/ports/net/cvsup-mirror
Path: /usr/ports/net/cvsup-without-gui
Path: /usr/ports/net/cvsupchk
Path: /usr/ports/ports-mgmt/port-authoring-tools
Path: /usr/ports/ports-mgmt/port-maintenance-tools
Path: /usr/ports/sysutils/desktopbsd-tools
Path: /usr/ports/sysutils/fastest_cvsup
Path: /usr/ports/sysutils/maint
freebsd#
</pre>
I want the cvsup without the gui, so I change into that directory and install it.
<pre>
freebsd# cd net/cvsup-without-gui/
freebsd# sudo make install
===> Extracting for cvsup-without-gui-16.1h_4
=> MD5 Checksum OK for cvsup-snap-16.1h.tar.gz.
=> SHA256 Checksum OK for cvsup-snap-16.1h.tar.gz.
===> Patching for cvsup-without-gui-16.1h_4
===> Applying FreeBSD patches for cvsup-without-gui-16.1h_4
[...]
===> Registering installation for cvsup-without-gui-16.1h_4
===> SECURITY REPORT:
This port has installed the following files which may act as network
servers and may therefore pose a remote security risk to the system.
/usr/local/sbin/cvsupd
/usr/local/bin/cvsup
/usr/local/bin/cvpasswd
If there are vulnerabilities in these programs there may be a security
risk to the system. FreeBSD makes no guarantee about the security of
ports included in the Ports Collection. Please type 'make deinstall'
to deinstall the port if this is a concern.
For more information, and contact details about the security
status of this software, see the following webpage:
http://www.cvsup.org/
freebsd#
</pre>
I next need a config file to make cvsup update the ports tree.
<pre>
#vi /root/ports-supfile
# IMPORTANT: Change the next line to use one of the CVSup mirror sites
# listed at http://www.freebsd.org/doc/handbook/mirrors.html.
*default host=cvsup5.us.FreeBSD.org
*default base=/var/db
*default prefix=/usr
*default release=cvs tag=.
*default delete use-rel-suffix
# If you seem to be limited by CPU rather than network or disk bandwidth, try
# commenting out the following line. (Normally, today's CPUs are fast enough
# that you want to run compression.)
*default compress
## Ports Collection.
#
# The easiest way to get the ports tree is to use the "ports-all"
# mega-collection. It includes all of the individual "ports-*"
# collections,
ports-all
</pre>
Now we run rehash to rescan our PATH variable for new commands, then run cvsup to update our ports tree.
<pre>
freebsd# rehash
freebsd# cvsup /root/ports-supfile
Connected to cvsup5.us.FreeBSD.org
Updating collection ports-all/cvs
[...]
Finished successfully
freebsd#
</pre>
Now our ports tree is up to date. If you want to see what ports you've installed that have security updates available, run "/usr/local/sbin/portaudit". It will list any ports that need updating. If you get command not found, cd to /usr/ports/ports-mgmt/portaudit and do a "sudo make install".<div class="blogger-post-footer"><img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/2383541800224195596-2401538401419904151?l=jasonnoble.org'/></div>Jason Noblehttp://www.blogger.com/profile/12779892190227704860noreply@blogger.com0tag:blogger.com,1999:blog-2383541800224195596.post-6434924072312231042008-09-01T18:43:00.012-04:002008-09-01T22:17:25.094-04:00Setting up a FreeBSD VM with Parallels under OS X LeopardTo read this post with screen shots, use one of the following links:
<ul>
<li><A HREF=/long_posts/2008/09/setting-up-freebsd-vm-with-parallels.html>HTML</A></li>
<li><A HREF=/long_posts/2008/09/setting-up-freebsd-vm-with-parallels.pdf>PDF</A></li>
<li><A HREF=/long_posts/2008/09/setting-up-freebsd-vm-with-parallels.zip>PDF Zip</A></li>
</ul>
My goal is to have a FreeBSD 7.0 based virtual machine running under Parallels on my OS X 10.5 Leopard iMac.<P>
Open Parallels and select File -> New<P>
<img src=/images/freebsd_vm_install/freebsd_vm_1.png><P>
As my server is just for local development, I didn't give it a bunch of RAM. You may change this if you have more available RAM. I selected 256MB. Click Next.<P>
This is a new VM, so I want to create a new hard disk image. Click Next.<P>
Again, this is only a development box, so I used only 5GB for a hard drive. I would leave it as Expanding unless you want to pre-allocate the space on your mac. Click Next.<P>
The next window gives networking options. If you need other machines on your local network to access this machine, select Bridged Ethernet, otherwise use Shared Networking. Click Next.<P>
You can name the virtual machine whatever you want. You can uncheck the Create icon on your Desktop if you want. I would not share this virtual machine with other users. Click Next.<P>
This next screen is a user preference. Do you want your mac to give preference to OS X applications or the FreeBSD vm's applications? On a desktop, I think it's fine to leave it with the recommended option. Click Next.<P>
You can choose how to install the OS. If you have a physical disk, choose Real CD/DVD and insert it into your Mac's disk drive. I downloaded the FreeBSD ISOs so I chose ISO image (use disc1). Click Finish.<P>
Your vm should start booting. <P>
Once the installation starts, you need to select your country. Use the arrow keys (up and down) to select your contry, then hit Enter.<P>
Arrow down to Custom and hit Enter<P>
Arrow down to Partition and hit Enter to go into fdisk.<P>
Hit F to dedicate the entire (virtual) disk to FreeBSD. When the prompt comes up, hit Enter.<P>
Hit Q to finish.<P>
Leave the BootMgr default and hit Enter.<P>
Arrow down to Label and hit Enter.<P>
You will get a screen about creating BSD partitions. <P>
Hit C to create a partition. Type in 256M for the size. Hit Enter. <P>
In the partition type window, arrow down until Swap is highlighted. Hit Enter.<P>
Hit C to create a partition. Leave the default size entered and hit Enter. <P>
Leave FS (A file system) highlighted and hit Enter.<P>
For the mount point, type in / and hit Enter.<P>
Your Disklabel screen should now look like the following (assuming a 5GB disk drive)<P>
Hit Q to finish.<P>
Arrow down to Distributions and hit Enter.<P>
Arrow down until Minimal is highlighted. Hit space to select this option.<P>
Arrow down to Custom and hit space to select it.<P>
In this menu you can select the packages you want installed. For a base level system, I install base, kernels, man and catman. I also install the ports option so I can install additional software from the FreeBSD ports collection. Arrow to each option you want to install and hit <SPACE> to select/deselect it. Hit <TAB> to OK and hit Enter to continue.<P>
Arrow up to Exit and hit Enter to continue.<P>
(Optional) If you need to change the installation media (default is CD/DVD), highlight Media and hit Enter.<P>
(Optional) Select the installation media you would like to use and hit Enter to continue.<P>
Arrow down to Commit and hit Enter.<P>
At the Last Chance! screen hit Enter to start the installation.<P>
Go grab a coffee and take a break.<P>
After you get back, you should see a window asking if you want to further configure the box. Hit <TAB> to highlight Yes and hit Enter.<P>
You probably want to set a root password (blank by default). Arrow down to Root Password and hit Enter.<P>
Type in a good password when prompted. It will ask you to retype the password to make sure you didn't misstype the first time.<P>
(Optional) You may want to make a non-root user. Arrow down to User Management and hit Enter.<P>
(Optional) Arrow down to User and hit Enter.<P>
Fill out the fields for your user (<TAB> between fields). Highlight Ok and hit Enter to continue.<P>
The other options are mostly self explanatory. You probably want to set your Time Zone.<P>
You also need to setup your networking. Arrow down to Networking and hit Enter.<P>
Arrow down to Interfaces and hit Enter.<P>
Select the Novell NE1000/2000 highlighted and hit Enter.<P>
Do you need IPv6? Probably not, leave No highlighted and hit Enter.<P>
You should enable DHCP, as that is how Parallels gives it an IP address. Hit <TAB> to highlight Yes and hit Enter.<P>
You should give your vm a hostname. The IPv4 Gateway and IP info should be auto filled in.<P>
Hit <TAB> until Ok is highlighted and hit Enter to continue.<P>
You probably want SSH access, so hit arrow down to highlight ssh and hit Enter.<P>
You may want to enable the ntpdate service. It will keep your system clock synced. Highlight Ntpdate and hit Enter.<P>
Select a server to synchronize with. Pick one that is "close" to you. Hit Enter to continue.<P>
When you're done selecting services, hit <TAB> until Ok is selected and hit Enter.<P>
Arrow up to Exit and hit Enter.<P><div class="blogger-post-footer"><img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/2383541800224195596-643492407231223104?l=jasonnoble.org'/></div>Jason Noblehttp://www.blogger.com/profile/12779892190227704860noreply@blogger.com0tag:blogger.com,1999:blog-2383541800224195596.post-58467439453633986862008-08-21T10:49:00.008-04:002008-08-21T11:25:25.275-04:00Show mounted filesystems on a NetappWe're working on moving from an old Netapp to a new Netapp. Before doing so, we need to make sure that no systems have any mount points on the old filer still mounted.<P>
On a regular/unix nfs server, you would login to the box and run "showmount -a" which queries the mount daemon. The netapp systems don't offer the showmount command though.<P>
I found a mailing list <a href=http://toasters.mathworks.com/toasters/9199.html>article</a> that explained you can run "showmount -a XYZ" where XYZ is the hostname/ip of your netapp filer.<P>
<pre>
roadrunner:~ # showmount -a 172.16.2.6
All mount points on 172.16.2.6:
10.1.112.17:/vol/root
172.16.13.16:/vol/space/offsite.backup
172.16.13.18:/vol/space/offsite.backup
172.16.13.5:/vol/space/db.preprod
172.16.13.5:/vol/space/offsite.backup
172.16.13.7:/vol/space/offsite.backup
172.16.23.1:/vol/space/endeca.preprod
172.16.23.6:/vol/space/db.preprod
172.16.23.6:/vol/space/offsite.backup
172.16.3.6:/vol/root
mailhost:/vol/space/imagemgr
roadrunner:~ #
</pre>
Now I know what hosts I need to check before deprovisioning this Netapp.<P>
Update: From the <a href=http://toasters.mathworks.com/toasters/9199.html>article</a>, I missed the caveat. NFS is stateless, so some unix hosts could still have the system mounted, but not actively use the mount.<P>
As suggested, I turned the nfs.per_client_stats.enable bit on, and now I'll just wait and see who is actively using the Netapp.<P>
<pre>
netapp> options nfs
[...]
nfs.netgroup.strict off
nfs.per_client_stats.enable off
nfs.require_valid_mapped_uid off
[...]
netapp> options nfs.per_client_stats.enable on
netapp> nfsstat -l
172.16.2.2 <hostname unknown> NFSOPS = 6 ( 0%)
172.16.3.2 mailhost NFSOPS = 8 ( 0%)
netapp>
</pre><div class="blogger-post-footer"><img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/2383541800224195596-5846743945363398686?l=jasonnoble.org'/></div>Jason Noblehttp://www.blogger.com/profile/12779892190227704860noreply@blogger.com0tag:blogger.com,1999:blog-2383541800224195596.post-26295121467580250202008-08-19T10:02:00.003-04:002008-09-01T23:07:43.729-04:00DNS ResolverI had a list of domains that needed their MX records updated. I first wanted to make sure that our DNS servers were authoritative for those zones.
I used Perl and the Net::DNS CPAN module to figure out the NS records for each domain.
<pre>
#!/usr/bin/perl
use Net::DNS;
my $resolver = Net::DNS::Resolver->new;
my $query = "";
my $domain = "";
open(FILE, "</tmp/domains.txt");
while(<FILE>) {
chomp;
$domain = $_;
$query = $resolver->query($domain, "NS");
print $domain;
if($query) {
foreach my $rr ($query->answer) {
next unless $rr->type eq "NS";
print ",", $rr->nsdname;
}
print "\n";
} else {
print ",DOES NOT RESOLVE\n";
}
}
</pre><div class="blogger-post-footer"><img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/2383541800224195596-2629512146758025020?l=jasonnoble.org'/></div>Jason Noblehttp://www.blogger.com/profile/12779892190227704860noreply@blogger.com0tag:blogger.com,1999:blog-2383541800224195596.post-41050251153055728362008-08-13T12:03:00.006-04:002008-08-15T16:21:48.982-04:00Tcl version of ucfirstThe load balancers we use at work have a TCL command line interface. A co-worker needed a TCL function written that would take a string as input and return the string transformed so that the first letter was upper case and the rest were lower case. There is a built in string function called toupper but that capitalizes the full string instead of only the first.<p>
My solution (shown below) is to split the string into two strings, toupper the first string and tolower the second, then append them together. The result is the equivalent of Perl's ucfirst implemented in TCL.<P>
<pre>
proc ucfirst str {
set a [ string toupper [ string range $str 0 0 ]]
append a [ string tolower [ string range $str 1 end ]]
return $a
}
</pre>
You can use this from the command line by using tclsh. Below is what running this code will look like (the % is the tclsh prompt).<P>
<pre>
[jasonn@jnoble-vm ~]$ tclsh
% proc ucfirst str {
set a [ string toupper [ string range $str 0 0 ]]
append a [ string tolower [ string range $str 1 end ]]
return $a
}
%
% set mystring "firetruck"
firetruck
% set t [ ucfirst $mystring ]
Firetruck
% set mystring "cop car"
cop car
% set t [ ucfirst $mystring ]
Cop car
% exit
[jasonn@jnoble-vm ~]$
</pre><div class="blogger-post-footer"><img width='1' height='1' src='https://blogger.googleusercontent.com/tracker/2383541800224195596-4105025115305572836?l=jasonnoble.org'/></div>Jason Noblehttp://www.blogger.com/profile/12779892190227704860noreply@blogger.com0