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.