Creating a Linux File Server for Windows CIFS/SMB, NFS, etc.

Share on:

Recently I needed to build a multipurpose file server to host CIFS and NFS shares — CIFS for the Windows users, and NFS for VMWare to store ISOs. It needed to utilize back end storage (NetApp via iSCSI), provide Windows ACLs for the CIFS shares, and be able to authenticate against two different Active Directory domains. After careful consideration, I decided to use Red Hat Enterprise Linux 6.5 (RHEL) instead of Windows Server 2012.

Now you might be wondering, “Why on earth would you want to build a Linux file server to do all that when you can just use Windows?” There are a few reasons:

  1. Resources – Linux has a much, much smaller footprint than even a Server Core install of Windows Server 2012. If you’re building a file server for a branch office with limited server resources, this can be a big deal.

  2. Security – Yes, Red Hat Linux really is more secure out-of-the-box than Windows, plus it’s less of a target — at least for now.

  3. Cost… maybe – The cost of Linux vs. Windows is a funny thing. Obviously, Linux is free and Windows isn’t. But if you purchase enterprise Linux support from, say, Red Hat, the cost comes out nominally less for Linux. On the other hand, if your organization doesn’t require enterprise-level support on every server, building redundant Linux file servers is free. The cost is ultimately dependent upon the level of support needed.

In this post I’m going to show you how I built a fully-functional, Active Directory-friendly Linux file server in less than half-a-day.

The first step, of course, is to install Linux. I created a VMWare virtual machine with 4GB RAM and 1 vCPU. Having decided to go with RHEL 6.5, I first created a Kickstart script with the following packages:

@base

@console-internet

@network-tools

@core

@directory-client

@internet-browser

@large-systems

@network-file-system-client

@performance

@server-platform

(Kickstart is the name of Red Hat’s automated installation method. If you don’t want to use Kickstart, you can just install RHEL 6.5 the old fashioned way and select the above package groups when prompted)

I placed the Kickstart file and the Red Hat installation files on an FTP server, booted the RHEL ISO, and pointed the bootloader to the Kickstart file as shown here:

kickstart screen for linux file server

After the installation is complete, you’ll need to ensure yum, Red Hat’s package installer, knows where to find the Red Hat installation files as you be installing more things shortly. Log in as root and edit the file /etc/yum.repos.d/rhel-source.repo. Add the text enclosed in the yellow rectangle below, changing the path as necessary to suit your environment, then save it.

adding rhel ftp to /etc/yum.repos.d/rhel-source.repo

In the example above, I’m logging into the FTP server at 192.168.1.11 with the username and password “redhat”. Alternatively, you can just mount the Red Hat ISO into /media with the command

1mount -o loop /dev/cdrom /media
bash

and point the baseurl path to file:///media.

Now we’re going to install Samba4, Winbind, Kerberos, NTP, and NFS.

1yum -y install samba4 samba4-client samba4-winbind krb5-workstation ntp nfs
bash

Configure your NTP servers (typically your AD domain controllers) in /etc/ntp.conf, start the NTP daemon, and set it to start at bootup.

1chkconfig ntpd
2service ntpd start
bash

Next, since we’ll be dealing with the ever-picky Active Directory, we’ll need to ensure DNS is configured properly. Modify /etc/resolv.conf to add your DNS servers and any DNS suffixes you want Linux to search when it tries to look up a non-fully qualified domain name (For example, if I try to ping webserver01, I want Linux to perform a DNS lookup for webserv01.benpiper.com, so I’d add the suffix benpiper.com to the search space.)

Now configure /etc/hosts and put this server’s fully-qualified domain name (FQDN) right after 127.0.0.1.

Next we need to modify /etc/samba/smb.conf. Under the [config] section we need to add or change the following lines:

 1workgroup = BENPIPER
 2server string = FILE SERVER SAMBA V %v
 3netbios name = DCASFIS01
 4security = ADS
 5realm = BENPIPER.COM
 6encrypt passwords = yes
 7
 8winbind trusted domains only = no
 9winbind use default domain = yes
10winbind enum users  = yes
11winbind enum groups = yes
12
13idmap uid = 15000-20000
14idmap gid = 15000-20000
15
16vfs objects = acl_xattr
17map acl inherit = Yes
18nt acl support = yes
19store dos attributes = Yes
...
bash

