-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathday05-postgresql.jl
114 lines (98 loc) · 3.31 KB
/
day05-postgresql.jl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
using FunSQL
using LibPQ
using DBInterface
# Make LibPQ compatible with DBInterface.
DBInterface.connect(::Type{LibPQ.Connection}, args...; kws...) =
LibPQ.Connection(args...; kws...)
DBInterface.prepare(conn::LibPQ.Connection, args...; kws...) =
LibPQ.prepare(conn, args...; kws...)
DBInterface.execute(conn::Union{LibPQ.Connection, LibPQ.Statement}, args...; kws...) =
LibPQ.execute(conn, args...; kws...)
const funsql_as_integer = FunSQL.Fun."(?::integer)"
const funsql_array_get = FunSQL.Fun."?[?]"
const funsql_nullif = FunSQL.Fun.nullif
const funsql_regexp_matches = FunSQL.Fun.regexp_matches
const funsql_split_part = FunSQL.Fun.split_part
const funsql_string_agg = FunSQL.Agg."string_agg(?, NULL ORDER BY ?)"
const funsql_string_to_table = FunSQL.Fun.string_to_table
const funsql_substr = FunSQL.Fun.substr
const funsql_with_ordinality = FunSQL.Fun."? WITH ORDINALITY"
@funsql begin
parse_stacks() =
begin
from(
with_ordinality(string_to_table(split_part(:input, "\n 1", 1), "\n")),
columns = [line, row])
cross_join(
from(
with_ordinality(regexp_matches(line, ".(.). ?", "g")),
columns = [captures, col]))
define(
col => as_integer(col),
char => nullif(array_get(captures, 1), " "))
group(col)
define(stack => string_agg(char, row))
end
parse_moves() =
begin
from(
with_ordinality(regexp_matches(:input, "move (\\d+) from (\\d+) to (\\d+)", "g")),
columns = [captures, index])
define(
count => as_integer(array_get(captures, 1)),
from => as_integer(array_get(captures, 2)),
to => as_integer(array_get(captures, 3)))
end
maybe_reverse(str; reverse) =
$(reverse ? FunSQL.Fun(:reverse, str) : str)
apply_move(; reverse) =
begin
join(move => from(moves), on = index == move.index)
partition(order_by = [col])
define(
index => index + 1,
stack =>
if col == move.from
substr(stack, move.count + 1)
elseif col == move.to
concat(
maybe_reverse(substr(lead(stack, move.from - col), 1, move.count),
reverse = $reverse),
stack)
else
stack
end)
end
solve(name; reverse) =
begin
from(stacks)
define(index => 1)
iterate(apply_move(reverse = $reverse))
partition()
filter(index == max(index))
group()
define($name => string_agg(substr(stack, 1, 1), col))
end
solve_part1() =
solve(part1, reverse = true)
solve_part2() =
solve(part2, reverse = false)
solve_all() =
begin
solve_part1().cross_join(solve_part2())
with(
stacks => parse_stacks(),
moves => parse_moves())
end
const q = solve_all()
end # @funsql
if isempty(ARGS)
println(FunSQL.render(q, dialect = :postgresql))
else
const db = DBInterface.connect(FunSQL.DB{LibPQ.Connection}, "")
for file in ARGS
input = read(file, String)
output = first(DBInterface.execute(db, q, input = input))
println("[$file] part1: $(output.part1), part2: $(output.part2)")
end
end