Skip to content

Commit 6f1825c

Browse files
committed
Fix readme in light of Livebook hostility to tutorials
1 parent 606eb36 commit 6f1825c

File tree

1 file changed

+26
-20
lines changed

1 file changed

+26
-20
lines changed

livebooks/readme.livemd

+26-20
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@
55
## Dependencies
66

77
```elixir
8-
Mix.install([{:data_schema, path: "./"}])
8+
Mix.install([:data_schema, :sweet_xml])
99
```
1010

1111
Data schemas are declarative descriptions of how to create a struct from some input data. You can set up different schemas to handle different kinds of input data. By default we assume the incoming data is a map, but you can configure schemas to work with any arbitrary data input including XML and json.
@@ -72,7 +72,7 @@ end
7272
And:
7373

7474
```elixir
75-
defmodule Sandwich do
75+
defmodule SandwichV2 do
7676
require DataSchema
7777

7878
DataSchema.data_schema(field: {:list, "list", &{:ok, &1}, optional?: true, empty_values: [[]]})
@@ -82,7 +82,7 @@ end
8282
And:
8383

8484
```elixir
85-
defmodule Sandwich do
85+
defmodule SandwichV3 do
8686
require DataSchema
8787

8888
@options [optional?: true, empty_values: [nil], default: &DateTime.utc_now/0]
@@ -105,6 +105,8 @@ source_data = %{
105105

106106
And now let's assume the struct we wish to make is this one:
107107

108+
<!-- livebook:{"continue_on_error":true} -->
109+
108110
```elixir
109111
%BlogPost{
110112
content: "This is a blog post",
@@ -170,6 +172,8 @@ DataSchema.to_struct(source_data, BlogPost)
170172

171173
So imagine the input data came from an API response:
172174

175+
<!-- livebook:{"continue_on_error":true} -->
176+
173177
```elixir
174178
with {:ok, %{status: 200, response_body: body}} <- Http.get("https://www.my_thing.example.com"),
175179
{:ok, decoded} <- Jason.decode(body) do
@@ -182,21 +186,21 @@ end
182186
As we mentioned before we want to be able to handle multiple different kinds of source data in our schemas. For each type of source data we want to be able to specify how you access the data for each field type. We do that by providing a "data accessor" (a module that implements the `DataSchema.DataAccessBehaviour`) when we create the schema. We do this by providing a `@data_accessor` on the schema. By default if you do not provide this module attribute we use `DataSchema.MapAccessor`. That means the above example is equivalent to doing the following:
183187

184188
```elixir
185-
defmodule DraftPost do
189+
defmodule DraftPostV2 do
186190
import DataSchema, only: [data_schema: 1]
187191

188192
@data_accessor DataSchema.MapAccessor
189193
data_schema(field: {:content, "content", &{:ok, to_string(&1)}})
190194
end
191195

192-
defmodule Comment do
196+
defmodule CommentV2 do
193197
import DataSchema, only: [data_schema: 1]
194198

195199
@data_accessor DataSchema.MapAccessor
196200
data_schema(field: {:text, "text", &{:ok, to_string(&1)}})
197201
end
198202

199-
defmodule BlogPost do
203+
defmodule BlogPostV2 do
200204
import DataSchema, only: [data_schema: 1]
201205
@data_accessor DataSchema.MapAccessor
202206

@@ -206,8 +210,8 @@ defmodule BlogPost do
206210
]
207211
data_schema(
208212
field: {:content, "content", &{:ok, to_string(&1)}},
209-
has_many: {:comments, "comments", Comment},
210-
has_one: {:draft, "draft", DraftPost},
213+
has_many: {:comments, "comments", CommentV2},
214+
has_one: {:draft, "draft", DraftPostV2},
211215
aggregate: {:post_datetime, @date_time_fields, &NaiveDateTime.new(&1.date, &1.time)}
212216
)
213217
end
@@ -216,7 +220,7 @@ end
216220
When creating the struct DataSchema will call the relevant function for the field we are creating, passing it the source data and the path to the value(s) in the source. Our `DataSchema.MapAccessor` looks like this:
217221

218222
```elixir
219-
defmodule DataSchema.MapAccessor do
223+
defmodule DataSchema.MapAccessorV2 do
220224
@behaviour DataSchema.DataAccessBehaviour
221225

222226
@impl true
@@ -248,7 +252,7 @@ defmodule MapSchema do
248252
defmacro __using__(_) do
249253
quote do
250254
import DataSchema, only: [data_schema: 1]
251-
@data_accessor DataSchema.MapAccessor
255+
@data_accessor DataSchema.MapAccessorV2
252256
end
253257
end
254258
end
@@ -257,7 +261,7 @@ end
257261
Then use it like so:
258262

259263
```elixir
260-
defmodule DraftPost do
264+
defmodule DraftPostV3 do
261265
use MapSchema
262266

263267
data_schema(field: {:content, "content", &{:ok, to_string(&1)}})
@@ -266,6 +270,8 @@ end
266270

267271
This means should we want to change how we access data (say we wanted to use `Map.fetch!` instead of `Map.get`) we would only need to change the accessor used in one place - inside the `__using__` macro. It also gives you a handy place to provide other functions for the structs that get created, perhaps implementing a default Inspect protocol implementation for example:
268272

273+
<!-- livebook:{"continue_on_error":true} -->
274+
269275
```elixir
270276
defmodule MapSchema do
271277
defmacro __using__(opts) do
@@ -293,6 +299,8 @@ This could help ensure you never log sensitive fields by requiring you to explic
293299

294300
Now let's imagine instead that our source data was XML. What would it require to enable that? First a new Xpath data accessor:
295301

302+
<!-- livebook:{"continue_on_error":true} -->
303+
296304
```elixir
297305
defmodule XpathAccessor do
298306
@behaviour DataSchema.DataAccessBehaviour
@@ -342,21 +350,21 @@ source_data = """
342350
Let's define our schemas like so:
343351

344352
```elixir
345-
defmodule DraftPost do
353+
defmodule DraftPostV4 do
346354
import DataSchema, only: [data_schema: 1]
347355

348356
@data_accessor XpathAccessor
349357
data_schema(field: {:content, "./Content/text()", &{:ok, to_string(&1)}})
350358
end
351359

352-
defmodule Comment do
360+
defmodule CommentV4 do
353361
import DataSchema, only: [data_schema: 1]
354362

355363
@data_accessor XpathAccessor
356364
data_schema(field: {:text, "./text()", &{:ok, to_string(&1)}})
357365
end
358366

359-
defmodule BlogPost do
367+
defmodule BlogPostV4 do
360368
import DataSchema, only: [data_schema: 1]
361369

362370
@data_accessor XpathAccessor
@@ -366,8 +374,8 @@ defmodule BlogPost do
366374
]
367375
data_schema(
368376
field: {:content, "/Blog/Content/text()", &{:ok, to_string(&1)}},
369-
has_many: {:comments, "//Comment", Comment},
370-
has_one: {:draft, "/Blog/Draft", DraftPost},
377+
has_many: {:comments, "//Comment", CommentV4},
378+
has_one: {:draft, "/Blog/Draft", DraftPostV4},
371379
aggregate: {:post_datetime, @datetime_fields, &NaiveDateTime.new(&1.date, &1.time)}
372380
)
373381
end
@@ -389,7 +397,7 @@ source_data = """
389397
</Blog>
390398
"""
391399

392-
DataSchema.to_struct(source_data, BlogPost)
400+
DataSchema.to_struct(source_data, BlogPostV4)
393401

394402
# This will output:
395403

@@ -436,7 +444,7 @@ end
436444

437445
DataSchema.to_struct(data, Foo)
438446
# => Outputs the following:
439-
%Foo{a_rocket: "enables space travel"}
447+
# %Foo{a_rocket: "enables space travel"}
440448
```
441449

442450
## to_struct/5
@@ -461,7 +469,6 @@ fields = [
461469
]
462470

463471
DataSchema.to_struct(input, Run, fields, DataSchema.MapAccessor)
464-
{:ok, %Run{time: "10:00"}}
465472
```
466473

467474
Creating a map:
@@ -474,7 +481,6 @@ fields = [
474481
]
475482

476483
DataSchema.to_struct(input, %{}, fields, DataSchema.MapAccessor)
477-
{:ok, %{time: "10:00"}}
478484
```
479485

480486
## data_schema/1

0 commit comments

Comments
 (0)