A little explanation is in order. The workgroup directive is the NetBIOS name of the domain the server is going to join.

netbios name is the NetBIOS name of the server that’s going to be joining the domain.

realm is the Kerberos realm which we are going to specify when we configure Kerberos. A common misconception is that the Kereros realm is just the FQDN of the domain. Even though we’ll be using the FQDN for clarity, this could actually be any value we want, as long as it matches what we specify later on /etc/krb5.conf.

nt acl support = yes tells Samba to store NTFS Access Control Lists (ACLs) as Linux File Access Control Lists.

Join the domain. Be sure to replace <em>domainadmin</em> with an actual domain adminsitrator.

1net ads join -U domainadmin
bash

Samba’s name service switch module sits in the 32-bit /usr/local.samba/lib directory by default. Samba running on our 64-bit system is not going to look there, so we’ll need to create a symbolic link to the /lib64 directory.

1ln -s /usr/local/samba/lib/libnss_winbind.so /lib64
2ldconfig
bash

Add the following to /etc/nsswitch.conf

1passwd: compat winbind
2group: compat winbind
bash

Modify /etc/krb5.conf as follows.

 1[logging]
 2default = FILE:/var/log/krb5libs.log
 3kdc = FILE:/var/log/krb5kdc.log
 4admin_server = FILE:/var/log/kadmind.log
 5
 6[libdefaults]
 7default_realm = BENPIPER.COM
 8dns_lookup_realm = true
 9dns_lookup_kdc = true
10ticket_lifetime = 24h
11renew_lifetime = 7d
12forwardable = true
13
14[realms]
15BENPIPER.COM = {
16}
17
18SECONDARYDOMAIN.COM = {
19}
20
21[domain_realm]
22.benpiper.com = BENPIPER.COM
23benpiper.com = BENPIPER.COM
24.secondarydomain.com = SECONDARYDOMAIN.COM
25secondarydomain.com = SECONDARYDOMAIN.COM
...
bash

Start winbind, smb, and nmb.

1service winbind start
2service smb start
3service nmb start
4chkconfig winbind on
5chkconfig smb on
6chkconfig nmb on
bash

Now we’re ready to test whether CIFS sharing actually works. Create a test folder to share via CIFS.

1mkdir /var/lib/samba/testshare
bash

Grant rwx permissions to the directory owner and group.

1chmod 770 /var/lib/samba/testshare/
bash

Give ownership to user “root” and Active Directory group BENPIPER\file server admins. This will ensure both “root” and members of the “file server admins” group can modify ACLs. Note that

1chown root:"BENPIPER\file server admins" /var/lib/samba/testshare/
bash

Modify the ACL to grant the BENPIPER\file server admins AD group read, write, and execute access to the new folder.

1setfacl -m g:"BENPIPER\file server admins":rwx /var/lib/samba/testshare/
2setfacl -m g::--- /var/lib/samba/testshare
bash

Now try to access the share from a Windows machine. If you are a member of the BENPIPER\file server admins group, you should be able to modify ACLs directly from Windows Explorer.

If you’re not going to be using shared storage, you can skip this section. Here we’re going to configure an iSCSI connection to our NetApp SAN. We’ve already created a 200 GB LUN and made it available to the iQN initiator name that we’ll set in just a moment.

First, install the iSCSI initiator utilities.

1yum -y install iscsi-initiator-utils
bash

Change the initiator name in /etc/iscsi/initiatorname.iscsi to whatever you want. I usually use the hostname for the storage identifier (for example, iqn.1994-05.com.benpiper:dcasfis01)

If you don’t want iSCSI traffic to ride over a separate NIC, skip this step. Assuming iSCSI traffic will use eth1, modify /var/lib/iscsi/ifaces/iface.eth1 as follows:

1iface.transport_name = tcp
2iface.iscsi_ifacename = eth1
bash

Run discovery against the NetApp SAN with IP 192.168.59.31 and login.

1iscsiadm -m discovery -t st -p 192.168.59.31
2iscsiadm -m node -l
bash

