Files: EDN - Working With
Introduction
Working with .edn files in Clojure allows storing and exchanging data in a simple and readable format. .edn stands for “extensible data notation” and is similar to JSON or XML but designed explicitly for Clojure. It provides a way to represent complex data structures using lists, vectors, maps, and sets. With .edn files, you can easily save and load data, making it convenient for configuration files, serialization, and interchanging data between different systems.
To work with .edn files in Clojure, you can use the language’s built-in functions. Clojure provides functions like slurp
to read .edn data from a file and edn/read-string
to read .edn data from a string. Similarly, you can use spit
to write .edn data to a file. These functions make reading and writing data in .edn format straightforward, allowing you to work efficiently with structured data in your Clojure programs.
Simple EDN File
Code
simple_example_read.edn
;; Example EDN file
^{:author "John" :created-at "2022-01-01"}
{:name "John Doe"
:age 30
:email "john.doe@example.com"
:address {:street "123 Main St"
:city "New York"
:state "NY"
:zip "10001"}
:interests ["programming" "reading" "hiking"]}
files_edn/core.clj
(ns files-edn.core
(:require
[clojure.edn :as edn]))
;; Reading Simple EDN Files
(def edn_simple_sample_data (edn/read-string (slurp "data_set/simple_example_read.edn")))
edn_simple_sample_data
;; => {:name "John Doe", :age 30, :email "john.doe@example.com", :address {:street "123 Main St", :city "New York", :state "NY", :zip "10001"}, :interests ["programming" "reading" "hiking"]}
(class edn_simple_sample_data)
;; => clojure.lang.PersistentArrayMap
(:name edn_simple_sample_data)
;; => "John Doe"
(:age edn_simple_sample_data)
;; => 30
(:email edn_simple_sample_data)
; => "john.doe@example.com"
(:address edn_simple_sample_data)
;; => {:street "123 Main St", :city "New York", :state "NY", :zip "10001"}
(:street (:address edn_simple_sample_data))
;; => "123 Main St"
(:interests edn_simple_sample_data)
;; => ["programming" "reading" "hiking"]
(def edn_simple_sample_meta_data (meta edn_simple_sample_data))
edn_simple_sample_meta_data
;; => {:author "John", :created-at "2022-01-01"}
(:author edn_simple_sample_meta_data)
;; => "John"
(:created-at edn_simple_sample_meta_data)
;; => "2022-01-01"
;; Writing Simple EDN Files
(spit "data_set/simple_example_write.edn" edn_simple_sample_data)
Explanation
The provided Clojure code reads data from an EDN (Extensible Data Notation) file, processes the data, and then writes the data back to another EDN file. Here’s the explanation in itemized format:
- Namespace Declaration:
(ns files-edn.core (:require [clojure.edn :as edn]))
- This code declares a namespace named
files-edn.core
and imports theclojure.edn
library, aliasing it asedn
for later use.
- This code declares a namespace named
- Reading EDN Files:
(def edn_simple_sample_data (edn/read-string (slurp "data_set/simple_example_read.edn")))
- Here, an EDN (Extensible Data Notation, a subset of Clojure used for data transfer) file is being read from the file system. The
slurp
function is used to read the contents of the file as a string.edn/read-string
parses this string into a Clojure data structure. The parsed data is assigned to theedn_simple_sample_data
var.
- Here, an EDN (Extensible Data Notation, a subset of Clojure used for data transfer) file is being read from the file system. The
- Inspecting the Data:
edn_simple_sample_data
- This line is simply outputting the contents of
edn_simple_sample_data
, which is an example of a map with nested map and vector.
- This line is simply outputting the contents of
(class edn_simple_sample_data)
- This line returns the class of
edn_simple_sample_data
which isclojure.lang.PersistentArrayMap
, indicating that this is an instance of an immutable Clojure map.
- This line returns the class of
(:name edn_simple_sample_data)
,(:age edn_simple_sample_data)
, etc.- These are examples of using keywords as functions to get the value of a map for a given key. For instance,
(:name edn_simple_sample_data)
returns “John Doe”.
- These are examples of using keywords as functions to get the value of a map for a given key. For instance,
- Meta Data Inspection:
(def edn_simple_sample_meta_data (meta edn_simple_sample_data))
- This line retrieves metadata associated with the
edn_simple_sample_data
var and assigns it toedn_simple_sample_meta_data
.
- This line retrieves metadata associated with the
edn_simple_sample_meta_data
- This line outputs the content of the metadata.
(:author edn_simple_sample_meta_data)
,(:created-at edn_simple_sample_meta_data)
- These lines retrieve the value associated with the keywords in the metadata map.
- Writing EDN Files:
(spit "data_set/simple_example_write.edn" edn_simple_sample_data)
- This writes the
edn_simple_sample_data
back to an EDN file using thespit
function, which writes data to a file. Theedn_simple_sample_data
is automatically converted back to a string before writing.
- This writes the
EDN with Custom Reader
Code
custom_example_read.edn
;; Example EDN file
^{:author "John" :created-at "2022-01-01"}
{:name "John Doe"
:age 30
:email "john.doe@example.com"
:address {:street "123 Main St"
:city "New York"
:state "NY"
:zip "10001"}
:interests ["programming" "reading" "hiking"]
:wishes
[#myapp/wishes "Learn 100 languages"
#myapp/wishes "Teleport to office"]}
files_edn/core.clj
(ns files-edn.core
(:require
[clojure.edn :as edn]))
;; Read Custom EDN Files
(defn wishes-reader [data]
(println "Parsing myapp/wishes tag with data:" data)
(str data " >>|<< " data))
(def edn_custom_sample_data (edn/read-string {:readers {'myapp/wishes wishes-reader}} (slurp "data_set/custom_example_read.edn")))
edn_custom_sample_data
;; => {:name "John Doe", :age 30, :email "john.doe@example.com", :address {:street "123 Main St", :city "New York", :state "NY", :zip "10001"}, :interests ["programming" "reading" "hiking"], :wishes ["Learn 100 languages >>|<< Learn 100 languages" "Teleport to office >>|<< Teleport to office"]}
(:wishes edn_custom_sample_data)
;; => ["Learn 100 languages >>|<< Learn 100 languages" "Teleport to office >>|<< Teleport to office"]
(:age edn_custom_sample_data)
;; => 30
(:author (meta edn_custom_sample_data))
;; => "John"
Explanation
This Clojure code snippet is focused on reading and manipulating EDN (Extensible Data Notation) files using custom tags. Here’s what’s happening:
- Namespace Definition:
(ns files-edn.core (:require [clojure.edn :as edn]))
is defining a namespace for this Clojure code and importing the Clojure EDN module. - Custom Tag Reader Definition:
defn
is being used to define a functionwishes-reader
that takes a data argument. This function logs a message and then creates a string that consists of the input data, the string>>|<<
, and the input data again. - Reading an EDN file with a Custom Tag:
(edn/read-string {:readers {'myapp/wishes wishes-reader}} (slurp "data_set/custom_example_read.edn"))
reads the content of an EDN file and parses it using the custom reader for the tag ‘myapp/wishes’. - Custom EDN File Parsing: The
edn/read-string
function parses the EDN string with a map as a first argument. The map{:readers {'myapp/wishes wishes-reader}}
indicates that whenever the parser encounters ‘myapp/wishes’ as a tag, it should use thewishes-reader
function to interpret the data. - Storing the Parsed EDN Data:
def
is used to bind the result of theedn/read-string
function to the symboledn_custom_sample_data
. - Accessing Data from the EDN: Keywords like
(:wishes edn_custom_sample_data)
,(:age edn_custom_sample_data)
, and(:author (meta edn_custom_sample_data))
are used to access specific data within the parsed EDN data structure. - Keyword Access: The first two expressions are straightforward: they access the ‘wishes’ and ‘age’ fields of the map.
- Lastly,
(:author (meta edn_custom_sample_data))
is used to fetch metadata attached to theedn_custom_sample_data
map. Themeta
function is used to fetch the metadata of a collection. In this case, it’s fetching the author of the data, which is “John”.