forked from efixler/scrape
-
Notifications
You must be signed in to change notification settings - Fork 2
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add token entry/storage functionality to test console #7
Merged
Merged
Changes from 13 commits
Commits
Show all changes
21 commits
Select commit
Hold shift + click to select a range
ffecac3
Working Alpine, with iframe
efixler 2a13044
- Remove iframe
efixler 25fc6a7
Update CSS box model
efixler feb5392
Re-hide the token field to cut this PR before material auth changes
efixler 7362a59
Review feedback
efixler 6433379
Add textarea for token entry
efixler e2212e1
Change checkbox to button(s)
efixler 920211a
Make url entry row properly responsive
efixler f8d6ff0
- Tweak CSS layout
efixler 5bd3d68
- Save and clear token
efixler 0ae3e74
Alert when token saved
efixler 2a5cd93
Remove unused CSS classes
efixler 5263f71
Add a little more padding-bottom to the Enter a URL label
efixler 3c6846f
Change Enter Token toggle from button to +/-
efixler 94d1eaa
Merge remote-tracking branch 'origin/main' into token-control
efixler 17e2663
Fix js error when auth is disabled
efixler 71b969b
Formatting
efixler 94bd09d
Tabs to spaces
efixler 2c52338
Fix Makefile when go not present
efixler f52bf20
grr remove extra line
efixler 9b77e61
Add TODO
efixler File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -107,9 +107,10 @@ var home embed.FS | |
|
||
func (h scrapeServer) mustHomeTemplate() *template.Template { | ||
tmpl := template.New("home") | ||
var keyF = func() string { return "" } | ||
if len(h.SigningKey) > 0 { | ||
keyF = func() string { | ||
var authTokenF = func() string { return "" } | ||
var authEnabledF = func() bool { return len(h.SigningKey) > 0 } | ||
if authEnabledF() { | ||
authTokenF = func() string { | ||
c, err := auth.NewClaims( | ||
auth.WithSubject("home"), | ||
auth.ExpiresIn(60*time.Minute), | ||
|
@@ -126,7 +127,11 @@ func (h scrapeServer) mustHomeTemplate() *template.Template { | |
return s | ||
} | ||
} | ||
tmpl = tmpl.Funcs(template.FuncMap{"AuthToken": keyF}) | ||
funcMap := template.FuncMap{ | ||
"AuthToken": authTokenF, | ||
"AuthEnabled": authEnabledF, | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This adds the hook so we can skip the token entry widget when auth is not enabled |
||
} | ||
tmpl = tmpl.Funcs(funcMap) | ||
homeSource, _ := home.ReadFile("templates/index.html") | ||
tmpl = template.Must(tmpl.Parse(string(homeSource))) | ||
return tmpl | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -4,92 +4,207 @@ | |
<meta charset="utf-8" /> | ||
<title>scrape</title> | ||
<meta name="viewport" content="width=device-width, initial-scale=1.0"> | ||
<script defer src="https://cdn.jsdelivr.net/npm/[email protected]/dist/cdn.min.js"></script> | ||
<style> | ||
body, html { | ||
margin: 0; | ||
padding: 0; | ||
height: 100vh; | ||
} | ||
div { | ||
box-sizing: border-box; | ||
} | ||
pre { | ||
white-space: pre-wrap; | ||
width: 100%; | ||
} | ||
label { | ||
display: block; | ||
font: 1rem 'Verdana' | ||
font-family: Verdana, Arial, Helvetica, sans-serif; | ||
font-size: 1.0rem; | ||
} | ||
input, | ||
label { | ||
margin: 0.2rem 0; | ||
input { | ||
padding-bottom: 0.1rem; | ||
} | ||
.page-container { | ||
display: flex; | ||
flex-direction: column; | ||
gap: 0.6rem; | ||
height: 100%; | ||
padding: 0.6rem; | ||
} | ||
.controls-container { | ||
width: 100%; | ||
background-color: #ffffff; | ||
border: 1px solid #414141; | ||
padding: 0.6rem; | ||
} | ||
.controls-container form { | ||
display: flex; | ||
flex-direction: column; | ||
} | ||
.controls-container label { | ||
padding-bottom: 0.3rem; | ||
} | ||
.url-entry-container { | ||
display: flex; | ||
align-items: center; | ||
gap: 0.3rem; | ||
} | ||
.url-entry-container input[type="url"] { | ||
flex-grow: 1; | ||
} | ||
.url-entry-container input[type="submit"] { | ||
padding-inline: 0.6rem; | ||
} | ||
.token-container { | ||
background-color: #ffffff; | ||
border: 1px solid #414141; | ||
padding: 0.6rem; | ||
display: flex; | ||
flex-direction: column; | ||
gap: 0.6rem; | ||
} | ||
.token-controls { | ||
display: flex; | ||
justify-content: space-between; | ||
} | ||
.token-entry { | ||
flex-grow: 1; | ||
margin-right: 0.6rem; | ||
} | ||
.data-iframe { | ||
position: relative; | ||
top: 16px; | ||
width: 98%; | ||
min-width: 768px; | ||
height: calc(100vh - 108px); | ||
.token-textarea { | ||
box-sizing: border-box; | ||
resize: vertical; | ||
width: 100%; | ||
height: 100%; | ||
} | ||
.token-info { | ||
white-space: nowrap; | ||
display: flex; | ||
flex-direction: column; | ||
align-items: flex-start; | ||
} | ||
.token-info button { | ||
width: 100%; | ||
padding: 0.1rem 1.0rem; | ||
cursor: pointer; | ||
} | ||
.token-info button:first-child { | ||
margin-top: 0.1rem; | ||
margin-bottom: 0.4rem; | ||
} | ||
.response-container { | ||
width: 100%; | ||
flex: 1; | ||
overflow: auto; | ||
border: 0.1rem inset #ccc; | ||
padding-block: 0.2rem; | ||
padding-inline: 0.4rem; | ||
user-select: text; | ||
} | ||
</style> | ||
</head> | ||
<body> | ||
<p> | ||
<form id="urlForm" action="/extract" method="POST" name="scrape" target="data-iframe"> | ||
<label for="url">Enter a URL:</label> | ||
<input type="submit" value="Hit It"> | ||
<input type="url" name="url" id="url" value="https://" size="96" maxlength="200" pattern="https?://.*" required title="URL"> | ||
<select title="URL Type" id="actionSelect" onchange="updateFormAction()"> | ||
<option value="/extract">Page</option> | ||
<option value="/extract/headless">Headless</option> | ||
<option value="/feed">Feed</option> | ||
</select> | ||
<input type="hidden" name="token" value="{{AuthToken}}"> | ||
<input type="hidden" name="pp" value="1"> | ||
</form> | ||
</p> | ||
<script> | ||
function updateFormAction() { | ||
var selected = document.getElementById("actionSelect").value; | ||
document.getElementById("urlForm").action = selected; | ||
} | ||
</script> | ||
<div class="page-container"> | ||
<div class="controls-container"> | ||
<form x-data="formHandler" @submit.prevent="handleSubmit"> | ||
<label for="url">Enter a URL:</label> | ||
<div class="url-entry-container"> | ||
<select title="URL Type" x-model="formAction"> | ||
<option value="/extract">Page</option> | ||
<option value="/extract/headless">Headless</option> | ||
<option value="/feed">Feed</option> | ||
</select> | ||
<input type="url" name="url" id="url" value="https://" maxlength="200" pattern="https?://.*" required title="URL"> | ||
<input type="submit" value="Hit It"> | ||
<input type="hidden" name="pp" value="1"> | ||
</div> | ||
</form> | ||
</div><!-- controls-container --> | ||
<!-- {{if AuthEnabled}} --> | ||
<div class="token-container" x-data="tokenHandler()"> | ||
<div class="token-header"> | ||
<button x-on:click="isCollapsed = !isCollapsed">Enter Token</button> | ||
</div> | ||
<div x-show="!isCollapsed" class="token-controls" x-transition.scale> | ||
<div class="token-entry"> | ||
<textarea x-model="authToken" class="token-textarea" id="token" rows="2" placeholder="Enter your access token here"></textarea> | ||
</div> | ||
<div class="token-info"> | ||
<button @click="saveToken()">Save Token</button> | ||
<button @click="clearToken()">Clear Token</button> | ||
</div> | ||
</div> | ||
</div> | ||
<!-- {{end}} --> | ||
<div class="response-container"> | ||
<pre x-ref="responseContent" id="responseContent"></pre> | ||
</div> | ||
</div><!-- page-container --> | ||
<script type="text/javascript"> | ||
document.addEventListener('DOMContentLoaded', function() { | ||
const iframe = document.getElementById('data'); | ||
iframe.contentDocument.body.innerHTML = ''; | ||
const preElement = document.createElement('pre'); | ||
preElement.style.whiteSpace = 'pre-wrap'; | ||
preElement.style.width = '100%'; | ||
iframe.contentDocument.body.appendChild(preElement); | ||
const form = document.getElementById('urlForm'); | ||
form.addEventListener('submit', async function(event) { | ||
event.preventDefault(); | ||
const action = form.action; | ||
const headers = new Headers(); | ||
const token = form.token.value; | ||
if (token) { | ||
headers.append('Authorization', `Bearer ${token}`); | ||
} | ||
const formData = new FormData(form); | ||
formData.delete('token'); | ||
try { | ||
const response = await fetch(action, { | ||
method: 'POST', | ||
headers: headers, | ||
body: formData | ||
}) | ||
if (response.ok) { | ||
const json = await response.json(); | ||
const jsonStr = JSON.stringify(json, null, 2); | ||
preElement.textContent = jsonStr; | ||
} else { | ||
const text = await response.text(); | ||
preElement.textContent = `Error ${response.status}:\n${text}`; | ||
throw new Error(`${response.status} - ${text}`); | ||
function formHandler() { | ||
return { | ||
formAction: '/extract', | ||
async handleSubmit(event) { | ||
const form = event.target; | ||
const headers = new Headers(); | ||
const token = document.getElementById('token').value; | ||
if (token) { | ||
headers.append('Authorization', `Bearer ${token}`); | ||
} | ||
const formData = new FormData(form); | ||
try { | ||
const response = await fetch(this.formAction, { | ||
method: 'POST', | ||
headers: headers, | ||
body: formData | ||
}) | ||
|
||
if (response.ok) { | ||
const json = await response.json(); | ||
const jsonStr = JSON.stringify(json, null, 2); | ||
this.updateContent(jsonStr); | ||
} else { | ||
const text = await response.text(); | ||
this.updateContent(`Error ${response.status}:\n${text}`); | ||
throw new Error(`${response.status} - ${text}`); | ||
} | ||
} catch (error) { | ||
console.error(error); | ||
} | ||
}, | ||
updateContent(content) { | ||
document.getElementById('responseContent').textContent = content; | ||
// following should work but doesn't | ||
// this.$refs.responseContent.textContent = content; | ||
} | ||
} catch (error) { | ||
console.error(error); | ||
} | ||
}); | ||
}); | ||
</script> | ||
<div> | ||
<iframe | ||
id="data" | ||
title="Scrape Results" | ||
name="data-iframe" | ||
class="data-iframe" | ||
> | ||
</div> | ||
} | ||
} | ||
</script> | ||
<script type="text/javascript"> | ||
function tokenHandler() { | ||
return { | ||
isCollapsed: true, | ||
authToken: localStorage.getItem('scrapeToken') || '{{AuthToken}}', | ||
async saveToken() { | ||
try { | ||
const token = document.getElementById('token').value; | ||
if (token) { | ||
localStorage.setItem('scrapeToken', token); | ||
alert('Token saved'); | ||
} | ||
} catch (e) { | ||
console.error('Error saving token to local storage', e); | ||
} | ||
}, | ||
async clearToken() { | ||
localStorage.removeItem('scrapeToken'); | ||
document.getElementById('token').value = ''; | ||
}, | ||
} | ||
} | ||
</script> | ||
|
||
</body> | ||
</html> |
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Just renamed this to make it a little more descriptive.