Now verify. You should see the IP addresses, ports, and iQN of the SAN.

1# iscsiadm -m node
2192.168.59.31:3260,2001 iqn.1992-08.com.netapp:sn.437593081
3192.168.60.31:3260,2001 iqn.1992-08.com.netapp:sn.437593081
bash

Rescan the session to pick up the LUNs.

1iscsiadm -m node -R
bash

Verify the new storage is there. Note the name of the disk (/dev/sdb).

1# fdisk -l
2...
3Disk /dev/sdb: 214.7 GB, 214748364800 bytes
4255 heads, 63 sectors/track, 26108 cylinders
5Units = cylinders of 16065 * 512 = 8225280 bytes
6Sector size (logical/physical): 512 bytes / 512 bytes
7I/O size (minimum/optimal): 4096 bytes / 65536 bytes
8Disk identifier: 0x00000000
9...
bash

We’re going to use the Linux Logical Volume Manager (LVM) to partition the block storage. The advantage of using LVM over using fdisk to create MBR partitions is that resizing LVM volumes is much easier. If you don’t know your storage requirements up front, you can create relatively small logical volumes and leave the rest of the disk unallocated. When you have a better idea of what size the volumes should be (like when you run out of space), you can non-destructively resize them.

Creating an initial LVM volume requires a few steps. We’ll still have to use fdisk to create a primary partition, but instead of creating it as ext4 partition type, we’ll select Linux LVM (type 8e). Then we’ll create a physical LVM volume on the partition, followed by a logical volume group, and finally a logical volume.

