The scenario:
A group of people are working on a sensitive data set that for practical reasons needs to be stored in a place that we’re not 100% happy with the security (e.g., Dropbox), or we’re concerned that files stored in plain text on users computers (e.g. laptops) may lead to the data being compromised.
If the data can be stored encrypted but everyone in the group can
still read and write the data then we’ve improved the situation
somewhat. But organising for everyone to get a copy of the key to
decrypt the data files is non-trivial. The workflow described here aims
to simplify this procedure using lower-level functions in the
cyphr
package.
The general procedure is this:
A person will set up a set of personal keys and a key for the data. The data key will be encrypted with their personal key so they have access to the data but nobody else does. At this point the data can be encrypted.
Additional users set up personal keys and request access to the data. Anyone with access to the data can grant access to anyone else.
Before doing any of this, everyone needs to have ssh keys set up. By default the package will use your ssh keys found at “~/.ssh”; see the main package vignette for how to use this.
For clarity here we will generate two sets of key pairs for two actors Alice and Bob:
path_key_alice <- cyphr::ssh_keygen(password = FALSE)
path_key_bob <- cyphr::ssh_keygen(password = FALSE)
These would ordinarily be on different machines (nobody has access to
anyone else’s private key) and they would be password protected. In the
function calls below, all the path_user
arguments would be
omitted.
We’ll store data in the directory data
; at present there
is nothing there (this is in a temporary directory for compliance with
CRAN policies but would ordinarily be somewhere persistent and under
version control ideally).
## character(0)
First, create a personal set of keys. These will be
shared across all projects and stored away from the data. Ideally one
would do this with ssh-keygen
at the command line,
following one of the many guides available. A utility function
ssh_keygen
(which simply calls ssh-keygen
for
you) is available in this package though. You will need to generate a
key on each computer you want access from. Don’t copy the key around. If
you lose your user key you will lose access to the data!
Second, create a key for the data and encrypt that key with your personal key. Note that the data key is never stored directly - it is always stored encrypted by a personal key.
## Generating data key
## Authorising ourselves
## Adding key d7:00:df:cc:99:a6:27:62:46:c7:e1:92:5d:c0:12:48:da:0c:01:f2:b7:e8:ac:60:ac:16:9a:e7:3f:66:1d:48
## user: root
## host: 51a0d6e26653
## date: 2025-01-15 10:21:42.171638
## Verifying
The data key is very important. If it is deleted, then the data
cannot be decrypted. So do not delete the directory
data_dir/.cyphr
! Ideally add it to your version control
system so that it cannot be lost. Of course, if you’re working in a
group, there are multiple copies of the data key (each encrypted with a
different person’s personal key) which reduces the chance of total
loss.
This command can be run multiple times safely; if it detects it has been rerun and the data key will not be regenerated.
## Already set up at /tmp/RtmpWWimol/data
## Verifying
Third, you can add encrypted data to the directory
(or to anywhere really). When run, cyphr::config_data
will
verify that it can actually decrypt things.
This object can be used with all the cyphr
functions
(see the “cyphr” vignette; vignette("cyphr")
)
filename <- file.path(data_dir, "iris.rds")
cyphr::encrypt(saveRDS(iris, filename), key)
dir(data_dir)
## [1] "iris.rds"
The file is encrypted and so cannot be read with
readRDS
:
## Error in readRDS(filename): unknown input format
But we can decrypt and read it:
## Sepal.Length Sepal.Width Petal.Length Petal.Width Species
## 1 5.1 3.5 1.4 0.2 setosa
## 2 4.9 3.0 1.4 0.2 setosa
## 3 4.7 3.2 1.3 0.2 setosa
## 4 4.6 3.1 1.5 0.2 setosa
## 5 5.0 3.6 1.4 0.2 setosa
## 6 5.4 3.9 1.7 0.4 setosa
Fourth, have someone else join in. Recall that to
simulate another person here, I’m going to pass an argument
path_user = path_key_bob
though to the functions. This
contains the path to “Bob”’s ssh keypair. If run on an actually
different computer this would not be needed; this is just to simulate
two users in a single session for this vignette (see minimal example
below where this is simulated). Again, typically this user would also
not use the cyphr::ssh_keygen
function but use the
ssh-keygen
command from their shell.
We’re going to assume that the user can read and write to the data. This is the case for my use case where the data are stored on dropbox and will be the case with GitHub based distribution, though there would be a pull request step in here.
This user cannot read the data, though trying to will print a message explaining how you might request access:
But bob
is your collaborator and needs access! What they
need to do is run:
## A request has been added
## Email someone with access to add you
##
## hash: 48:c9:75:1b:e8:78:3e:e7:c1:57:57:8b:f5:6e:ac:37:5e:bd:84:c6:e4:df:b0:cd:98:1b:c4:10:f3:ed:95:15
##
## If you are using git, you will need to commit and push first:
##
## git add .cyphr
## git commit -m "Please add me to the dataset"
## git push
(again, ordinarily you would not need the bob
bit
here)
The user should the send an email to someone with access and quote the hash in the message above.
Fifth, back on the first computer we can authorise the second user. First, see who has requested access:
## 1 key:
## 48:c9:75:1b:e8:78:3e:e7:c1:57:57:8b:f5:6e:ac:37:5e:bd:84:c6:e4:df:b0:cd:98:1b:c4:10:f3:ed:95:15
## user: root
## host: 51a0d6e26653
## date: 2025-01-15 10:21:42.283258
We can see the same hash here as above
(48c9751be8783ee7c157578bf56eac375ebd84c6e4dfb0cd981bc410f3ed9515
)
…and then grant access to them with the
cyphr::data_admin_authorise
function.
## There is 1 request for access
## Adding key 48:c9:75:1b:e8:78:3e:e7:c1:57:57:8b:f5:6e:ac:37:5e:bd:84:c6:e4:df:b0:cd:98:1b:c4:10:f3:ed:95:15
## user: root
## host: 51a0d6e26653
## date: 2025-01-15 10:21:42.283258
## Added 1 key
## If you are using git, you will need to commit and push:
##
## git add .cyphr
## git commit -m "Authorised root"
## git push
If you do not specify yes = TRUE
will prompt for
confirmation at each key added.
This has cleared the request queue:
## (empty)
and added it to our set of keys:
## 2 keys:
## 48:c9:75:1b:e8:78:3e:e7:c1:57:57:8b:f5:6e:ac:37:5e:bd:84:c6:e4:df:b0:cd:98:1b:c4:10:f3:ed:95:15
## user: root
## host: 51a0d6e26653
## date: 2025-01-15 10:21:42.283258
## d7:00:df:cc:99:a6:27:62:46:c7:e1:92:5d:c0:12:48:da:0c:01:f2:b7:e8:ac:60:ac:16:9a:e7:3f:66:1d:48
## user: root
## host: 51a0d6e26653
## date: 2025-01-15 10:21:42.171638
Finally, as soon as the authorisation has happened, the user can encrypt and decrypt files:
key_bob <- cyphr::data_key(data_dir, path_user = path_key_bob)
head(cyphr::decrypt(readRDS(filename), key_bob))
## Sepal.Length Sepal.Width Petal.Length Petal.Width Species
## 1 5.1 3.5 1.4 0.2 setosa
## 2 4.9 3.0 1.4 0.2 setosa
## 3 4.7 3.2 1.3 0.2 setosa
## 4 4.6 3.1 1.5 0.2 setosa
## 5 5.0 3.6 1.4 0.2 setosa
## 6 5.4 3.9 1.7 0.4 setosa
As above, but with less discussion:
Setup, on Alice’s computer:
## Generating data key
## Authorising ourselves
## Adding key d7:00:df:cc:99:a6:27:62:46:c7:e1:92:5d:c0:12:48:da:0c:01:f2:b7:e8:ac:60:ac:16:9a:e7:3f:66:1d:48
## user: root
## host: 51a0d6e26653
## date: 2025-01-15 10:21:42.389204
## Verifying
Get the data key key:
Encrypt a file:
Request access, on Bob’s computer:
## A request has been added
## Email someone with access to add you
##
## hash: 48:c9:75:1b:e8:78:3e:e7:c1:57:57:8b:f5:6e:ac:37:5e:bd:84:c6:e4:df:b0:cd:98:1b:c4:10:f3:ed:95:15
##
## If you are using git, you will need to commit and push first:
##
## git add .cyphr
## git commit -m "Please add me to the dataset"
## git push
Alice authorises this request::
## There is 1 request for access
## Adding key 48:c9:75:1b:e8:78:3e:e7:c1:57:57:8b:f5:6e:ac:37:5e:bd:84:c6:e4:df:b0:cd:98:1b:c4:10:f3:ed:95:15
## user: root
## host: 51a0d6e26653
## date: 2025-01-15 10:21:42.43524
## Added 1 key
## If you are using git, you will need to commit and push:
##
## git add .cyphr
## git commit -m "Authorised root"
## git push
Bob can get the data key:
Bob can read the secret data:
## Sepal.Length Sepal.Width Petal.Length Petal.Width Species
## 1 5.1 3.5 1.4 0.2 setosa
## 2 4.9 3.0 1.4 0.2 setosa
## 3 4.7 3.2 1.3 0.2 setosa
## 4 4.6 3.1 1.5 0.2 setosa
## 5 5.0 3.6 1.4 0.2 setosa
## 6 5.4 3.9 1.7 0.4 setosa
Encryption does not work through security through obscurity; it works because we can rely on the underlying maths enough to be open about how things are stored and where.
Most encryption libraries require some degree of security in the underlying software. Because of the way R works this is very difficult to guarantee; it is trivial to rewrite code in running packages to skip past verification checks. So this package is not designed to (or able to) avoid exploits in your running code; an attacker could intercept your private keys, the private key to the data, or skip the verification checks that are used to make sure that the keys you load are what they say they are. However, the data are safe; only people who have keys to the data will be able to read it.
cyphr
uses two different encryption algorithms; it uses
RSA encryption via the openssl
package for user keys,
because there is a common file format for these keys so it makes user
configuration easier. It uses the modern sodium package (and through
that the libsodium library) for data encryption because it is very fast
and simple to work with. This does leave two possible points of weakness
as a vulnerability in either of these libraries could lead to an exploit
that could allow decryption of your data.
Each user has a public/private key pair. Typically this is in
~/.ssh/id_rsa.pub
and ~/.ssh/id_rsa
, and if
found these will be used. Alternatively the location of the keypair can
be stored elsewhere and pointed at with the USER_KEY
or
USER_PUBKEY
environment variables. The key may be password
protected (and this is recommended!) and the password will be requested
without ever echoing it to the terminal.
The data directory has a hidden directory .cyphr
in
it.
## [1] ".cyphr" "iris.rds"
This does not actually need to be stored with the data but it makes sense to (there are workflows where data is stored remotely where storing this directory might make sense). The “keys” directory contains a number of files; one for each person who has access to the data.
## [1] "48c9751be8783ee7c157578bf56eac375ebd84c6e4dfb0cd981bc410f3ed9515"
## [2] "d700dfcc99a6276246c7e1925dc01248da0c01f2b7e8ac60ac169ae73f661d48"
## [1] "48c9751be8783ee7c157578bf56eac375ebd84c6e4dfb0cd981bc410f3ed9515"
## [2] "d700dfcc99a6276246c7e1925dc01248da0c01f2b7e8ac60ac169ae73f661d48"
(the file test
is a small file encrypted with the data
key used to verify everything is working OK).
Each file is stored in RDS format and is a list with elements:
h <- names(cyphr::data_admin_list_keys(data_dir))[[1]]
readRDS(file.path(data_dir, ".cyphr", "keys", h))
## $user
## [1] "root"
##
## $host
## [1] "51a0d6e26653"
##
## $date
## [1] "2025-01-15 10:21:42 UTC"
##
## $pub
## [2048-bit rsa public key]
## md5: c13c092415ccc9bea1a30495465c81d4
## sha256: 48c9751be8783ee7c157578bf56eac375ebd84c6e4dfb0cd981bc410f3ed9515
##
## $key
## [1] 43 8a 11 2d 5e 1d d9 4d 07 01 d9 1a f7 bf 12 76 96 4b 80 61 16 e4 e6 ff 16
## [26] da 8c ef e6 7c 68 1b c5 f5 b9 67 9d 1a 73 56 87 02 7b 95 cf 62 94 05 d1 72
## [51] 9d 90 89 d1 94 ab 64 99 37 f2 bc 7b 22 3f 75 bf 34 77 ef df d2 95 4b a9 12
## [76] c1 43 e3 30 6f 22 94 66 e9 80 c3 48 2a 1a 68 4d d0 3f cf ef 13 45 78 e1 48
## [101] 62 6b 75 2e 3b bb 85 be 91 de 26 b0 ff 67 e6 eb 75 c0 f4 eb 4f 15 78 4b 4f
## [126] 7f 59 3a e5 dd e5 1e a7 42 83 b5 a1 30 cc d9 ea 04 8e aa ee 6b 6a 43 31 ff
## [151] 5a cd 7c 7b ac 4f f7 9b c5 11 ae 81 13 e8 20 27 68 d4 8d 7a c8 5c 43 56 c5
## [176] f1 0a 82 80 95 88 b5 30 de b6 8f 5d a2 64 d2 cc 9e 39 ff a8 02 94 5e 40 c8
## [201] 67 39 ee c1 a4 28 b1 26 8f 4c 98 89 e7 38 8a b3 9a ea b3 93 63 b5 b7 a0 79
## [226] bf 95 9e b6 c9 aa 0c 1e a3 3e 5a 5b 2f 52 6b 81 fa d5 e9 71 ae 3a d3 7b 47
## [251] 55 03 a1 7d 42 69
You can see that the hash of the public key is the same as name of the stored file here (which is used to prevent collisions when multiple people request access at the same time).
## [1] "48c9751be8783ee7c157578bf56eac375ebd84c6e4dfb0cd981bc410f3ed9515"
When a request is posted it is an RDS file with all of the above
except for the key
element, which is added during
authorisation.
(Note that the verification relies on the package code not being
attacked, and given R’s highly dynamic nature an attacker could easily
swap out the definition for the verification function with something
that always returns TRUE
.)
When an authorised user creates the data_key
object
(which allows decryption of the data) secret
will:
~/.ssh/id_rsa
)$key
element from the list above).In the Dropbox scenario, non-password protected keys will afford only limited protection. This is because even though the keys and data are stored separately on Dropbox, they will be in the same place on a local computer; if that computer is lost then the only thing preventing an attacker recovering the data is security through obscurity (the data would appear to be random junk but they will be able to run your analysis scripts as easily as you can). Password protected keys will improve this situation considerably as without a password the data cannot be recovered.
The data is not encrypted during a running R session. R allows arbitrary modification of code at runtime so this package provides no security from the point where the data can be decrypted. If your computer was compromised then stealing the data while you are running R should be assumed to be straightforward.