archive 1.1.2

  r-lib

  Jim Hester

archive 1.1.2 is now on CRAN. archive lets you work with file archives, such as ZIP, tar, 7-Zip and RAR and compression formats like gzip, bzip2, XZ and Zstandard. It does this by building on top of the libarchive C library.

You can install it from CRAN with:

install.packages("archive")

This blog post will explain the main functions of archive, and show how you can use them to read from and write to archives.

You can see a full list of changes in the release notes

library(archive)
my_dir <- fs::file_temp() |> fs::dir_create()
knitr::opts_knit$set(root.dir = my_dir)

Displaying archive contents

Use archive() to return a tibble of the files contained in a given archive.

archive("nycflights13.zip")
#> # A tibble: 5 × 3
#>   path                       size date               
#>   <chr>                     <int> <dttm>             
#> 1 nycflights13/airlines.csv   386 2021-11-04 15:14:15
#> 2 nycflights13/airports.csv 71209 2021-11-04 15:14:15
#> 3 nycflights13/flights.csv  90886 2021-11-04 15:14:16
#> 4 nycflights13/planes.csv   72927 2021-11-04 15:14:16
#> 5 nycflights13/weather.csv  86753 2021-11-04 15:14:16

Reading single files from an archive

archive_read() is used to read a single file from an archive. This function returns an R connection, which can be passed to many R functions that take a connection object as input. All base R file system functions use connections, as well as some packages like readr.

The file= argument accepts numeric positions in the archive, or filenames as input.

con1 <- archive_read("nycflights13.zip", file = 2)
readLines(con1, n = 5)
#> [1] "faa,name,lat,lon,alt,tz,dst,tzone"                                                
#> [2] "04G,Lansdowne Airport,41.1304722,-80.6195833,1044,-5,A,America/New_York"          
#> [3] "06A,Moton Field Municipal Airport,32.4605722,-85.6800278,264,-6,A,America/Chicago"
#> [4] "06C,Schaumburg Regional,41.9893408,-88.1012428,801,-6,A,America/Chicago"          
#> [5] "06N,Randall Airport,41.431912,-74.3915611,523,-5,A,America/New_York"
close(con1)

con2 <- archive_read("nycflights13.zip", file = "nycflights13/planes.csv")
readLines(con2, n = 5)
#> [1] "tailnum,year,type,manufacturer,model,engines,seats,speed,engine"                 
#> [2] "N10156,2004,Fixed wing multi engine,EMBRAER,EMB-145XR,2,55,NA,Turbo-fan"         
#> [3] "N102UW,1998,Fixed wing multi engine,AIRBUS INDUSTRIE,A320-214,2,182,NA,Turbo-fan"
#> [4] "N103US,1999,Fixed wing multi engine,AIRBUS INDUSTRIE,A320-214,2,182,NA,Turbo-fan"
#> [5] "N104UW,1999,Fixed wing multi engine,AIRBUS INDUSTRIE,A320-214,2,182,NA,Turbo-fan"
close(con2)

Writing single files to an archive

Similarly archive_write() is used to write a single file to an archive. Again this creates a writable R connection. Like reading, many base R functions work with writable connections, as well as some packages like readr.

The archive and compression formats are automatically guessed based on the output filename file extensions. However you can also specify them explicity with the format and filter options.

Here we create a new zip archive containing the file mtcars.csv.

readr::write_csv(mtcars, archive_write("my-cars.zip", "mtcars.csv"))

archive("my-cars.zip")
#> # A tibble: 1 × 3
#>   path        size date               
#>   <chr>      <int> <dttm>             
#> 1 mtcars.csv  1281 1980-01-01 00:00:00

Writing multiple files to an archive

archive_write_files() writes multiple files to a new archive. In this case the files to be added to the archive should already be written on disk.

archive_write_dir() is a helper to archive all the files in a given directory.

library(readr)

# Write a few files to the temp directory
write_csv(iris, "iris.csv")
write_csv(mtcars, "mtcars.csv")
write_csv(airquality, "airquality.csv")

# Add them to a new XZ compressed tar archive
archive_write_files("data.tar.xz",
                    c("iris.csv", "mtcars.csv", "airquality.csv"))

# View archive contents
archive("data.tar.xz")
#> # A tibble: 3 × 3
#>   path            size date               
#>   <chr>          <int> <dttm>             
#> 1 iris.csv        3716 2021-11-04 15:14:17
#> 2 mtcars.csv      1281 2021-11-04 15:14:17
#> 3 airquality.csv  2890 2021-11-04 15:14:17

Extracting multiple files from an archive

archive_extract() allows you to extract one or more files to disk from an archive.

Note the archive and compression formats will be automatically detected.

# Create a new directory
my_dir <- fs::file_temp() |> fs::dir_create()

# Extract two of the files in the archive to that directory
archive_extract("data.tar.xz", dir = my_dir, files = c("iris.csv", "mtcars.csv"))

# Show the extracted files
fs::dir_ls(my_dir) |> fs::path_file()
#> [1] "iris.csv"   "mtcars.csv"

Acknowledgements

Thanks to the following users who have tried out the development versions of archive and opened issues and feature suggestions to improve it! @cboettig, @jennybc, @jeroen, and @JMcrocs.