Create a primary partition for the Linux Logical Volume Manager (LVM).

 1# fdisk /dev/sdb
 2
 3WARNING: DOS-compatible mode is deprecated. It`s strongly recommended to
 4switch off the mode (command 'c') and change display units to
 5sectors (command 'u').
 6
 7Command (m for help): n
 8Command action
 9e extended
10p primary partition (1-4)
11p
12Partition number (1-4): 1
13First cylinder (1-26108, default 1):
14Using default value 1
15Last cylinder, +cylinders or +size{K,M,G} (1-26108, default 26108):
16Using default value 26108
17
18Command (m for help): t
19Selected partition 1
20Hex code (type L to list codes): 8e
21Changed system type of partition 1 to 8e (Linux LVM)
22
23Command (m for help): w
24The partition table has been altered!
25
26Calling ioctl() to re-read partition table.
27Syncing disks.
...
bash

Create the LVM physical volume on /dev/sdb1

1# pvcreate /dev/sdb1
2Physical volume "/dev/sdb1" successfully created
bash

Create a volume group called volgroup1

1# vgcreate volgroup1 /dev/sdb1
2Volume group "volgroup1" successfully created
bash

Now we’ll create a small 40GB volume for ISO storage. LVM sizes volumes based on the number of extents. When we created the logical volume group we didn’t select an extent size, so it used the default which is 4MB. The size of the logical volume must be an integer multiple of the extent size. Since we’re creating a 40GB volume and the extent size is 4MB, we’ll specify 10240 extents (40960MB / 4MB).

1# lvcreate -l 10240 -n ISO volgroup1
2Logical volume "ISO" created
bash

Verify the new volume information is correct. Note the path of the new volume as this is what we’re going to be formatting.

 1# lvdisplay
 2--- Logical volume ---
 3LV Path /dev/volgroup1/ISO
 4LV Name ISO
 5VG Name volgroup1
 6LV UUID apui74-26k5-AcWK-VtaF-UTbY-DuKp-QsCXf1
 7LV Write Access read/write
 8LV Creation host, time dcasfis01.benpiper.com, 2014-07-07 00:00:00 -0000
 9LV Status available
10# open 0
11LV Size 40.00 GiB
12Current LE 10240
13Segments 1
14Allocation inherit
15Read ahead sectors auto
16- currently set to 256
17Block device 253:4
...
bash

Create an ext3 filesystem.

 1# mkfs.ext3 /dev/volgroup1/ISO
 2mke2fs 1.41.12 (17-May-2010)
 3Discarding device blocks: done
 4Filesystem label=
 5OS type: Linux
 6Block size=4096 (log=2)
 7Fragment size=4096 (log=2)
 8Stride=1 blocks, Stripe width=16 blocks
 92621440 inodes, 10485760 blocks
10524288 blocks (5.00%) reserved for the super user
11First data block=0
12Maximum filesystem blocks=4294967296
13320 block groups
1432768 blocks per group, 32768 fragments per group
158192 inodes per group
16Superblock backups stored on blocks:
1732768, 98304, 163840, 229376, 294912, 819200, 884736, 1605632, 2654208,
184096000, 7962624
19
20Writing inode tables: done
21Creating journal (32768 blocks): done
22Writing superblocks and filesystem accounting information: done
23
24This filesystem will be automatically checked every 33 mounts or
25180 days, whichever comes first. Use tune2fs -c or -i to override.
...
bash

By default, Samba looks for shares in the /var/lib/samba folder, so we’ll create mount point for for our new ISO volume and then mount it there.

1mkdir /var/lib/samba/iso
2mount /dev/volgroup1/ISO /var/lib/samba/iso
bash

In order to ensure that the volume is mounted when the system reboots, we need to add the volume’s UUID to /etc/fstab.

1# blkid /dev/volgroup1/ISO
2/dev/volgroup1/ISO: UUID="a8c77eb9-422a-4d8c-bec6-46cc811dc535" SEC_TYPE="ext2" TYPE="ext3"
bash

Add the above UUID to /etc/fstab:

1UUID=a8c77eb9-422a-4d8c-bec6-46cc811dc535 /var/lib/samba/iso ext3 _netdev,acl 0 0
bash

The _netdev directive indicates that this is an iSCSI-bound volume, and the acl directive indicates that the filesystem should be mounted with ACL support.

Now we’re ready to create a CIFS share for the new volume. Edit /etc/samba/smb.conf as follows:

1[iso]
2comment = ISO share
3browseable = yes
4writable = yes
5path = /var/lib/samba/iso
6inherit acls = yes
7inherit permissions = yes
bash

Reload the Samba config to apply the changes.

1smbcontrol smbd reload-config
bash

Set the permissions on the new share to make root the user owner and BENPIPER\file server admins the group owner. Add BENPIPER\file server admins to the ACL and grant rwx access.

1chmod 770 /var/lib/samba/iso/
2chown root:"BENPIPER\file server admins" /var/lib/samba/iso/
3setfacl -m g:"BENPIPER\file server admins":rwx /var/lib/samba/iso/
4setfacl -m g::--- /var/lib/samba/iso/
bash

Now it’s time to test. Go to a Windows machine and browse to the new share.

Testing Linux CIFS file server ISO share

It works! But before we get too excited, let’s reboot. When the system comes back up, verify that the volume is still mounted in the right location.

1# df -h
2Filesystem                     Size  Used Avail Use% Mounted on
3/dev/mapper/vg_dcasfis01-root   18G  1.5G   15G  10% /
4tmpfs                          1.9G     0  1.9G   0% /dev/shm
5/dev/sda1                      485M   39M  421M   9% /boot
6/dev/mapper/vg_dcasfis01-home  9.9G  151M  9.2G   2% /home
7/dev/mapper/vg_dcasfis01-var   7.9G  223M  7.3G   3% /var
8/dev/mapper/volgroup1-ISO       40G  177M   38G   1% /var/lib/samba/iso
bash

Looks good. Add some ISOs to the share for VMWare to use and let’s start configuring NFS.

Modify /etc/exports to share the volume as read-only.

1/var/lib/samba/iso *(ro,sync)
bash

Start NFS and configure it to start at boot.

1service nfs start
2chkconfig nfs on
bash

Grant others read-only access so VMWare can see the ISOs.

1chmod o+r /var/lib/samba/iso
bash

Now mount the NFS volume in VMWare as read-only. The path is going to be /var/lib/samba/iso. Open the datastore browser and verify you can see the ISOs.

Read-only NFS volume mounted in VMWare

We’re done! If you’re feeling adventurous you can even install Apache or another webserver and give access to these ISOs via HTTP. This is perfect for providing images to OpenStack’s Glance or CloudStack’s Secondary Storage.

I hope you enjoyed this little tutorial. If you have any questions or comments please feel free to leave them below.