An introduction to simple CVS commands
            ======================================
                by gfoot@users.sourceforge.net
                 revision 2, updated May 2001

I wrote this file as a quick walkthrough of the most basic and
most useful CVS commands, as far as I have experienced them.
Please let me know if I've left out something important (because
that probably means I don't know it yet, and I'd like to!), or
made any mistakes.


1. Why you might want to use CVS
--------------------------------
The version of the library in CVS will contain any
between-release changes made to the library.  It's very helpful
if you can test these changes before we make a new release,
because it's nice to have the released versions stable.  That's
not to say that the CVS version is necessarily unstable, but it
might contain very new code.

Also, using CVS to keep up to date, even if you don't go for
between-release versions, is also simpler than downloading new
versions in .zip or .tar.gz format.  Once you've got your first
CVS version of the library, you just issue one short command to
check for changes and bring it up to date.  The changes are
downloaded in patch format, so you don't need to redownload
files that haven't changed.

Similarly, for developers CVS is great, because after making
some changes to the library you can run another short command
and have CVS work out what you changed and merge it with the
repository.

If anything does get broken, you can always download any older
version you like -- you specify the date, and CVS gives you
whatever was in the repository at that time.


2. Configuring CVS
------------------
CVS reads some default command line arguments from .cvsrc in
your home directory.  I suggest putting the following lines:

    cvs -q -z3
    update -d -P
    diff -u

The first line specifies that the global options `-q' (quiet
mode) and `-z3' (compress network traffic at level 3) should be
set.  The second line states that updates should create
directories as required (-d) but also remove empty directories
(-P for prune).  The third line states that diffs should be in
unified mode by default.

