-
Notifications
You must be signed in to change notification settings - Fork 75
/
Copy pathhistdb-merge
executable file
·62 lines (51 loc) · 2.37 KB
/
histdb-merge
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
#!/usr/bin/env zsh
typeset -g HERE=$(dirname "${(%):-%N}")
local ancestor=${1:?three databases required}; shift
local ours=${1:?three databases required}; shift
local theirs=${1:?three databases required}
$HERE/histdb-migrate $ancestor # this is always reasonable to do
$HERE/histdb-migrate $ours # this also seems fine
V_OURS=$(sqlite3 -batch -noheader $ours 'PRAGMA user_version')
V_THEIRS=$(sqlite3 -batch -noheader $theirs 'PRAGMA user_version')
if [[ ${V_OURS} -lt ${V_THEIRS} ]] ; then
echo "Attempting to merge with a database from a future version (${V_THEIRS})."
echo "You need to update your histdb version to continue."
exit 1
elif [[ ${V_THEIRS} -lt ${V_OURS} ]]; then
echo "Merging different database versions: ours ${V_OURS}, theirs ${V_THEIRS}."
echo "To continue merging, their database needs migrating to newer version."
echo "If you do this, and push the changes, then the remote version of histdb will need upgrading too."
read -q "REPLY?Try migrating their database? [y/n] "
if [[ $REPLY != "y" ]]; then
echo "Cancelled."
exit 1
fi
fi
# for reasons I cannot use the encryption filter here.
# most annoying.
#
echo "Ancestor has $(sqlite3 ${ancestor} 'select count(*) from history') entries"
echo "We have $(sqlite3 ${ours} 'select count(*) from history') entries"
echo "Theirs have $(sqlite3 ${theirs} 'select count(*) from history') entries"
sqlite3 -batch -noheader "${theirs}" <<EOF
ATTACH DATABASE '${ours}' AS n;
ATTACH DATABASE '${ancestor}' AS a;
-- copy missing commands and places
INSERT INTO commands (argv) SELECT argv FROM n.commands as NC where NC.id > (SELECT max(id) FROM a.commands);
INSERT INTO places (host, dir) SELECT host, dir FROM n.places as NP where NP.id > (SELECT max(id) FROM a.places);
-- insert missing history, rewriting IDs
-- could uniquify sessions by host in this way too
INSERT INTO history (session, command_id, place_id, exit_status, start_time, duration)
SELECT NO.session, C.id, P.id, NO.exit_status, NO.start_time, NO.duration
FROM n.history NO
LEFT JOIN n.places PO ON NO.place_id = PO.id
LEFT JOIN n.commands CO ON NO.command_id = CO.id
LEFT JOIN commands C ON C.argv = CO.argv
LEFT JOIN places P ON (P.host = PO.host
AND P.dir = PO.dir)
WHERE NO.id > (SELECT MAX(id) FROM a.history)
;
VACUUM;
EOF
cp ${theirs} ${ours}
echo "Now we have $(sqlite3 ${ours} 'select count(*) from history') entries"