Wednesday, April 28, 2010

Using x3270if commands against s3270 without calling a script

Recently I needed to start playing around with automating SLES installs on IBM z Series mainframe guests.  I needed to be able to have a script be able to send commands to the guest thorugh a 3270 terminal.  I found s3270 which starts a 3270 terminal that can then be passed commands via a script to be ran against the guest.  This was good but I ran into a limitation that I had to open s3270 and then tell it to run a script that contained what I wanted to due.  This was limiting because I wanted the script to be more dynamic in asking for username/passwords/ips/etc and monitoring output with expect to work through different scenarios.

This initially led me to develop multiple shell and expect scripts that called eachother passing arguments between eachother that was received from a 'master' script.  This is very ugly though codewise and also that you could see entered passwords via 'ps' because I was passing it as an argument to other scripts.

To get around my issues I needed to:
  • Use 1 script
  • Have script start s3270 and be able to pass commands to it (not have s3270 call a script)
  • have expect functionality
First I figured out how to pass commands to s3270 instead of having s3270 call a script with a list of commands.  This was done be creating 2 FIFOs and redirecting stdin and stdout to those FIFOs when starting s3270:
mkfifo /tmp/3270.in /tmp/3270.out
s3270 /tmp/3270.out &
I also backgorunded the s3270 process because otherwise you are stuck in s3270 to enter commands for it.

I then created filedescripors that copied my FIFOs:
exec 5>/tmp/3270.in 6
rm -f /tmp/3270.in /tmp/3270.out
Now I set 2 enviornment variables that x3270if uses to know what filedescriptors to use when communicating with an s3270 process:
X3270INPUT=5
X3270OUTPUT=6
export X3270INPUT X3270OUTPUT
That's it for goal #1!  I can now do commands such as the following:
x3270if "connect(192.168.1.100)"
Which would instruct my s3270 terminal to connect to a frame on IP 192.168.1.100.

With this functionality alone I can then use expect scripting alone to do everything I want which meets my other goals.  However, I don't like Tcl expect so I chose to use python scripting using pexpect for expect functionality to script everything.

I found the redirect solution I detailed above after scouring google and finding the following link to a script that someone else made @ http://www.sfr-fresh.com/unix/misc/x3270-3.3.9ga12.tgz:a/x3270-3.3/Examples/cms_logon.sh

Thursday, March 18, 2010

access files below a mountpoint

Today I needed to be able to access a directory that is normally used as a mountpoint. I needed to do this because we wanted to transition some automounts to hardmounts without interruption.

So an example is we have auto mount table /test specfied in /etc/auto.master which automounts /test/1 and /test/2 automatically.

Say someone was on a system using /test/1 actively. I would want to 'fix' the system by setting up new entries in /etc/fstab for /test/1 and /test/2 and remove /test from /etc/auto.master. Then I would want to restart automount.

/test/1 would still be mounted since its in use which is good as then there is no interruption to the user. I would create /test/2 and mount a 'hard' mount on that. Since automount was restarted and no longer has an entry for /test in /etc/auto.master then /test/1 will never timeout from automount anymore.

However, the base mountpoint that automount was using is /test ... so on a reboot /test will still exist but directories /test/1 and /test/2 will no longer exist and the system will fail to mount on those /etc/fstab entries ...

What I ended up doing was:
  • mount --bind / /mnt
  • cd /mnt/test
This allows me to be 'underneath' the /test mountpoint which currently has /test/1 mounted from automount earlier.
  • mkdir 1 2
  • cd /
  • umount /mnt
Now if the system reboots /test/1 and /test/2 will exist and the mounts specified in /etc/fstab will mount correctly.

Saturday, March 13, 2010

mkinitrd: device /dev/ does not have a driver

I recently ran into an issue when trying to update the kernel on my openSUSE 11.0 system. When installing the kernel and mkinitrd ran during the kernel rpm post-scriptlet mkinitrd would fail. When I ran it on my own I would get:
Fatal storage error. device /dev/ does not have a driver
I dove into the code for mkinitrd and found 2 spots where it was getting hung up on having '/dev/' as a value in a variable called blockdev.

One was in /lib/mkinitrd/scripts/setup-block.sh and another in /lib/mkinitrd/scripts/setup-iscsi.sh.

The variable blockdev is a space delimited list of devices that these scripts loop through and execute commands against with for loops. The for loops looks like:
for bd in $blockdev; do
So each value in blockdev then get assigned temporarily to bd. I therefore added a further check right after the for bd in $blockdev; do line to check if variable bd equals exactly /dev/ and if so to skip that value:
if [ "$bd" == "/dev/" ]; then
verbose "[BLOCK] ignoring $bd (gaf)"
continue
fi
So before:
...
for bd in $blockdev; do
...
And after:
...
for bd in $blockdev; do
if [ "$bd" == "/dev/" ]; then
verbose "[BLOCK] ignoring $bd (gaf)"
continue
fi
...
I'm guessing /dev/ is being included into the blockdev variable erroneously, but I'm not gonna waste anymore time figuring that part out. :P

Hope this helps as I didn't see anyone with a solution to this issue when searching for one myself (to save time ...).

NOTE: This issue seems to be addressed in 11.2 after some update (still an issue in base 11.2). I ran into it while upgrading to 11.2 from 11.0.

Tuesday, February 9, 2010