If you choose not to put these in your .cvsrc, you'll probably
still want to use them in your command lines.  Note that in CVS
the global options (e.g. `-q') before the command (e.g.
`update') are independent of, and have different meanings to,
the options after the command.  e.g. `-d' before the command
specifies a CVS root (as you'll see below in real command lines)
but `-d' after `update' tells it to create missing directories
in your checkout.

The remainder of this section applies to registered users (with
write access to the repository) only.

For security reasons, SourceForge only allows write access to
CVS through an SSH connection.  This means you need an SSH
client, and you must set the environment variable `CVS_RSH' to
`ssh' or whatever the executable name of your SSH client is.  In
bash:

    export CVS_RSH=ssh

or in DOS's command.com:

    set CVS_RSH=ssh

You probably want to put these in your .bashrc or autoexec.bat
files.

Because each CVS command opens a new ssh connection, you'll be
prompted for your password once per command.  If this bothers
you, you can run ssh-keygen to generate a public/private key
pair.  Accept the default location to save the private key, and
leave the passphrase blank, to avoid having to enter it (the
procedure is kind of pointless in the context of CVS if you use
a passphrase).  You'll be told your public key, and it will be
saved in a file for future reference.  If you go to your member
page on the SourceForge website you can register your public
key(s -- one per machine you log in from) there.  When they are
registered (it may take a few hours) you'll be able to run cvs
commands without needing a password.

3. Doing your first CVS checkout
--------------------------------
This is the most complicated part of the process, since it
involves specifying a CVS server, login method and module.
It will remember this information, so in future you don't have
to specify it over and over again.

There are two separate sets of instructions here.  The first is
for anonymous users, who have read-only access.  The second is
for registered developers, who can write their changes back into
the repository.

3.1 Anonymous logins
~~~~~~~~~~~~~~~~~~~~
First you need to log in -- this is a sort of authentication.
Using Libnet as an example, here is the command:

    cvs -d :pserver:anonymous@cvs.libnet.sourceforge.net:/cvsroot/libnet login

and enter a blank password.  You should get no further messages.
For AllegroGL, replace all occurences of `libnet' with
`allegrogl'.

Next, you can do your checkout.  Again, using Libnet as the
first example:

    cvs -d :pserver:anonymous@cvs.libnet.sourceforge.net:/cvsroot/libnet co libnet

(Incidentally, `co' is short for `checkout'.)  This is the last
time you need the long `-d' parameter.

The last parameter specifies which module to check out -- if you
specify a directory, you get all the files and subdirectories
too.  For AllegroGL, the last parameter should be `alleggl', and
the other occurences of `libnet' should be replaced with
`allegrogl' as before.

After this command you should see the system unpacking all the
files into a subdirectory called `libnet' (or `alleggl').  In
the future, do CVS commands from within that directory, or a
subdirectory, and you won't need the long `-d' parameter.

3.2 Authenticated logins
~~~~~~~~~~~~~~~~~~~~~~~~
(Please read the previous section first, to get a general feel
for the checkout command.)

Registered users don't need to log in in advance, as in the
anonymous case.  Just issue a checkout command with the
appropriate `-d' option, using your username instead of
":pserver:anonymous":

    cvs -d gfoot@cvs.allergogl.sourceforge.net:/cvsroot/allegrogl co alleggl

If you didn't do the last paragraph of section 2 above, you'll
be prompted, by `ssh', for your password.  This will happen
every time your CVS client needs to connect to the server.
Apart from these differences, and your write access, everything
is the same as in the anonymous case.


4. Updating to the most recent version
--------------------------------------
To check for updates, and perform any upgrade, just use the
command:

    cvs update

or `cvs up' for short if you prefer.

If you just want to see how the latest version in CVS differs
from the version on your hard disk, you can get a diff:

    cvs diff

This can be used to check for updates without checking them out,
and see what's changed, or to check whether you yourself have
edited any of the files.  As an anonymous user you're not
allowed to check in these changes.  In any case, if you do a
"cvs update", it will merge the upgrade with any changes you've
made, where possible, and alert you of conflicts otherwise.


5. Requesting a specific version
--------------------------------
There are two ways in which you can get a version other than the
latest.  The first is by using tags.  When I create a release, I
tag all the files with that release's ID.  You can ask for a
specific release by putting "-r <release>" after your `update'
or `co' command.  Like this, for Libnet:

    cvs update -r v0-10-6

Try `alpha-6' if you're playing with AllegroGL instead of
Libnet..

This is sticky, meaning that you'll only ever get the requested
version on future updates (so unless the tag moves, you won't
get any updates at all).  To undo this request and get back to
the most recent version, type:

    cvs update -A

The second way to request a different version is by date.  Use
the `-D' parameter after `update' or `co', specifying a date or
a difference from today's date:

    cvs update -D "yesterday"
    cvs update -D "3 months ago"
    cvs co -D "last Wednesday"

Again, this is sticky so you need to use `-A' to get up to date
again.


6. Discarding changes
---------------------
If you modify a file and want to discard all your changes,
simply delete the file then do a `cvs update'.  This will copy
the file back into place.  If you only want to discard some of
the changes, you could use `cvs diff' to see what changes you
made, and apply some of the patch to undo some of them.


7. Committing your modifications (registered developers only)
--------------------------------
The basic command to merge all your changes with the repository
is:

    cvs commit

I recommend first running an `update' command, in case there are
conflicts between your changes and somebody else's.  Conflicts
are marked in the file like this:

    <<<<<
    some lines
    of code from your
    modified version
    -----
    the lines from the
    updated version
    >>>>>

Resolve conflicts by editing the files, check they work, and
then try to commit them.

Try to keep the repository in a tidy state -- avoid making
updates which break working code until you've fixed the
breakages, and above all, it's nice if the repository compiles!

One thing to beware of is forgetting to add files (see below for
how to do it).  When you do a `cvs update', CVS lists all
`abnormal' files.  First it lists files which exist in your copy
but not in the repository, preceded by a `?'.  Then for all the
files in the repository, it lists them if they are:

    ? = not in the repository
    A = not in the repository, but scheduled to be added
    R = not in your copy, but scheduled to be removed
    U = not in your copy, but now updated
    P = old in your copy, but now patched
    M = modified in your copy
    C = modified in your copy, and conflicting with changes in the repository

This is a useful way to track how your copy differs from the
repository, and to make sure you've added all new files and
committed all your changes.  When your copy is in sync, `cvs
update' will print no output at all.

8. Adding/removing files (registered developers only)
------------------------
To add a file to the repository, use `cvs add <filename>' and
then do a commit.  Note that you're not allowed to specify a
path to the file -- you have to run the command from the
directory containing the file.

If the file is binary, write `cvs add -kb <filename>'.  This
disables keyword expansion and end-of-line mangling.

To delete a file, first remove it from your disk, and then issue
a `cvs remove <filename>' command.  You don't need to pass the
filename, if you want to remove all missing files in the current
directory.

Finally, commit your changes.  If you change your mind before
committing, issue a `cvs add' on the file and then when you
update, it will be created again.