Most readers will be at least somewhat familiar with git
. Git
is typically described with a phrase similar to the following:
Git is an open source distributed version control system
The Wikipedia article on Git
is an interesting read. I was surprised to learn that Linus Torvalds was the original author of git
. Yes - the same guy that was also the original author of the Linux kernel. It's been said that necessity is the mother of invention - or as Plato put it, "our need will be the creator" - and this is precisely what happened with git
. Torvalds needed a distributed version control system to manage development of the Linux kernel, and so he invented it.
As demonstrated here, git
has applicability beyond large, complex projects like kernel development - it's also useful on a small scale. And that is our objective with this recipe: Start small, and learn by doing.
Before getting into some details, let's digress briefly to provide some background on GitHub
- the other item that figures largely in this recipe. There's also a Wikipedia article on GitHub
that provides some background and history. Not all of GitHub's history is commendable. It has been weaponized to a certain extent by social extremists, and its purchase by Microsoft in 2018 has not been entirely positive for the platform. Nevertheless, it remains the largest - and best known - remote repository, and perhaps most importantly remains a "free" service - at least for now.
The following "initial conditions" are assumed:
you have a github account - in this recipe, I use mine:
https://github.com/seamusdemora
.you have installed the
git
package on your Raspberry Pi (or equivalent hardware)you now have a file:
~/.gitconfig
(ref); you should review it; run this command (optional):$ git config --global init.defaultBranch main # or, 'master' instead of 'main'
you have your files (code, instructions, documentation, etc) stored on your local host under a single directory:
~/pi-motd
in this example. This is called your repository, or repo.you have some familiarity with the (extensive) system manuals provided with
git
;man git
will list the various man pages supportinggit
; there are many.you wish to maintain a copy of your local repo on a remote server; i.e. you are familiar with the pros and cons of services provided by GitHub and similar providers.
These initial conditions may differ from your situation - or they may not. In either case you should read this to help choose a workflow that fits your situation. OTOH, if that's too much to think about now, you can read this, and make adjustments as you go along.
Before we begin, know that (N.B.) the semantics of git
and GitHub can be confusing. Do not despair! However you should be prepared to spend time to become at least slightly familiar the vocabulary of git
. Here's a search to help you find introductory material - you'll need to filter this to match your own tastes of course. I found this Atlassian Bitbucket tutorial on git
a useful resource at times, but that's not an "endorsement" or recommendation!
When you set up your GitHub account, you'll need (among other things) a userid and a password. That allows you to access your GitHub account - but not your RPi host. Since August, 2021, GitHub no longer accepts account passwords for computer host authentication. There are several methods that may be used for authentication, but for RPi - especially for headless RPi - SSH authentication may be the simplest. It's the only authentication method we'll cover here.
In this case, RPi is the client, and GitHub is the server. SSH key generation is done on the client, and the process is similar to setting up RPi as an SSH server. The procedure is straightforward, but it is also rigorous! I found GitHub's documentation for SSH auth both boring and incorrect/misleading, so here's mine - but you may certainly use GitHub's if it makes sense to you.
NOTE: I opted for no passphrase, and the email address added under -C
should be your email address - as supplied to GitHub.
$ ssh-keygen -t ed25519 -f ed25519-gh -C "seamusdemora@gmail.com"
Generating public/private ed25519 key pair.
...
# start your text editor; for example:
$ nano ~/.ssh/config
# make the following entries appropriate for your situation:
Host github github.com/seamusdemora
HostName github.com
IdentityFile /home/git/.ssh/id_ed25519_gh
User seamusdemora
# save the file & exit the editor
# note you may have other entries in 'config'; this one for github should "probably"
# be first - at least this was my experience.
For this step, use both an SSH terminal window to the RPi, and a web browser connected to your GitHub account.
- In the SSH terminal, show the public key on the CLI:
$ less ~/.ssh/id_ed25519_gh.pub
ssh-ed25519 AAAABBBBCCCCDDDDEEEEFFFFG...LLMMMMNNNNOOOOPPPPQQQQ seamusdemora@gmail.com
-
After logging in to your GitHub account:
-
Click your profile picture in the upper, right-hand corner of the page, and select
Settings
. -
This will produce a list on the left-hand side of the page; from that list click
SSH & GPG Keys
. -
This will yield another page/dialog titled SSH keys; on the right, click the
New SSH Key
button. -
This will yield yet another dialog/page titled Add new SSH Key; choose a suitable Title, select
Authentication key
for Key type, and finally, copy the entire string fromless
in the SSH terminal window, and paste that string into the text field named Key. -
If you're so inclined, you may repeat the process just above to also create a
Signing key
. -
When you're finished, click the button labeled
Add SSH key
-
-
The public key generated above should now be listed on the page titled SSH keys.
~$ ssh -T git@github.com
Hi seamus! You successfully authenticated, but GitHub does not provide shell access.
And so this is where I felt that GitHub's documentation left me stranded. It failed to explain there is more to their SSH authentication than simply generating keys & storing them on your GitHub site. The omission becomes apparent when an attempt is made to - for example - push
a commit from the RPi to the remote GitHub repo:
# F.A.I.L.U.R.E.:
$ cd pi-motd.git # pi-motd.git is the folder containing the local
# copy of my repository
$ git push --dry-run origin master
Username for 'https://github.com': seamusdemora # Huh !?!? what's with Username?
Password for 'https://seamusdemora@github.com': # Huh !?!? what's with Password?
remote: Support for password authentication was removed on August 13, 2021. Please use a personal access token instead.
remote: Please see https://github.blog/2020-12-15-token-authentication-requirements-for-git-operations/ for more information.
fatal: Authentication failed for 'https://github.com/seamusdemora/pi-motd/'
GitHub responded with prompts for Username
& Password
as if it were not aware its own administrators had deprecated password authentication. GitHub also declared itself ignorant of the fact that SSH keys had already been loaded! This seemed a spectacular failure to me at the time. Worse, no clues are given in the error report, or the GitHub documentation for a resolution.
After some research (mixed with much teeth-gnashing and swearing), the solution was found!
The 'remote origin' must be declared inside of each repo
Without belaboring this any further, this is what must be done to enable SSH authentication for each of the local repos on your RPi. This command modifies the .git/config
file (or in the case of a bare/"server" repo, just the config
file):
# S.U.C.C.E.S.S.:
# Point 1: Learn how to check the remote origin of a repo:
# From *inside* the repo, issue this command:
$ git remote show origin
# --OR--
$ git remote -v
# If the remote origin is not what it should be... CHANGE IT:
# 'git remote set-url origin git@github.com:gitusername/repository.git'
# NOTE THE SSH-LIKE PREFIX: 'git@github.com:' THIS DEFINES SSH ACCESS
$ git remote set-url origin git@github.com:seamusdemora/pi-motd.git # run from inside the repo
# And afterwards:
$ git push --dry-run origin master
# will work fine!
With the Authentication of Part II now sorted, let's continue:
- Log into your GitHub account from your browser.
- In the top row, click the Create new... button, and select New repository
- From the New repository creation page, enter a suitable Repository name; leave all other choices empty or unselected (i.e. README, .gitignore & license)
- Click the Create repository button - this creates an empty repository, and lands you on a page with lots of "helpful" GitHub instructions - ignore them for now, but leave the page open in a browser tab.
-
In my case, I began working on my "code" by copying the folder
/etc/update-motd.d
into my/home/pi
folder. I added some files, did some testing, and when I got it like I wanted it, I simply went into the folder/home/pi/pi-motd
, and initialized it usinggit
:$ git init Initialized empty Git repository in /home/pi/pi-motd/.git/
-
Go through each step in detail:
$ cd ~/pi-motd $ git add -v * # '*' = add all files in folder to repo, '-v' = verbose add '10-intro' add '10-uname' add '10-uname.moved' add '20-uptime' add '20-who.moved' add '30-temp' add '40-sdcard' add '50-network' add '55-osver' add '60-kernel' add '70-backup' add '75-imgutil' add '99-source' $ git commit -v -m "adding all files to empty repo" [master (root-commit) 332c66c] adding all files to empty repo 14 files changed, 44 insertions(+) create mode 100755 10-intro create mode 100644 10-uname create mode 100644 10-uname.moved create mode 100755 20-uptime create mode 100644 20-who.moved create mode 100755 30-temp create mode 100755 40-sdcard create mode 100755 50-network create mode 100755 55-osver create mode 100755 60-kernel create mode 100755 70-backup create mode 100755 75-imgutil create mode 100755 99-sourc # But this commit was a mistake, as it included some files that had been # "effectively removed" by changing their mode from 755 (ex) to 644 (non-ex). # Correct this by doing a 'git rm --cached <>' to remove from repo, NOT from fs $ git rm --cached 10-uname 10-uname.moved 20-who.moved [master 0b4b14f] removed 10-uname,10-uname.moved,20-who.moved from repo 3 files changed, 6 deletions(-) delete mode 100644 10-uname delete mode 100644 10-uname.moved delete mode 100644 20-who.moved # Just to make sure the 'rm --cached' files are no longer part of the repo: $ git ls-tree master --name-only 10-intro 20-uptime 30-temp 40-sdcard 50-network 55-osver 60-kernel 70-backup 75-imgutil 99-source
-
At this point, the repo appears to be correct. All files were added & committed, then 3 were removed/un-committed.
-
Recalling the lessons learned from Step II-5 wrt
git remote set-url origin ...
, we are now ready topush
the repo to GitHub using SSH authentication:$ cd ~/pi-motd $ git remote add origin git@github.com:seamusdemora/pi-motd.git # NOTE 'add' instead of 'set-url' # 'add' to create a new remote; 'set-url' to make a change $ git branch -M master $ git push -u origin master Enumerating objects: 17, done. Counting objects: 100% (17/17), done. Delta compression using up to 4 threads Compressing objects: 100% (13/13), done. Writing objects: 100% (17/17), 1.87 KiB | 1.87 MiB/s, done. Total 17 (delta 1), reused 0 (delta 0), pack-reused 0 remote: Resolving deltas: 100% (1/1), done. To github.com:seamusdemora/pi-motd * [new branch] master -> master branch 'master' set up to track 'origin/master'. $
-
And you should now be able to load the page for the pi-motd repo at GitHub in your browser. Note that some of the niceties have been added: a 'README.md' file, a license file and a few other files that may be useful.
- Getting Started - First-Time Git Setup
- Chapter 2 - Git Basics; highly recommended !
- Git: Show Remote URL & Check Origin
- Q&A: git - remote add origin vs remote set-url origin
- Bare vs. Non-Bare Repositories in Git
- Multiple SSH Keys settings for different github account; a GitHub gist
- Managing Multiple SSH RSA Keys; using the
~/.ssh/config
file - How to Manage Multiple SSH Keys; more on using the
~/.ssh/config
file - Managing several SSH identities explained; still more on using the
~/.ssh/config
file - Q&A: Best way to use multiple SSH private keys on one client; several approaches & nuances
- Q&A: Can I have multiple ssh keys in my .ssh folder?; re the
~/.ssh/config
file - How to create (initialize) a local Git Repository; short & sweet
- 'git ls-files' - Show information about files in the index and working tree
- Q&A: How do I delete a file from a Git repository?; good brief answer
- Getting Started with Git
- Q&A: How to move a git repository into another directory and make that directory a git repository? For shuffling things around!
- From GitHub, instructions for configuring
git
; set username, cache password, use HTTPS or SSH to access a GitHub repo frombash
. - Q&A: How can I stage and commit all files, including newly added files, using a single command?
- Q&A: Pushing local changes to a remote repository; this answer is relevant.
- Q&A: Discard all uncommitted changes, modified files, added files and non-added
- Q&A: Revert to commit by SHA hash in Git? dup.; don't let this Q&A confuse you! check this answer.
- Q&A: How to reset local file to the most recent commit on GitHub?; note use of
git log
. - How do I revert a Git repository to a previous commit?; too much information?
- Q&A: How to save username and password in GIT - a cacophony of answers!
- How do I provide a username and password when running “git clone git@remote.git”? - more answers!
- Git – Config Username & Password – Store Credentials - still more.
- About authentication to GitHub - a child of this page on Authentication
- Installing
gh
- GitHub's CLI - on Linux and BSD - How to install Github CLI on Linux - an alternative set of install instructions
- The GitHub CLI manual