install ubuntu 9.10 karmic on lvm

Recently I upgraded my Ubuntu install on my netbook to the latest version of 9.10 karmic. I wasn't too happy with the leftovers and some applet configurations that were missing after upgrading so I decided to just reinstall directly to 9.10. I thought this would also give me a chance to redo my filesystem layouts to better utilize my 4GB SSD and 12G SSD in my netbook.

Things quickly turned ugly as I found out that the 'advanced partitioning' provided by ubuntu is no where near what I am used to when installing RHEL or SLES at work. I soon found out to install to LVM you first need to setup your LVM configuration and mkfs your volumes BEFORE you start the ubuntu install. After everything is setup it will see them and you can then specify what you want to mount where and the install will work as you want it.

Of course there are some workarounds to doing all this ... here is the process I used.

  1. Get your Ubuntu 9.10 install/live cd
  2. If you do not have network for the system your installing to (wired or wireless) then download the following to a USB drive (these are 32-bit versions since I have a netbook, you may need to find 64-bit versions for your install)
  3. Boot your system using the Ubuntu 9.10 install/live cd
  4. Once booted open a terminal and get root
    • sudo su -
  5. For systems with no network:
    • Insert your USB stick with the packages you downloaded above
    • cd to /media/[USBdriveName]/[directoryWithPackages]
    • Install lvm2 and watershed to the live cd OS
      • dpkg -i lvm2*.dev watershed*.deb
  6. For systems with network:
    • apt-get update
    • apt-get install lvm2
Now that lvm2 is installed you can create your PVs, VGs, LVs. You can do this however you want. The following is what I did as a reference for people who may not know much about LVM. My setup including striping between 2 disks so you may want to adjust how you do things ...

  1. fdisk /dev/sda
    • o
    • w
    • n
    • p
    • 1
    • 1
    • +100M
    • n
    • p
    • 2


    • t
    • 2
    • 8e
    • w
  2. fdisk /dev/sdb
    • o
    • w
  3. This is where I had some issues. I had some type of lock on /dev/sdb when trying to do the next step of pvcreate to pull it into LVM ... so I needed to reboot to release it. If you need to reboot to use your disk you will need to follow steps 1-6 from the first section to get a live boot enviornment with LVM running again.
  4. pvcreate /dev/sda2
  5. pvcreate /dev/sdb
  6. mkfs.ext2 /dev/sda1
  7. vgcreate rootdisk /dev/sda2 /dev/sdb
  8. lvcreate -i2 -I4 -L6G -nroot_lv rootdisk
  9. lvcreate -l100%PVS -nvar_lv rootdisk /dev/sda2
  10. lvcreate -l100%PVS -nhome_lv rootdisk /dev/sdb
  11. mkfs.ext4 /dev/rootdisk/root_lv
  12. mkfs.ext4 /dev/rootdisk/var_lv
  13. mkfs.ext4 /dev/rootdisk/home_lv
We now have LVM setup to host /, /var and /home filesystems. /boot will exist directly on /dev/sda1 (I later found out I could have tried to include it in LVM because 9.10 uses grub2, but too late now :( ) . So now you can initiate the install.
  1. Start install
  2. When you get to partition/disk setup section choose advanced
  3. You will see the /dev/mapper/[vg]-[lv] devices you created in LVM. Below them will be the filesystem you put on each. Highlight the filesystem one and click the "Change..." button to specify the filesystem to use (should match how you mkfs above) and where to mount. DO NOT HAVE THE INSTALL REFORMAT THE FILESYSTEM.
  4. For /dev/sda1 highlight and click "Change...". Specify the filesystem, where to mount (/boot) and DO HAVE IT FORMAT IT!
  5. Proceed through the rest of the questions to start the install
One the install is finished WE ARE NOT DONE YET. The install installed it to where we want it, but it did NOT install LVM to our new OS! So now we need to mount everything by hand and chroot to our new OS to install LVM and then remake the initramfs. (If you reboot at this point it will not boot stating it can not find your /dev/mapper/[rootVG]-[rootLV]

These instructions follow how I setup my system, you need to make adjustments for the VG/LV names you created.
  1. mount /dev/rootdisk/root_lv /mnt
  2. mount /dev/rootdisk/var_lv /mnt/var
  3. mount /dev/rootdisk/home_lv /mnt/home
  4. mount /dev/sda1 /mnt/boot
  5. mount -t proc proc /mnt/proc
  6. mount -t sysfs sys /mnt/sys
  7. mount -o bind /dev /mnt/dev
  8. IF YOU DO NOT HAVE NETWORK:
    • Insert your USB stick with the packages you downloaded above
    • cd to /media/[USBdriveName]/[directoryWithPackages]
    • cp lvm2*.dev watershed*.deb libreadline5*.deb /mnt/var/tmp
    • chroot /mnt
    • cd /var/tmp
    • dpkg -i lvm2*.dev watershed*.deb libreadline5*.deb
  9. IF YOU HAVE NETWORK:
    • chroot /mnt
    • apt-get update
    • apt-get install lvm2
  10. mkinitramfs -o /boot/[filenameOfCurrentInitramfs]
  11. reboot
That should be it. I made these instructions all after-the-fact, so its all from memory trying to walkthrough what I did. I hope I didn't miss anything, but this should set you on the right track regardless. Below are links I referenced while trying to figure this out that may be helpful as well: