forked from everruler12/roam2github
-
Notifications
You must be signed in to change notification settings - Fork 0
/
roam2github.js
222 lines (164 loc) · 7.66 KB
/
roam2github.js
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
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
const path = require('path')
const fs = require('fs-extra') // for mkdirp() and output() and remove() ~~and move()~~ and to promisfy all so don't have to use fs.promises
const puppeteer = require('puppeteer')
const extract = require('extract-zip')
console.time('R2G Exit after')
// NEED better check, because .env could exist in repo. like check of secrets exist in process.env, if so, IS_GITHUB_ACTION = true, other wise try local .env, and check again
let IS_LOCAL
try {
// check for local .env
if (fs.existsSync(path.join(__dirname, '.env'))) {
require('dotenv').config()
IS_LOCAL = true
} else {
IS_LOCAL = false
}
} catch (err) { error(`.env file existence error: ${err}`) }
const download_dir = path.join(__dirname, 'tmp')
const backup_dir = IS_LOCAL ? path.join(__dirname, 'backup') : getRepoPath()
const { R2G_EMAIL, R2G_PASSWORD, R2G_GRAPH } = process.env
if (!R2G_EMAIL) error('Secrets error: R2G_EMAIL not found')
if (!R2G_PASSWORD) error('Secrets error: R2G_PASSWORD not found')
if (!R2G_GRAPH) error('Secrets error: R2G_GRAPH not found')
function getRepoPath() {
// This works because actions/checkout@v2 duplicates repo name in path /home/runner/work/roam-backup/roam-backup
const parent_dir = path.join(__dirname, '..')
const repo_name = path.basename(parent_dir)
return path.join(parent_dir, repo_name)
}
init()
async function init() {
try {
// deleteDir(download_dir)
log('Creating browser')
const browser = await puppeteer.launch({ args: ['--no-sandbox'] }) // to run in GitHub Actions
// const browser = await puppeteer.launch({ headless: false }) // to test locally and see what's going on
const page = await browser.newPage()
await page._client.send('Page.setDownloadBehavior', { behavior: 'allow', downloadPath: download_dir })
await page.setDefaultNavigationTimeout(0)
await roam_login(page)
await roam_export(page)
log('Closing browser')
browser.close()
await extract_json()
// deleteDir(download_dir)
} catch (err) { error(err) }
console.timeEnd('R2G Exit after')
}
async function roam_login(page) {
return new Promise(async (resolve, reject) => {
try {
log('Navigating to login page')
await page.goto('https://roamresearch.com/#/signin')
const email_selector = 'input[name="email"]'
log('Waiting for login form')
await page.waitForSelector(email_selector)
log('Filling email field')
await page.type(email_selector, R2G_EMAIL)
log('Filling password field')
await page.type('input[name="password"]', R2G_PASSWORD)
log('Clicking "Sign In"')
await page.evaluate(() => {
[...document.querySelectorAll('button')].find(button => button.innerText == 'Sign In').click()
})
const login_error_selector = 'div[style="font-size: 12px; color: red;"]' // error message on login page
const graphs_selector = '.my-graphs' // successful login, on graphs selection page
await page.waitForSelector(login_error_selector + ', ' + graphs_selector)
const error_el = await page.$(login_error_selector)
if (error_el) {
const error_message = await page.evaluate(el => el.innerText, error_el)
reject(`Login error. Roam says: "${error_message}"`)
} else if (await page.$(graphs_selector)) {
log('Login successful')
resolve()
} else { // timeout
reject('Login error: unknown')
}
} catch (err) { reject(err) }
})
}
async function roam_export(page) {
return new Promise(async (resolve, reject) => {
try {
log('Navigating to graph')
await page.goto('https://roamresearch.com/404')// workaround to get disablecss and disablejs parameters to work by navigating away due to issue with puppeteer and # hash navigation (used in SPAs like Roam)
await page.goto(`https://roamresearch.com/#/app/${R2G_GRAPH}?disablecss=true&disablejs=true`)
log('Waiting for graph to load')
// CHECK if have permission to view graph
// IDEAS check for .navbar for app
// IDEAS wait for astrolabe spinner to stop
// IDEAS allow multiple graphs
await page.waitForSelector('.bp3-icon-more')
// log('Clicking "Share, export and more"')
await page.click('.bp3-icon-more')
// log('Clicking "Export All"')
await page.evaluate(() => {
[...document.querySelectorAll('li .bp3-fill')].find(li => li.innerText == 'Export All').click()
})
// log('Waiting for export dialog')
await page.waitForSelector('.bp3-dialog .bp3-button-text')
// log('Clicking Export Format')
await page.click('.bp3-dialog .bp3-button-text')
// log('Clicking "JSON"')
await page.evaluate(() => {
[...document.querySelectorAll('.bp3-text-overflow-ellipsis')].find(dropdown => dropdown.innerText == 'JSON').click()
})
// log('Clicking "Export All"')
await page.evaluate(() => {
[...document.querySelectorAll('button')].find(button => button.innerText == 'Export All').click()
})
log('Waiting for download to start')
await page.waitForSelector('.bp3-spinner')
await page.waitForSelector('.bp3-spinner', { hidden: true })
log('JSON container downloaded')
resolve()
} catch (err) { reject(err) }
})
}
async function extract_json() {
return new Promise(async (resolve, reject) => {
try {
// log('Checking download_dir')
const files = await fs.readdir(download_dir)
if (files.length === 0) {
reject('Extraction error: download dir is empty')
} else if (files) {
// log('Found', files)
const file = files[0]
const source = path.join(download_dir, file)
const target = path.join(download_dir, '_extraction')
log('Extracting JSON from ' + file)
await extract(source, { dir: target })
// log('Extraction complete')
// IDEA change JSON downloaded log to Downloaded Roam-Export-1234567890.zip
const json_filename = `${R2G_GRAPH}.json`
const json_fullpath = path.join(target, json_filename)
const new_json_fullpath = path.join(backup_dir, 'json', json_filename)
log('Formatting JSON')
const json = await fs.readJson(json_fullpath)
const new_json = JSON.stringify(json, null, 2)
// log('Saving formatted JSON')
await fs.outputFile(new_json_fullpath, new_json)
log('Cleaning up')
// log('Deleting download_dir')
await fs.remove(download_dir, { recursive: true })
// log('download_dir deleted')
resolve()
}
} catch (err) { reject(err) }
})
}
// async function deleteDownloads(dir) {
// // if already doesn't exist, don't log
// fs.rmdir(download_dir, { recursive: true })
// log('download dir deleted')
// }
function log(...messages) {
const timestamp = new Date().toISOString().replace('T', ' ').replace('Z', '')
console.log(timestamp, 'R2G', ...messages)
}
function error(err) {
log('ERROR -', err)
console.timeEnd('R2G Exit after')
process.exit(1)
}