mirror of
https://github.com/neongamerbot-qk/hackclub-nest
synced 2024-11-14 21:19:36 +00:00
update(project): finish up + ready for prod
Signed-off-by: Saahil <neon@saahild.com>
This commit is contained in:
parent
89753ca60d
commit
e32bce7249
4 changed files with 58 additions and 10 deletions
|
@ -4,12 +4,12 @@
|
||||||
- - [x] oauth [docs](https://tools.slack.dev/node-slack-sdk/oauth)
|
- - [x] oauth [docs](https://tools.slack.dev/node-slack-sdk/oauth)
|
||||||
- - [ ] fix the [channel](https://app.slack.com/client/T0266FRGM/C07RE4N7S4B)
|
- - [ ] fix the [channel](https://app.slack.com/client/T0266FRGM/C07RE4N7S4B)
|
||||||
- - - [x] Add ping for new-song event & send message ovs
|
- - - [x] Add ping for new-song event & send message ovs
|
||||||
- [ ] Spotify
|
- [x] Spotify
|
||||||
- - [x] web api [docs](https://developer.spotify.com/documentation/web-api/)
|
- - [x] web api [docs](https://developer.spotify.com/documentation/web-api/)
|
||||||
- - [x] oauth [docs](https://developer.spotify.com/documentation/general/guides/authorization-guide/)
|
- - [x] oauth [docs](https://developer.spotify.com/documentation/general/guides/authorization-guide/)
|
||||||
- - [x] refresh token [docs](https://developer.spotify.com/documentation/general/guides/authorization-guide/#refresh-an-access-token)
|
- - [x] refresh token [docs](https://developer.spotify.com/documentation/general/guides/authorization-guide/#refresh-an-access-token)
|
||||||
- - [x] playlist tools (creation,modifcation,deletion)
|
- - [x] playlist tools (creation,modifcation,deletion)
|
||||||
- [x] keydb (quick db or smthing)
|
- [x] keydb (quick db or smthing)
|
||||||
- [x] express
|
- [x] express
|
||||||
- [ ] transparency of added songs
|
- [x] transparency of added songs
|
||||||
- - [ ] export db -> into csv with properties (slack_id, url, added_at)
|
- - [x] export db -> into csv with properties (slack_id, url, added_at)
|
||||||
|
|
|
@ -17,6 +17,17 @@ const { QuickDB } = require("quick.db");
|
||||||
const db = new QuickDB({
|
const db = new QuickDB({
|
||||||
filePath: "./data/songs.sqlite",
|
filePath: "./data/songs.sqlite",
|
||||||
});
|
});
|
||||||
|
|
||||||
|
function arrayToCsv(data){
|
||||||
|
return data.map(row =>
|
||||||
|
row
|
||||||
|
.map(String) // convert every value to String
|
||||||
|
.map(v => v.replaceAll('"', '""')) // escape double quotes
|
||||||
|
.map(v => `"${v}"`) // quote it
|
||||||
|
.join(',') // comma-separated
|
||||||
|
).join('\r\n'); // rows starting on new lines
|
||||||
|
}
|
||||||
|
|
||||||
let cacheDb = {};
|
let cacheDb = {};
|
||||||
const app = express();
|
const app = express();
|
||||||
const userScopes = ["identity.avatar", "identity.basic", "identity.team"];
|
const userScopes = ["identity.avatar", "identity.basic", "identity.team"];
|
||||||
|
@ -69,7 +80,7 @@ app.use(
|
||||||
path: path.join(__dirname, "../data/sessions"),
|
path: path.join(__dirname, "../data/sessions"),
|
||||||
}),
|
}),
|
||||||
saveUninitialized: true,
|
saveUninitialized: true,
|
||||||
cookie: { secure: "auto", maxAge: 1000 * 60 * 60 * 24 * 365 },
|
cookie: { secure: "auto", maxAge: 1000 * 60 * 60 * 24 * 7 },
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -157,6 +168,18 @@ app.get("/home", async (req, res) => {
|
||||||
s: req.query.s,
|
s: req.query.s,
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
app.get('/download/db', async (req,res) => {
|
||||||
|
if(!req.session.info) return res.redirect("/login")
|
||||||
|
const allSongs = await db.all();
|
||||||
|
const csvData = arrayToCsv([["slack_id", "url","song_id", "added_at"],
|
||||||
|
...allSongs.map(d => {
|
||||||
|
return [d.value.added_by, d.value.song_url, d.id, d.value.added_at]
|
||||||
|
})
|
||||||
|
])
|
||||||
|
res.setHeader('Content-Type', 'text/csv');
|
||||||
|
res.setHeader('Content-Disposition', 'attachment; filename="songs.csv"');
|
||||||
|
res.send(csvData);
|
||||||
|
})
|
||||||
app.post("/spotify/submitsong", async (req, res) => {
|
app.post("/spotify/submitsong", async (req, res) => {
|
||||||
if (!req.session.token) return res.redirect("/login");
|
if (!req.session.token) return res.redirect("/login");
|
||||||
if (!cacheDb[req.query.token]) return res.redirect(`/home?error=0`);
|
if (!cacheDb[req.query.token]) return res.redirect(`/home?error=0`);
|
||||||
|
@ -230,3 +253,10 @@ app.listen(process.env.PORT || 3000, async () => {
|
||||||
// if(!await db.has())
|
// if(!await db.has())
|
||||||
if (getCredentials() !== null) refreshToken(getCredentials().refresh_token);
|
if (getCredentials() !== null) refreshToken(getCredentials().refresh_token);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
process.on('uncaughtException', function(err) {
|
||||||
|
console.log(err);
|
||||||
|
});
|
||||||
|
process.on('unhandledRejection', function(err) {
|
||||||
|
console.log(err);
|
||||||
|
});
|
|
@ -633,3 +633,12 @@ td {
|
||||||
max-width: var(--size-narrow-plus);
|
max-width: var(--size-narrow-plus);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
/* custom additions */
|
||||||
|
.error {
|
||||||
|
color: var(--red);
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
.success {
|
||||||
|
color: var(--green);
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
|
@ -23,18 +23,21 @@
|
||||||
<div class="card container" style="max-width: 550px;">
|
<div class="card container" style="max-width: 550px;">
|
||||||
<h2 class="headline">Submit Song</h2>
|
<h2 class="headline">Submit Song</h2>
|
||||||
<!-- <p> -->
|
<!-- <p> -->
|
||||||
<form action="/spotify/submitsong?token=<%=onetimetoken%>" method="POST">
|
<form action="/spotify/submitsong?token=<%=onetimetoken%>" method="POST" disabled="<%=s%>">
|
||||||
|
|
||||||
<div class="interactive">
|
<div class="interactive">
|
||||||
<label>Song URL</label>
|
<label>Song URL</label>
|
||||||
<input type="url" name="songurl" placeholder="https://open.spotify.com/track/6aWOvqmjb3343D5sq7zMgl?si=b72b81a97e564dc3" />
|
<input type="url" name="songurl" disabled="<%=s%>" placeholder="https://open.spotify.com/track/6aWOvqmjb3343D5sq7zMgl?si=b72b81a97e564dc3" />
|
||||||
<br>
|
<br>
|
||||||
</div>
|
</div>
|
||||||
<!-- </p> -->
|
<!-- </p> -->
|
||||||
<% if (s) { %>
|
<% if (s) { %>
|
||||||
|
<!-- <meta property="hhtp-equiv" content="refresh"> -->
|
||||||
|
<meta http-equiv="refresh" content="5; url='/home'" />
|
||||||
<div class="success">
|
<div class="success">
|
||||||
<h2>Success!</h2>
|
<h2>Success!</h2>
|
||||||
<p>Your song has been added to the playlist!</p>
|
<p class="success">Your song has been added to the playlist!</p>
|
||||||
|
<p>refreshing page in 5s</p>
|
||||||
<br>
|
<br>
|
||||||
</div>
|
</div>
|
||||||
<% } %>
|
<% } %>
|
||||||
|
@ -46,16 +49,22 @@
|
||||||
<% } %>
|
<% } %>
|
||||||
<br>
|
<br>
|
||||||
|
|
||||||
<button type="submit">Submit</button>
|
<button type="submit" disabled="<%=s%>">Submit</button>
|
||||||
</form>
|
</form>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
<footer>
|
<footer style="display: inline-flex;padding-left: 2px;padding-right: 2px;">
|
||||||
<a href="/logout" class="button">Logout</a>
|
<a href="/logout" class="button" style="padding-left: 2px;padding-right: 2px;">Logout</a>
|
||||||
|
<div>
|
||||||
|
<a href="/download/db" download class="button" style="margin-left: 8px;margin-right: 2px;">Download all song entries</a>
|
||||||
|
</div>
|
||||||
|
|
||||||
</footer>
|
</footer>
|
||||||
|
<p style="max-width: 250px; inline-size: max-content; word-break: break-all;color: gray;"> The "Download all song entries" button includes all songs submited thru the form (even if removed from the playlist)</p>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
</center>
|
</center>
|
||||||
</body>
|
</body>
|
||||||
|
|
Loading…
Reference in a new issue