= Introduction to Fossil

:Author: Klaatu
:email: [email protected]
:Revision: 1.0

Keeping track of changes in code is vital, as any programmer is well aware.
There are many reasons for this.
Sometimes you just want a history of how your project started, and how it's evolved, as a matter of curiousity or education.
Other times, you want to enable other coders to contribute to your project and need a reliable way to merge disparate parts.
And still other times, an adjustment you make to fix one problem ends up breaking something else that was formerly working.
The Fossil source code management system is an all-in-one version control system, bug tracker, wiki, forum, and documentation solution from the creator of the famous SQLite database.


== Install Fossil

Fossil is a single self-contained C program, so you it's very likely that you can just https://fossil-scm.org/home/uv/download.html[download Fossil] from its homepage and place it anywhere in your system https://opensource.com/article/17/6/set-path-linux[PATH].
For example, assuming `/usr/local/bin` is in your path, as it usually is by default:

----[source,bash]
$ wget https://fossil-scm.org/home/uv/fossil-linux-x64-X.Y.tar.gz
$ sudo tar xvf fossil-linux-x64-X.Y.tar.gz \
--directory /usr/local/bin
----

You might also find Fossil in your software repository through your package manager, or you can compile it from source code yourself.


== Creating a Fossil repository

Assume you have a coding project you want to track with Fossil.
The first step is to create a Fossil repository.

----[source,bash]
$ fossil init myproject.fossil
project-id: 010836ac6112fefb0b015702152d447c8c1d8604
server-id:  54d837e9dc938ba1caa56d31b99c35a4c9627f44
admin-user: klaatu (initial password is "14b605")
----

As you can see, the creation of a Fossil repo returns three items: a unique project ID, a unique server ID, and an admin ID and password.
The project and server IDs are version numbers, and the admin credentials establishes your ownership of this repository, and can be used later should you decide to run Fossil as a server for other users to access.


== Work in a Fossil repository

You To start work in a Fossil repo, you must create a working location for its data.
You might think of this process as creating a virtual environment in Python, or unzipping a ZIP file that you intend to zip back up again later.

Create a working directory:

----[source,bash]
$ mkdir myprojectdir
$ cd myprojectdir
----

Change directory and open your Fossil repository into the directory you created:

----[source,bash]
$ fossil open ../myproject
project-name: <unnamed>
repository:   /home/klaatu/myprojectdir/../myproject
local-root:   /home/klaatu/myprojectdir/
config-db:    /home/klaatu/.fossil
project-code: 010836ac6112fefb0b015702152d447c8c1d8604
checkout:     9e6cd96dd675544c58a246520ad58cdd460d1559 2020-11-09 04:09:35 UTC
tags:         trunk
comment:      initial empty check-in (user: klaatu)
check-ins:    1
----

You might notice that Fossil has created a hidden file called `.fossil` in your home directory to track your global Fossil preferences.
This is not specific to your project, it's just an artifact of this being the first time you've used Fossil.

=== Adding files

To add files to your repository, use the `add` and `commit` subcommands.
For the sake of this example, create a simple README file and add it to the repository:

----[source,bash]
$ echo "My first Fossil project" > README
$ fossil add README
ADDED  README
$ fossil commit -m 'My first commit'
New_Version: 2472a43acd11c93d08314e852dedfc6a476403695e44f47061607e4e90ad01aa
----

=== Fossil branches

By default, a Fossile repository starts with a main branch called `trunk`.
You can branch off of the trunk when you want to work on code without effecting your primary code base.
Creating a new branch requires the `branch` subcommand, along with a new branch name and the branch you want to use as the basis for your new one.
In this case, there's no other branch to base off of aside from `trunk`, so try creating a new branch called `dev`:

----[source,bash]
$ fossil branch --help
Usage: fossil branch new BRANCH-NAME BASIS ?OPTIONS?
$ fossil branch new dev trunk
New branch: cb90e9c6f23a9c98e0c3656d7e18d320fa52e666700b12b5ebbc4674a0703695
----

You've create a new branch, but your current branch is still `trunk`:

----[source,bash]
$ fossil branch current
trunk
----

To make your new `dev` branch active, use the `checkout` command:

----[source,bash]
$ fossil checkout dev
dev
----

=== Merging changes

Suppose you've added an exciting new file to your `dev` branch, and having tested it you're satisfied that it's ready to take its place in trunk.
This is called a _merge_.

First, change your branch back to your target branch (in this example, that's `trunk`).
Your new file doesn't exist here, or if you modified a file that already existed then the changes to it don't exist here, but that's what the merge is going to take care of.

----[source,bash]
$ fossil checkout trunk
trunk
$ ls
README
----

Now you can perform the merge:

----[source,bash]
$ fossil merge dev
"fossil undo" is available to undo changes to the working checkout.
$ ls
myfile.lua  README
----

== View the Fossil timeline

To see the history of your repository, use the `timeline` option.
This shows you a detailed list of all activity in your repository, including a hash representing the change, the commit message you provided when committing code, and who made the change.

----[source,bash]
$ fossil timeline
=== 2020-11-09 ===
06:24:16 [5ef06e668b] added exciting new file (user: klaatu tags: dev)
06:11:19 [cb90e9c6f2] Create new branch named "dev" (user: klaatu tags: dev)
06:08:09 [a2bb73e4a3] *CURRENT* some additions were made (user: klaatu tags: trunk)
06:00:47 [2472a43acd] This is my first commit. (user: klaatu tags: trunk)
04:09:35 [9e6cd96dd6] initial empty check-in (user: klaatu tags: trunk)
+++ no more data (5) +++
----

== Making your Fossil repository public

Because Fossil features a built-in web interface, there's no need for a Fossil hosting service in the style of Gitlab or Gitea.
Fossil is its own hosting service, as long as you have a server to put it on.
Before making your Fossil project public, though, you must configure some attributes through the web UI.
Launch a local instance with the `ui` subcommand:

----[source,bash]
$ pwd
/home/klaatu/myprojectdir/
$ fossil ui
----

image:fossil-ui.jpg

Specifically, you should look at *Users* and *Settings* for security, and *Configuration* for project properties (including a proper title).
The web interface isn't just a convenience function.
It's intended for actual use, and indeed _is_ in use as the host for the Fossil project.
There are several surprisingly advance options, from user management (or self management, if you please) to single-signon (SSO) with other Fossil repositories on the same server.

Once you're satisfied with your changes, close the web interface and press *Ctrl+C* to stop the UI engine from running.
Commit your web changes just as you would any other update:

----[source,bash]
$ fossil commit -m 'web ui updates'
New_Version: 11fe7f2855a3246c303df00ec725d0fca526fa0b83fa67c95db92283e8273c60
----

Now you're ready to setup your Fossil server.

1. Copy your Fossil repository (in this example, `myproject.fossil`) to your server. All you need is the single file.
2. Install Fossil on your server, if it's not already installed. The process for installing Fossil to your server is the same as it was for your local computer.
3. In your `cgi-bin` directory (or the equivalent of that directory, depending upon which HTTP daemon you're running), create a file called `repo_myproject.cgi`:

----[source,php]
#!/usr/local/bin/fossil
repository: /home/klaatu/public_html/myproject.fossil
----

Make the script executable:

----[source,bash]
$ chmod +x repo_myproject.cgi
----

That's all there is to it.
Your project is now live on the Internet.

You can visit the web UI by navigating to your CGI script, such as `https://example.com/cgi-bin/repo_myproject.cgi`.

Or you can interact with your repository from a terminal through the same URL:

----[source,bash]
$ fossil clone https://klaatu@example.com/cgi-bin/repo_myproject.cgi
----

Working with a local clone requires you to use the `push` subcommand to send local changes back to your remote repository, and the `pull` subcommand to get changes made remotely into your local copy.

----[source,bash]
$ fossil push https://klaatu@example.com/cgi-bin/repo_myproject.cgi
----

== Use Fossil for independent hosting

Fossil places a lot of power into your hands (and the hands of your collaborators) as a coder, and makes you independent of hosting services.
This article has only hinted at the basics.
There's so much more to Fossil to aid you and your teams in your code projects.
Give Fossil a try.
It won't just change the way you think about version control, it'll help you _stop_ thinking about version control altogether.