Compare commits
274 Commits
Author | SHA1 | Date | |
---|---|---|---|
ee0da63862 | |||
971f14bb55 | |||
9a1733ac99 | |||
c32d62fbd5 | |||
1a0d12d2ff | |||
2a27850914 | |||
bfcc4c985d | |||
1653d4fb4c | |||
79027c4c75 | |||
269bb0bfb6 | |||
7933d840b3 | |||
b875e9377e | |||
8c80946121 | |||
21d96e261f | |||
9c58d23b41 | |||
4ae2191392 | |||
d62a3ab86b | |||
9b7cd1da5a | |||
a301f1ecb6 | |||
f14639ee00 | |||
b527735f6f | |||
8cc01c58f3 | |||
a1d800a0f0 | |||
449899962a | |||
dc2030e6f3 | |||
ef5a1cd66e | |||
11e4ff42ed | |||
c71df35b22 | |||
345308a9ac | |||
75bbcefbec | |||
49a6168607 | |||
f55ea5a353 | |||
30c33d91e1 | |||
00b135fb0f | |||
5fe9ce8d7b | |||
8c04365049 | |||
d5b1c3a5bb | |||
f038aa61f4 | |||
f72c9d39be | |||
e6c2d08425 | |||
e901e99278 | |||
acd2cff747 | |||
8f913e696c | |||
226d39328c | |||
b2ad2f636c | |||
18fe7ff8cf | |||
077c222a4e | |||
2270b6cf95 | |||
758b627660 | |||
baf7272cfd | |||
6641e242af | |||
610fcfbf87 | |||
dea7f33910 | |||
c299e128ab | |||
53fa946c75 | |||
5d44a071f9 | |||
e29e203188 | |||
6ead6e08dc | |||
7360503234 | |||
140c1b1bfa | |||
040982f1fd | |||
4b0677d10e | |||
616751e054 | |||
5df957f193 | |||
7f9cb1b35a | |||
c030771d36 | |||
a562395c26 | |||
2bcdf68e40 | |||
72eaa685d0 | |||
899a414cf6 | |||
524538eeb8 | |||
a184559c21 | |||
1c9fd46e98 | |||
738941d830 | |||
06ab7a4181 | |||
6981d94417 | |||
dd60cb5b2b | |||
1d57e29d56 | |||
2d973707f3 | |||
cbb937b494 | |||
d45ee03122 | |||
162e00b243 | |||
7a32ba087e | |||
801216dfe9 | |||
21763c51cd | |||
138f8320e9 | |||
571ba3392c | |||
090ca1a140 | |||
6127f2a90c | |||
ef9bc791e1 | |||
894323becf | |||
4c89d31948 | |||
471d181284 | |||
0e48c66b8c | |||
a0bc1732cf | |||
6d5fd1dbf6 | |||
0f6e73dd87 | |||
151490faf0 | |||
fdf60e7255 | |||
ab102ca32c | |||
998b301229 | |||
d7839899e6 | |||
2385fa33ec | |||
1fd688eeed | |||
65543a43b2 | |||
0099021478 | |||
3a9b2dba32 | |||
59021b9331 | |||
078d6fe25b | |||
373ce55203 | |||
aef0442e9d | |||
21ff8d7b6f | |||
bca2a7e540 | |||
0c014ad41b | |||
32b8637c7e | |||
5ed122d92c | |||
45660816ce | |||
d19e73f059 | |||
18684c934b | |||
cf4c5e1fe8 | |||
7ef4a20aff | |||
292f8fbbb7 | |||
735f79d80b | |||
a85a4278f6 | |||
dbe617d7eb | |||
842d97e9fa | |||
0bf5576427 | |||
dd027bff4b | |||
f95ef51017 | |||
740641cb4e | |||
09c98c8da6 | |||
33c8bdffb9 | |||
5ab88567de | |||
c6627ceece | |||
d9affcdefc | |||
96607256fc | |||
eb9a0dcb4a | |||
89fa0d5489 | |||
22589c8296 | |||
b0540d2c57 | |||
41c4661bbb | |||
d2314580a9 | |||
a4d77926b6 | |||
bbe7024323 | |||
32e1469e11 | |||
2d4ca2379f | |||
374f53eb32 | |||
add7efea3c | |||
065d82a5f5 | |||
1895bbc025 | |||
65f1a2afb2 | |||
6eb9e6f0c0 | |||
eb735a42fe | |||
541c741bde | |||
7a33ed3434 | |||
48d2943f72 | |||
6bbc90bc0d | |||
4d18dc0bb8 | |||
6dbd002acd | |||
bf6245a505 | |||
91746908a1 | |||
bb8273bab4 | |||
62bcc31305 | |||
08683fa5a6 | |||
c58b077330 | |||
f445c42f55 | |||
a0866b251e | |||
aa819544f6 | |||
fac56d7f87 | |||
ef1ad17234 | |||
b8cdc605a2 | |||
ef2f9ad12b | |||
b13874d0db | |||
3d142afd03 | |||
7fcb7fcfed | |||
747d5a7c67 | |||
770c4d3630 | |||
e7b448a282 | |||
c7c787dff1 | |||
59a34a0e85 | |||
6e8cf69227 | |||
3444989f9a | |||
7e96bb3d80 | |||
0adbb1556e | |||
710eecdb9d | |||
8a57fa8a1d | |||
b33d79ed9b | |||
0f506fc41b | |||
c9cd825d55 | |||
e63384e6a6 | |||
3260a4d596 | |||
da5c4603d9 | |||
b50fa6f3ae | |||
aa7b4b2af7 | |||
2b0193f5ea | |||
2185d895c0 | |||
9c1a932214 | |||
8c0269af1c | |||
df89c5076e | |||
f819ad2bc6 | |||
f5884a5270 | |||
c046d00060 | |||
5934e34ea0 | |||
463b44ac52 | |||
b40d21e559 | |||
a422a74747 | |||
4124fa87d3 | |||
1dd0c4ee20 | |||
0dd114c166 | |||
67090e9b08 | |||
d97fb49fde | |||
9263b0657f | |||
a3384cbaa6 | |||
5d26b5c764 | |||
516403ee47 | |||
5ea504e6e8 | |||
f49bff9853 | |||
4ec529cdb8 | |||
779de6f8af | |||
0925a9b334 | |||
2f2ed6169d | |||
59ef30c76d | |||
d43b49e7e4 | |||
64a92195dd | |||
a7925ed62d | |||
39ba50dada | |||
bc1b29246d | |||
2d77a91150 | |||
93c1db502d | |||
a6dc7ee043 | |||
c7282520cd | |||
a866c1d068 | |||
aa9aad6743 | |||
f65ee2eb6a | |||
44c4341e67 | |||
1c886f8003 | |||
b481d26be2 | |||
f00ef59404 | |||
3115ff3436 | |||
443b198c12 | |||
ac84d8d2db | |||
e27cf94fbf | |||
68495fb280 | |||
bec5c78709 | |||
abfcfdf09e | |||
dad01749e6 | |||
2efb73cee3 | |||
ace21b21d5 | |||
280e16bd7f | |||
44d44a529c | |||
0957f2e339 | |||
3516404a5f | |||
d96daa335f | |||
285d9da26d | |||
9ab7a72bce | |||
46dd905509 | |||
63d595c67d | |||
dc0b5f42e6 | |||
9ecbd25488 | |||
83816fbcc6 | |||
11cfbdc3ed | |||
4b7cbb3de2 | |||
b1a572072c | |||
b1071e9579 | |||
da971f8680 | |||
b596f86cc2 | |||
3bcf0832a1 | |||
565f4f23b3 | |||
ef3820a2e1 | |||
1678245750 | |||
3594b6d41f | |||
a754d42b9e | |||
c7e0234d33 | |||
11a9ff53e4 |
24
.github/ISSUE_TEMPLATE/bug_report.md
vendored
Normal file
24
.github/ISSUE_TEMPLATE/bug_report.md
vendored
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
---
|
||||||
|
name: Bug report
|
||||||
|
about: Create a report to help us improve
|
||||||
|
title: Bug Report | [title]
|
||||||
|
labels: bug
|
||||||
|
assignees: ''
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Describe the bug**
|
||||||
|
A clear and concise description of what the bug is.
|
||||||
|
|
||||||
|
**To reproduce**
|
||||||
|
Steps to reproduce the behavior:
|
||||||
|
1. Go to '...'
|
||||||
|
2. Click on '....'
|
||||||
|
3. Scroll down to '....'
|
||||||
|
4. See error
|
||||||
|
|
||||||
|
**Expected behavior**
|
||||||
|
A clear and concise description of what you expected to happen.
|
||||||
|
|
||||||
|
**Additional context**
|
||||||
|
Add any other context about the problem here.
|
20
.github/ISSUE_TEMPLATE/feature_request.md
vendored
Normal file
20
.github/ISSUE_TEMPLATE/feature_request.md
vendored
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
---
|
||||||
|
name: Feature request
|
||||||
|
about: Suggest an idea for this project
|
||||||
|
title: Feature Request | [title]
|
||||||
|
labels: enhancement
|
||||||
|
assignees: ''
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Is your feature request related to a problem? Please describe.**
|
||||||
|
A clear and concise description of what the problem is. Ex. I'm always frustrated when [...]
|
||||||
|
|
||||||
|
**Describe the solution you'd like**
|
||||||
|
A clear and concise description of what you want to happen.
|
||||||
|
|
||||||
|
**Describe alternatives you've considered**
|
||||||
|
A clear and concise description of any alternative solutions or features you've considered.
|
||||||
|
|
||||||
|
**Additional context**
|
||||||
|
Add any other context or screenshots about the feature request here.
|
822
Cargo.lock
generated
822
Cargo.lock
generated
File diff suppressed because it is too large
Load Diff
24
Cargo.toml
24
Cargo.toml
@ -3,21 +3,19 @@ name = "libreddit"
|
|||||||
description = " Alternative private front-end to Reddit"
|
description = " Alternative private front-end to Reddit"
|
||||||
license = "AGPL-3.0"
|
license = "AGPL-3.0"
|
||||||
repository = "https://github.com/spikecodes/libreddit"
|
repository = "https://github.com/spikecodes/libreddit"
|
||||||
version = "0.2.0"
|
version = "0.2.9"
|
||||||
authors = ["spikecodes <19519553+spikecodes@users.noreply.github.com>"]
|
authors = ["spikecodes <19519553+spikecodes@users.noreply.github.com>"]
|
||||||
edition = "2018"
|
edition = "2018"
|
||||||
|
|
||||||
[features]
|
|
||||||
default = ["proxy"]
|
|
||||||
proxy = ["actix-web/rustls", "base64"]
|
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
base64 = { version = "0.13.0", optional = true }
|
base64 = "0.13"
|
||||||
actix-web = "3.2.0"
|
actix-web = { version = "3.3", features = ["rustls"] }
|
||||||
reqwest = { version = "0.10", default_features = false, features = ["rustls-tls"] }
|
futures = "0.3"
|
||||||
askama = "0.8.0"
|
askama = "0.10"
|
||||||
serde = "1.0.117"
|
ureq = "2"
|
||||||
|
serde = { version = "1.0", default_features = false, features = ["derive"] }
|
||||||
serde_json = "1.0"
|
serde_json = "1.0"
|
||||||
pulldown-cmark = "0.8.0"
|
async-recursion = "0.3"
|
||||||
chrono = "0.4.19"
|
url = "2.2"
|
||||||
async-recursion = "0.3.1"
|
regex = "1.4"
|
||||||
|
time = "0.2"
|
186
README.md
186
README.md
@ -2,58 +2,141 @@
|
|||||||
|
|
||||||
> An alternative private front-end to Reddit
|
> An alternative private front-end to Reddit
|
||||||
|
|
||||||
Libre + Reddit = Libreddit
|

|
||||||
|
|
||||||
- 🚀 Fast: written in Rust for blazing fast speeds and safety
|
---
|
||||||
- ☁️ Light: no javascript, no ads, no tracking
|
|
||||||
|
**10 second pitch:** Libreddit is a portmanteau of "libre" (meaning freedom) and "Reddit". It is a private front-end like [Invidious](https://github.com/iv-org/invidious) but for Reddit. Browse the coldest takes of [r/unpopularopinion](https://libredd.it/r/unpopularopinion) without being [tracked](#reddit).
|
||||||
|
|
||||||
|
- 🚀 Fast: written in Rust for blazing fast speeds and memory safety
|
||||||
|
- ☁️ Light: no JavaScript, no ads, no tracking, no bloat
|
||||||
- 🕵 Private: all requests are proxied through the server, including media
|
- 🕵 Private: all requests are proxied through the server, including media
|
||||||
- 🔒 Safe: does not rely on Reddit OAuth or require a Reddit API Key
|
- 🔒 Secure: strong [Content Security Policy](https://developer.mozilla.org/en-US/docs/Web/HTTP/CSP) prevents browser requests to Reddit
|
||||||
- 📱 Responsive: works great on mobile!
|
|
||||||
|
|
||||||
Think Invidious but for Reddit. Watch your cat videos without being watched.
|
---
|
||||||
|
|
||||||
## Screenshot
|
## Jump to...
|
||||||
|
- [About](#about)
|
||||||
|
- [Teddit Comparison](#how-does-it-compare-to-teddit)
|
||||||
|
- [Comparison](#comparison)
|
||||||
|
- [Installation](#installation)
|
||||||
|
- [Cargo](#a-cargo)
|
||||||
|
- [Docker](#b-docker)
|
||||||
|
- [AUR](#c-aur)
|
||||||
|
- [GitHub Releases](#d-github-releases)
|
||||||
|
- [Repl.it](#e-replit)
|
||||||
|
- [Deployment](#deployment)
|
||||||
|
|
||||||

|
---
|
||||||
|
|
||||||
## About
|
# Instances
|
||||||
|
|
||||||
### Elsewhere
|
Feel free to [open an issue](https://github.com/spikecodes/libreddit/issues/new) to have your [selfhosted instance](#deployment) listed here!
|
||||||
Find Libreddit on...
|
|
||||||
- 💬 Matrix: [#libreddit:matrix.org](https://matrix.to/#/#libreddit:matrix.org)
|
|
||||||
- 🐋 Docker: [spikecodes/libreddit](https://hub.docker.com/r/spikecodes/libreddit)
|
|
||||||
- :octocat: GitHub: [spikecodes/libreddit](https://github.com/spikecodes/libreddit)
|
|
||||||
- 🦊 GitLab: [spikecodes/libreddit](https://gitlab.com/spikecodes/libreddit)
|
|
||||||
|
|
||||||
### Info
|
| Website | Country | Cloudflare |
|
||||||
Libreddit hopes to provide an easier way to browse Reddit, without the ads, trackers and bloat. Libreddit was inspired by other alternative front-ends to popular services such as [Invidious](https://github.com/iv-org/invidious) for YouTube, [Nitter](https://github.com/zedeus/nitter) for Twitter, and [Bibliogram](https://sr.ht/~cadence/bibliogram/) for Instagram.
|
|-|-|-|
|
||||||
|
| [libredd.it](https://libredd.it) (official) | 🇺🇸 US | |
|
||||||
|
| [libreddit.spike.codes](https://libreddit.spike.codes) (official) | 🇺🇸 US | |
|
||||||
|
| [libreddit.dothq.co](https://libreddit.dothq.co) | 🇺🇸 US | ✅ |
|
||||||
|
| [libreddit.insanity.wtf](https://libreddit.insanity.wtf) | 🇺🇸 US | ✅ |
|
||||||
|
| [libreddit.kavin.rocks](https://libreddit.kavin.rocks) | 🇮🇳 IN | ✅ |
|
||||||
|
| [libreddit.himiko.cloud](https://libreddit.himiko.cloud) | 🇧🇬 BG | |
|
||||||
|
| [spjmllawtheisznfs7uryhxumin26ssv2draj7oope3ok3wuhy43eoyd.onion](http://spjmllawtheisznfs7uryhxumin26ssv2draj7oope3ok3wuhy43eoyd.onion) | 🇮🇳 IN | |
|
||||||
|
|
||||||
Libreddit currently implements most of Reddit's functionalities but still lacks a few features that are being worked on below.
|
A checkmark in the "Cloudflare" category here refers to the use of the reverse proxy, [Cloudflare](https://cloudflare). The checkmark will not be listed for a site which uses Cloudflare DNS but rather the proxying service which grants Cloudflare the ability to monitor traffic to the website.
|
||||||
|
|
||||||
### In Progress
|
---
|
||||||
- Nested comments
|
|
||||||
- User flairs
|
|
||||||
- Searching
|
|
||||||
|
|
||||||
### How does it compare to Teddit?
|
# About
|
||||||
|
|
||||||
|
Find Libreddit on 💬 [Matrix](https://matrix.to/#/#libreddit:kde.org), 🐋 [Docker](https://hub.docker.com/r/spikecodes/libreddit), :octocat: [GitHub](https://github.com/spikecodes/libreddit), and 🦊 [GitLab](https://gitlab.com/spikecodes/libreddit).
|
||||||
|
|
||||||
|
## Built with
|
||||||
|
|
||||||
|
- [Rust](https://www.rust-lang.org/) - Programming language
|
||||||
|
- [Actix Web](https://github.com/actix/actix-web) - Web server
|
||||||
|
- [Askama](https://github.com/djc/askama) - Templating engine
|
||||||
|
- [ureq](https://github.com/algesten/ureq) - HTTP client
|
||||||
|
|
||||||
|
## Info
|
||||||
|
Libreddit hopes to provide an easier way to browse Reddit, without the ads, trackers, and bloat. Libreddit was inspired by other alternative front-ends to popular services such as [Invidious](https://github.com/iv-org/invidious) for YouTube, [Nitter](https://github.com/zedeus/nitter) for Twitter, and [Bibliogram](https://sr.ht/~cadence/bibliogram/) for Instagram.
|
||||||
|
|
||||||
|
Libreddit currently implements most of Reddit's (signed-out) functionalities but still lacks [a few features](https://github.com/spikecodes/libreddit/issues).
|
||||||
|
|
||||||
|
## How does it compare to Teddit?
|
||||||
|
|
||||||
Teddit is another awesome open source project designed to provide an alternative frontend to Reddit. There is no connection between the two and you're welcome to use whichever one you favor. Competition fosters innovation and Teddit's release has motivated me to build Libreddit into an even more polished product.
|
Teddit is another awesome open source project designed to provide an alternative frontend to Reddit. There is no connection between the two and you're welcome to use whichever one you favor. Competition fosters innovation and Teddit's release has motivated me to build Libreddit into an even more polished product.
|
||||||
|
|
||||||
If you are looking to compare, the biggest differences I have noticed are:
|
If you are looking to compare, the biggest differences I have noticed are:
|
||||||
- Libreddit is themed around Reddit's redesign whereas Teddit appears to stick much closer to Reddit's old design. This may suit some users better as design is always subjective.
|
- Libreddit is themed around Reddit's redesign whereas Teddit appears to stick much closer to Reddit's old design. This may suit some users better as design is always subjective.
|
||||||
- Libreddit is written in Rust for speed and memory safety. It uses Actix Web, which was [benchmarked as the fastest web server for single queries](https://www.techempower.com/benchmarks/#hw=ph&test=db).
|
- Libreddit is written in [Rust](https://www.rust-lang.org) for speed and memory safety. It uses [Actix Web](https://actix.rs), which was [benchmarked as the fastest web server for single queries](https://www.techempower.com/benchmarks/#hw=ph&test=db).
|
||||||
- Unlike Teddit (at the time of writing this), Libreddit does not require a Reddit API key to host.
|
|
||||||
|
|
||||||
## Instances
|
---
|
||||||
|
|
||||||
Feel free to [open an issue](https://github.com/spikecodes/libreddit/issues/new) to have your selfhosted instance listed here!
|
# Comparison
|
||||||
|
|
||||||
- [libredd.it](https://libredd.it) 🇺🇸 (Thank you to [YeapGuy](https://github.com/YeapGuy)!)
|
This section outlines how Libreddit compares to Reddit.
|
||||||
- [libreddit.spike.codes](https://libreddit.spike.codes) 🇺🇸
|
|
||||||
|
|
||||||
## Installation
|
## Speed
|
||||||
|
|
||||||
### A) Cargo
|
Lasted tested Jan 17, 2021.
|
||||||
|
|
||||||
|
Results from Google Lighthouse ([Libreddit Report](https://lighthouse-dot-webdotdevsite.appspot.com/lh/html?url=https%3A%2F%2Flibredd.it), [Reddit Report](https://lighthouse-dot-webdotdevsite.appspot.com/lh/html?url=https%3A%2F%2Fwww.reddit.com%2F)).
|
||||||
|
|
||||||
|
| | Libreddit | Reddit |
|
||||||
|
|------------------------|---------------|------------|
|
||||||
|
| Requests | 20 | 70 |
|
||||||
|
| Resource Size (card ui)| 1,224 KiB | 1,690 KiB |
|
||||||
|
| Time to Interactive | **1.5 s** | **11.2 s** |
|
||||||
|
|
||||||
|
## Privacy
|
||||||
|
|
||||||
|
### Reddit
|
||||||
|
|
||||||
|
**Logging:** According to Reddit's [privacy policy](https://www.redditinc.com/policies/privacy-policy), they "may [automatically] log information" including:
|
||||||
|
- IP address
|
||||||
|
- User-agent string
|
||||||
|
- Browser type
|
||||||
|
- Operating system
|
||||||
|
- Referral URLs
|
||||||
|
- Device information (e.g., device IDs)
|
||||||
|
- Device settings
|
||||||
|
- Pages visited
|
||||||
|
- Links clicked
|
||||||
|
- The requested URL
|
||||||
|
- Search terms
|
||||||
|
|
||||||
|
**Location:** The same privacy policy goes on to describe location data may be collected through the use of:
|
||||||
|
- GPS (consensual)
|
||||||
|
- Bluetooth (consensual)
|
||||||
|
- Content associated with a location (consensual)
|
||||||
|
- Your IP Address
|
||||||
|
|
||||||
|
**Cookies:** Reddit's [cookie notice](https://www.redditinc.com/policies/cookies) documents the array of cookies used by Reddit including/regarding:
|
||||||
|
- Authentication
|
||||||
|
- Functionality
|
||||||
|
- Analytics and Performance
|
||||||
|
- Advertising
|
||||||
|
- Third-Party Cookies
|
||||||
|
- Third-Party Site
|
||||||
|
|
||||||
|
### Libreddit
|
||||||
|
|
||||||
|
For transparency, I hope to describe all the ways Libreddit handles user privacy.
|
||||||
|
|
||||||
|
**Logging:** In production (when running the binary, hosting with docker, or using the official instances), Libreddit logs when Reddit is ratelimiting Libreddit and when Reddit's JSON responses can't be parsed. When debugging (running from source without `--release`), Libreddit logs post IDs and URL paths fetched to aid with troubleshooting.
|
||||||
|
|
||||||
|
**DNS:** Both official domains (`libredd.it` and `libreddit.spike.codes`) use Cloudflare as the DNS resolver. Though, the sites are not proxied through Cloudflare meaning Cloudflare doesn't have access to user traffic.
|
||||||
|
|
||||||
|
**Cookies:** Libreddit uses optional cookies to store any configured settings in [the settings menu](https://libredd.it/settings). This is not a cross-site cookie and the cookie holds no personal data, only a value of the possible layout.
|
||||||
|
|
||||||
|
**Hosting:** The official instances are hosted on [Repl.it](https://repl.it/) which monitors usage to prevent abuse. I can understand if this invalidates certain users' threat models and therefore, selfhosting and browsing through Tor are welcomed.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
# Installation
|
||||||
|
|
||||||
|
## 1) Cargo
|
||||||
|
|
||||||
Make sure Rust stable is installed along with `cargo`, Rust's package manager.
|
Make sure Rust stable is installed along with `cargo`, Rust's package manager.
|
||||||
|
|
||||||
@ -61,9 +144,9 @@ Make sure Rust stable is installed along with `cargo`, Rust's package manager.
|
|||||||
cargo install libreddit
|
cargo install libreddit
|
||||||
```
|
```
|
||||||
|
|
||||||
### B) Docker
|
## 2) Docker
|
||||||
|
|
||||||
Deploy the Docker image of Libreddit:
|
Deploy the [Docker image](https://hub.docker.com/r/spikecodes/libreddit) of Libreddit:
|
||||||
```
|
```
|
||||||
docker run -d --name libreddit -p 8080:8080 spikecodes/libreddit
|
docker run -d --name libreddit -p 8080:8080 spikecodes/libreddit
|
||||||
```
|
```
|
||||||
@ -73,39 +156,46 @@ Deploy using a different port (in this case, port 80):
|
|||||||
docker run -d --name libreddit -p 80:8080 spikecodes/libreddit
|
docker run -d --name libreddit -p 80:8080 spikecodes/libreddit
|
||||||
```
|
```
|
||||||
|
|
||||||
### C) AUR
|
## 3) AUR
|
||||||
|
|
||||||
For ArchLinux users, Libreddit is available from the AUR as [`libreddit-git`](https://aur.archlinux.org/packages/libreddit-git).
|
For ArchLinux users, Libreddit is available from the AUR as [`libreddit-git`](https://aur.archlinux.org/packages/libreddit-git).
|
||||||
|
|
||||||
Install:
|
|
||||||
```
|
```
|
||||||
yay -S libreddit-git
|
yay -S libreddit-git
|
||||||
```
|
```
|
||||||
|
|
||||||
### D) GitHub Releases
|
## 4) GitHub Releases
|
||||||
|
|
||||||
If you're on Linux and none of these methods work for you, you can grab a Linux binary from [the newest release](https://github.com/spikecodes/libreddit/releases/latest).
|
If you're on Linux and none of these methods work for you, you can grab a Linux binary from [the newest release](https://github.com/spikecodes/libreddit/releases/latest).
|
||||||
Currently Libreddit does not have Windows or MacOS binaries but those will be available soon.
|
|
||||||
|
|
||||||
## Deploy an Instance
|
## 5) Repl.it
|
||||||
|
|
||||||
Once installed, deploy Libreddit (unless you're using Docker) by running:
|
**Note:** Repl.it is a free option but they are *not* private and will monitor server usage to prevent abuse. If you need a free and easy setup, this method may work best for you.
|
||||||
|
|
||||||
|
1. Create a Repl.it account (see note above)
|
||||||
|
2. Visit [the official Repl](https://repl.it/@spikethecoder/libreddit) and fork it
|
||||||
|
3. Hit the run button to download the latest Libreddit version and start it
|
||||||
|
|
||||||
|
In the web preview (defaults to top right), you should see your instance hosted where you can assign a [custom domain](https://docs.repl.it/repls/web-hosting#custom-domains).
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
# Deployment
|
||||||
|
|
||||||
|
Once installed, deploy Libreddit to `0.0.0.0:8080` by running:
|
||||||
|
|
||||||
```
|
```
|
||||||
libreddit
|
libreddit
|
||||||
```
|
```
|
||||||
|
|
||||||
Specify a custom address for the server by passing the `-a` or `--address` argument:
|
## Options
|
||||||
```
|
|
||||||
libreddit --address=0.0.0.0:8111
|
|
||||||
```
|
|
||||||
|
|
||||||
To disable the media proxy built into Libreddit, run:
|
| Short | Long | Example |
|
||||||
```
|
|-------|--------------------|-----------------------------------|
|
||||||
libreddit --no-default-features
|
| `-a` | `--address` | `libreddit --adress=0.0.0.0:8111` |
|
||||||
```
|
| `-r` | `--redirect-https` | `libreddit --redirect-https` |
|
||||||
|
|
||||||
## Building from Source
|
## Building
|
||||||
|
|
||||||
```
|
```
|
||||||
git clone https://github.com/spikecodes/libreddit
|
git clone https://github.com/spikecodes/libreddit
|
||||||
|
136
src/main.rs
136
src/main.rs
@ -1,10 +1,15 @@
|
|||||||
// Import Crates
|
// Import Crates
|
||||||
use actix_web::{get, middleware::NormalizePath, web, App, HttpResponse, HttpServer};
|
use actix_web::{
|
||||||
|
dev::{Service, ServiceResponse},
|
||||||
|
middleware, web, App, HttpResponse, HttpServer,
|
||||||
|
};
|
||||||
|
use futures::future::FutureExt;
|
||||||
|
|
||||||
// Reference local files
|
// Reference local files
|
||||||
mod popular;
|
|
||||||
mod post;
|
mod post;
|
||||||
mod proxy;
|
mod proxy;
|
||||||
|
mod search;
|
||||||
|
mod settings;
|
||||||
mod subreddit;
|
mod subreddit;
|
||||||
mod user;
|
mod user;
|
||||||
mod utils;
|
mod utils;
|
||||||
@ -15,54 +20,123 @@ async fn style() -> HttpResponse {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async fn robots() -> HttpResponse {
|
async fn robots() -> HttpResponse {
|
||||||
HttpResponse::Ok().body(include_str!("../static/robots.txt"))
|
HttpResponse::Ok()
|
||||||
|
.header("Cache-Control", "public, max-age=1209600, s-maxage=86400")
|
||||||
|
.body("User-agent: *\nAllow: /")
|
||||||
}
|
}
|
||||||
|
|
||||||
#[get("/favicon.ico")]
|
|
||||||
async fn favicon() -> HttpResponse {
|
async fn favicon() -> HttpResponse {
|
||||||
HttpResponse::Ok().body("")
|
HttpResponse::Ok()
|
||||||
|
.content_type("image/x-icon")
|
||||||
|
.header("Cache-Control", "public, max-age=1209600, s-maxage=86400")
|
||||||
|
.body(include_bytes!("../static/favicon.ico").as_ref())
|
||||||
}
|
}
|
||||||
|
|
||||||
#[actix_web::main]
|
#[actix_web::main]
|
||||||
async fn main() -> std::io::Result<()> {
|
async fn main() -> std::io::Result<()> {
|
||||||
let args: Vec<String> = std::env::args().collect();
|
|
||||||
let mut address = "0.0.0.0:8080".to_string();
|
let mut address = "0.0.0.0:8080".to_string();
|
||||||
|
let mut force_https = false;
|
||||||
|
|
||||||
if args.len() > 1 {
|
for arg in std::env::args().collect::<Vec<String>>() {
|
||||||
for arg in args {
|
match arg.split('=').collect::<Vec<&str>>()[0] {
|
||||||
if arg.starts_with("--address=") || arg.starts_with("-a=") {
|
"--address" | "-a" => address = arg.split('=').collect::<Vec<&str>>()[1].to_string(),
|
||||||
let split: Vec<&str> = arg.split("=").collect();
|
"--redirect-https" | "-r" => force_https = true,
|
||||||
address = split[1].to_string();
|
_ => (),
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// start http server
|
// start http server
|
||||||
println!("Running Libreddit v{} on {}!", env!("CARGO_PKG_VERSION"), address.clone());
|
println!("Running Libreddit v{} on {}!", env!("CARGO_PKG_VERSION"), &address);
|
||||||
|
|
||||||
HttpServer::new(|| {
|
HttpServer::new(move || {
|
||||||
App::new()
|
App::new()
|
||||||
// TRAILING SLASH MIDDLEWARE
|
// Redirect to HTTPS if "--redirect-https" enabled
|
||||||
.wrap(NormalizePath::default())
|
.wrap_fn(move |req, srv| {
|
||||||
// GENERAL SERVICES
|
let secure = req.connection_info().scheme() == "https";
|
||||||
|
let https_url = format!("https://{}{}", req.connection_info().host(), req.uri().to_string());
|
||||||
|
srv.call(req).map(move |res: Result<ServiceResponse, _>| {
|
||||||
|
if force_https && !secure {
|
||||||
|
Ok(ServiceResponse::new(
|
||||||
|
res.unwrap().request().to_owned(),
|
||||||
|
HttpResponse::Found().header("Location", https_url).finish(),
|
||||||
|
))
|
||||||
|
} else {
|
||||||
|
res
|
||||||
|
}
|
||||||
|
})
|
||||||
|
})
|
||||||
|
// Append trailing slash and remove double slashes
|
||||||
|
.wrap(middleware::NormalizePath::default())
|
||||||
|
// Apply default headers for security
|
||||||
|
.wrap(
|
||||||
|
middleware::DefaultHeaders::new()
|
||||||
|
.header("Referrer-Policy", "no-referrer")
|
||||||
|
.header("X-Content-Type-Options", "nosniff")
|
||||||
|
.header("X-Frame-Options", "DENY")
|
||||||
|
.header(
|
||||||
|
"Content-Security-Policy",
|
||||||
|
"default-src 'none'; media-src 'self'; style-src 'self' 'unsafe-inline'; base-uri 'none'; img-src 'self' data:; form-action 'self'; frame-ancestors 'none';",
|
||||||
|
),
|
||||||
|
)
|
||||||
|
// Default service in case no routes match
|
||||||
|
.default_service(web::get().to(|| utils::error("Nothing here".to_string())))
|
||||||
|
// Read static files
|
||||||
.route("/style.css/", web::get().to(style))
|
.route("/style.css/", web::get().to(style))
|
||||||
.route("/favicon.ico/", web::get().to(|| HttpResponse::Ok()))
|
.route("/favicon.ico/", web::get().to(favicon))
|
||||||
.route("/robots.txt/", web::get().to(robots))
|
.route("/robots.txt/", web::get().to(robots))
|
||||||
// PROXY SERVICE
|
// Proxy media through Libreddit
|
||||||
.route("/proxy/{url:.*}/", web::get().to(proxy::handler))
|
.route("/proxy/{url:.*}/", web::get().to(proxy::handler))
|
||||||
// USER SERVICES
|
// Browse user profile
|
||||||
.route("/u/{username}/", web::get().to(user::page))
|
.service(
|
||||||
.route("/user/{username}/", web::get().to(user::page))
|
web::scope("/{scope:user|u}").service(
|
||||||
// SUBREDDIT SERVICES
|
web::scope("/{username}").route("/", web::get().to(user::profile)).service(
|
||||||
.route("/r/{sub}/", web::get().to(subreddit::page))
|
web::scope("/comments/{id}/{title}")
|
||||||
// POPULAR SERVICES
|
.route("/", web::get().to(post::item))
|
||||||
.route("/", web::get().to(popular::page))
|
.route("/{comment_id}/", web::get().to(post::item)),
|
||||||
// POST SERVICES
|
),
|
||||||
.route("/{id:.{5,6}}/", web::get().to(post::short))
|
),
|
||||||
.route("/r/{sub}/comments/{id}/{title}/", web::get().to(post::page))
|
)
|
||||||
|
// Configure settings
|
||||||
|
.service(web::resource("/settings/").route(web::get().to(settings::get)).route(web::post().to(settings::set)))
|
||||||
|
// Subreddit services
|
||||||
|
.service(
|
||||||
|
web::scope("/r/{sub}")
|
||||||
|
// See posts and info about subreddit
|
||||||
|
.route("/", web::get().to(subreddit::page))
|
||||||
|
.route("/{sort:hot|new|top|rising|controversial}/", web::get().to(subreddit::page))
|
||||||
|
// Handle subscribe/unsubscribe
|
||||||
|
.route("/{action:subscribe|unsubscribe}/", web::post().to(subreddit::subscriptions))
|
||||||
|
// View post on subreddit
|
||||||
|
.service(
|
||||||
|
web::scope("/comments/{id}/{title}")
|
||||||
|
.route("/", web::get().to(post::item))
|
||||||
|
.route("/{comment_id}/", web::get().to(post::item)),
|
||||||
|
)
|
||||||
|
// Search inside subreddit
|
||||||
|
.route("/search/", web::get().to(search::find))
|
||||||
|
// View wiki of subreddit
|
||||||
|
.service(
|
||||||
|
web::scope("/wiki")
|
||||||
|
.route("/", web::get().to(subreddit::wiki))
|
||||||
|
.route("/{page}/", web::get().to(subreddit::wiki)),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
// Front page
|
||||||
|
.route("/", web::get().to(subreddit::page))
|
||||||
|
.route("/{sort:best|hot|new|top|rising|controversial}/", web::get().to(subreddit::page))
|
||||||
|
// View Reddit wiki
|
||||||
|
.service(
|
||||||
|
web::scope("/wiki")
|
||||||
|
.route("/", web::get().to(subreddit::wiki))
|
||||||
|
.route("/{page}/", web::get().to(subreddit::wiki)),
|
||||||
|
)
|
||||||
|
// Search all of Reddit
|
||||||
|
.route("/search/", web::get().to(search::find))
|
||||||
|
// Short link for post
|
||||||
|
.route("/{id:.{5,6}}/", web::get().to(post::item))
|
||||||
})
|
})
|
||||||
.bind(address.clone())
|
.bind(&address)
|
||||||
.expect(format!("Cannot bind to the address: {}", address).as_str())
|
.unwrap_or_else(|e| panic!("Cannot bind to the address {}: {}", address, e))
|
||||||
.run()
|
.run()
|
||||||
.await
|
.await
|
||||||
}
|
}
|
||||||
|
@ -1,55 +0,0 @@
|
|||||||
// CRATES
|
|
||||||
use crate::utils::{fetch_posts, ErrorTemplate, Params, Post};
|
|
||||||
use actix_web::{http::StatusCode, web, HttpResponse, Result};
|
|
||||||
use askama::Template;
|
|
||||||
|
|
||||||
// STRUCTS
|
|
||||||
#[derive(Template)]
|
|
||||||
#[template(path = "popular.html", escape = "none")]
|
|
||||||
struct PopularTemplate {
|
|
||||||
posts: Vec<Post>,
|
|
||||||
sort: String,
|
|
||||||
ends: (String, String),
|
|
||||||
}
|
|
||||||
|
|
||||||
// RENDER
|
|
||||||
async fn render(sub_name: String, sort: Option<String>, ends: (Option<String>, Option<String>)) -> Result<HttpResponse> {
|
|
||||||
let sorting = sort.unwrap_or("hot".to_string());
|
|
||||||
let before = ends.1.clone().unwrap_or(String::new()); // If there is an after, there must be a before
|
|
||||||
|
|
||||||
// Build the Reddit JSON API url
|
|
||||||
let url = match ends.0 {
|
|
||||||
Some(val) => format!("https://www.reddit.com/r/{}/{}.json?before={}&count=25", sub_name, sorting, val),
|
|
||||||
None => match ends.1 {
|
|
||||||
Some(val) => format!("https://www.reddit.com/r/{}/{}.json?after={}&count=25", sub_name, sorting, val),
|
|
||||||
None => format!("https://www.reddit.com/r/{}/{}.json", sub_name, sorting),
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
let items_result = fetch_posts(url, String::new()).await;
|
|
||||||
|
|
||||||
if items_result.is_err() {
|
|
||||||
let s = ErrorTemplate {
|
|
||||||
message: items_result.err().unwrap().to_string(),
|
|
||||||
}
|
|
||||||
.render()
|
|
||||||
.unwrap();
|
|
||||||
Ok(HttpResponse::Ok().status(StatusCode::NOT_FOUND).content_type("text/html").body(s))
|
|
||||||
} else {
|
|
||||||
let items = items_result.unwrap();
|
|
||||||
|
|
||||||
let s = PopularTemplate {
|
|
||||||
posts: items.0,
|
|
||||||
sort: sorting,
|
|
||||||
ends: (before, items.1),
|
|
||||||
}
|
|
||||||
.render()
|
|
||||||
.unwrap();
|
|
||||||
Ok(HttpResponse::Ok().content_type("text/html").body(s))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// SERVICES
|
|
||||||
pub async fn page(params: web::Query<Params>) -> Result<HttpResponse> {
|
|
||||||
render("popular".to_string(), params.sort.clone(), (params.before.clone(), params.after.clone())).await
|
|
||||||
}
|
|
247
src/post.rs
247
src/post.rs
@ -1,12 +1,10 @@
|
|||||||
// CRATES
|
// CRATES
|
||||||
use crate::utils::{format_num, format_url, request, val, Comment, ErrorTemplate, Flair, Params, Post};
|
use crate::utils::*;
|
||||||
use actix_web::{http::StatusCode, web, HttpResponse, Result};
|
use actix_web::{HttpRequest, HttpResponse};
|
||||||
|
|
||||||
use async_recursion::async_recursion;
|
use async_recursion::async_recursion;
|
||||||
|
|
||||||
use askama::Template;
|
use askama::Template;
|
||||||
use chrono::{TimeZone, Utc};
|
|
||||||
use pulldown_cmark::{html, Options, Parser};
|
|
||||||
|
|
||||||
// STRUCTS
|
// STRUCTS
|
||||||
#[derive(Template)]
|
#[derive(Template)]
|
||||||
@ -15,157 +13,176 @@ struct PostTemplate {
|
|||||||
comments: Vec<Comment>,
|
comments: Vec<Comment>,
|
||||||
post: Post,
|
post: Post,
|
||||||
sort: String,
|
sort: String,
|
||||||
|
prefs: Preferences,
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn render(id: String, sort: String) -> Result<HttpResponse> {
|
pub async fn item(req: HttpRequest) -> HttpResponse {
|
||||||
// Log the post ID being fetched
|
// Build Reddit API path
|
||||||
dbg!(&id);
|
let mut path: String = format!("{}.json?{}&raw_json=1", req.path(), req.query_string());
|
||||||
|
|
||||||
// Build the Reddit JSON API url
|
// Set sort to sort query parameter
|
||||||
let url: String = format!("https://reddit.com/{}.json?sort={}", id, sort);
|
let mut sort: String = param(&path, "sort");
|
||||||
|
|
||||||
|
// Grab default comment sort method from Cookies
|
||||||
|
let default_sort = cookie(&req, "comment_sort");
|
||||||
|
|
||||||
|
// If there's no sort query but there's a default sort, set sort to default_sort
|
||||||
|
if sort.is_empty() && !default_sort.is_empty() {
|
||||||
|
sort = default_sort;
|
||||||
|
path = format!("{}.json?{}&sort={}&raw_json=1", req.path(), req.query_string(), sort);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Log the post ID being fetched in debug mode
|
||||||
|
#[cfg(debug_assertions)]
|
||||||
|
dbg!(req.match_info().get("id").unwrap_or(""));
|
||||||
|
|
||||||
// Send a request to the url, receive JSON in response
|
// Send a request to the url, receive JSON in response
|
||||||
let req = request(url).await;
|
match request(path).await {
|
||||||
|
// Otherwise, grab the JSON output from the request
|
||||||
|
Ok(res) => {
|
||||||
|
// Parse the JSON into Post and Comment structs
|
||||||
|
let post = parse_post(&res[0]).await;
|
||||||
|
let comments = parse_comments(&res[1]).await;
|
||||||
|
|
||||||
// If the Reddit API returns an error, exit and send error page to user
|
// Use the Post and Comment structs to generate a website to show users
|
||||||
if req.is_err() {
|
let s = PostTemplate {
|
||||||
let s = ErrorTemplate {
|
comments,
|
||||||
message: req.err().unwrap().to_string(),
|
post,
|
||||||
|
sort,
|
||||||
|
prefs: prefs(req),
|
||||||
|
}
|
||||||
|
.render()
|
||||||
|
.unwrap();
|
||||||
|
HttpResponse::Ok().content_type("text/html").body(s)
|
||||||
}
|
}
|
||||||
.render()
|
// If the Reddit API returns an error, exit and send error page to user
|
||||||
.unwrap();
|
Err(msg) => error(msg).await,
|
||||||
return Ok(HttpResponse::Ok().status(StatusCode::NOT_FOUND).content_type("text/html").body(s));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Otherwise, grab the JSON output from the request
|
|
||||||
let res = req.unwrap();
|
|
||||||
|
|
||||||
// Parse the JSON into Post and Comment structs
|
|
||||||
let post = parse_post(res[0].clone()).await;
|
|
||||||
let comments = parse_comments(res[1].clone()).await;
|
|
||||||
|
|
||||||
// Use the Post and Comment structs to generate a website to show users
|
|
||||||
let s = PostTemplate {
|
|
||||||
comments: comments.unwrap(),
|
|
||||||
post: post.unwrap(),
|
|
||||||
sort: sort,
|
|
||||||
}
|
|
||||||
.render()
|
|
||||||
.unwrap();
|
|
||||||
Ok(HttpResponse::Ok().content_type("text/html").body(s))
|
|
||||||
}
|
|
||||||
|
|
||||||
// SERVICES
|
|
||||||
pub async fn short(web::Path(id): web::Path<String>) -> Result<HttpResponse> {
|
|
||||||
render(id.to_string(), "confidence".to_string()).await
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn page(web::Path((_sub, id)): web::Path<(String, String)>, params: web::Query<Params>) -> Result<HttpResponse> {
|
|
||||||
match ¶ms.sort {
|
|
||||||
Some(sort) => render(id, sort.to_string()).await,
|
|
||||||
None => render(id, "confidence".to_string()).await,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// UTILITIES
|
|
||||||
async fn media(data: &serde_json::Value) -> (String, String) {
|
|
||||||
let post_type: &str;
|
|
||||||
let url = if !data["preview"]["reddit_video_preview"]["fallback_url"].is_null() {
|
|
||||||
post_type = "video";
|
|
||||||
format_url(data["preview"]["reddit_video_preview"]["fallback_url"].as_str().unwrap()).await
|
|
||||||
} else if !data["secure_media"]["reddit_video"]["fallback_url"].is_null() {
|
|
||||||
post_type = "video";
|
|
||||||
format_url(data["secure_media"]["reddit_video"]["fallback_url"].as_str().unwrap()).await
|
|
||||||
} else if data["post_hint"].as_str().unwrap_or("") == "image" {
|
|
||||||
post_type = "image";
|
|
||||||
format_url(data["preview"]["images"][0]["source"]["url"].as_str().unwrap()).await
|
|
||||||
} else {
|
|
||||||
post_type = "link";
|
|
||||||
data["url"].as_str().unwrap().to_string()
|
|
||||||
};
|
|
||||||
|
|
||||||
(post_type.to_string(), url)
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn markdown_to_html(md: &str) -> String {
|
|
||||||
let mut options = Options::empty();
|
|
||||||
options.insert(Options::ENABLE_TABLES);
|
|
||||||
options.insert(Options::ENABLE_FOOTNOTES);
|
|
||||||
options.insert(Options::ENABLE_STRIKETHROUGH);
|
|
||||||
options.insert(Options::ENABLE_TASKLISTS);
|
|
||||||
let parser = Parser::new_ext(md, options);
|
|
||||||
|
|
||||||
// Write to String buffer.
|
|
||||||
let mut html_output = String::new();
|
|
||||||
html::push_html(&mut html_output, parser);
|
|
||||||
html_output
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// POSTS
|
// POSTS
|
||||||
async fn parse_post(json: serde_json::Value) -> Result<Post, &'static str> {
|
async fn parse_post(json: &serde_json::Value) -> Post {
|
||||||
let post_data: &serde_json::Value = &json["data"]["children"][0];
|
// Retrieve post (as opposed to comments) from JSON
|
||||||
|
let post: &serde_json::Value = &json["data"]["children"][0];
|
||||||
|
|
||||||
let unix_time: i64 = post_data["data"]["created_utc"].as_f64().unwrap().round() as i64;
|
// Grab UTC time as unix timestamp
|
||||||
let score = post_data["data"]["score"].as_i64().unwrap();
|
let (rel_time, created) = time(post["data"]["created_utc"].as_f64().unwrap_or_default());
|
||||||
|
// Parse post score and upvote ratio
|
||||||
|
let score = post["data"]["score"].as_i64().unwrap_or_default();
|
||||||
|
let ratio: f64 = post["data"]["upvote_ratio"].as_f64().unwrap_or(1.0) * 100.0;
|
||||||
|
|
||||||
let media = media(&post_data["data"]).await;
|
// Determine the type of media along with the media URL
|
||||||
|
let (post_type, media) = media(&post["data"]).await;
|
||||||
|
|
||||||
let post = Post {
|
// Build a post using data parsed from Reddit post API
|
||||||
title: val(post_data, "title").await,
|
Post {
|
||||||
community: val(post_data, "subreddit").await,
|
id: val(post, "id"),
|
||||||
body: markdown_to_html(post_data["data"]["selftext"].as_str().unwrap()).await,
|
title: val(post, "title"),
|
||||||
author: val(post_data, "author").await,
|
community: val(post, "subreddit"),
|
||||||
url: val(post_data, "permalink").await,
|
body: rewrite_url(&val(post, "selftext_html")),
|
||||||
|
author: Author {
|
||||||
|
name: val(post, "author"),
|
||||||
|
flair: Flair {
|
||||||
|
flair_parts: parse_rich_flair(
|
||||||
|
val(post, "author_flair_type"),
|
||||||
|
post["data"]["author_flair_richtext"].as_array(),
|
||||||
|
post["data"]["author_flair_text"].as_str(),
|
||||||
|
),
|
||||||
|
background_color: val(post, "author_flair_background_color"),
|
||||||
|
foreground_color: val(post, "author_flair_text_color"),
|
||||||
|
},
|
||||||
|
distinguished: val(post, "distinguished"),
|
||||||
|
},
|
||||||
|
permalink: val(post, "permalink"),
|
||||||
score: format_num(score),
|
score: format_num(score),
|
||||||
post_type: media.0,
|
upvote_ratio: ratio as i64,
|
||||||
media: media.1,
|
post_type,
|
||||||
time: Utc.timestamp(unix_time, 0).format("%b %e %Y %H:%M UTC").to_string(),
|
media,
|
||||||
flair: Flair(
|
thumbnail: Media {
|
||||||
val(post_data, "link_flair_text").await,
|
url: format_url(val(post, "thumbnail").as_str()),
|
||||||
val(post_data, "link_flair_background_color").await,
|
width: post["data"]["thumbnail_width"].as_i64().unwrap_or_default(),
|
||||||
if val(post_data, "link_flair_text_color").await == "dark" {
|
height: post["data"]["thumbnail_height"].as_i64().unwrap_or_default(),
|
||||||
|
},
|
||||||
|
flair: Flair {
|
||||||
|
flair_parts: parse_rich_flair(
|
||||||
|
val(post, "link_flair_type"),
|
||||||
|
post["data"]["link_flair_richtext"].as_array(),
|
||||||
|
post["data"]["link_flair_text"].as_str(),
|
||||||
|
),
|
||||||
|
background_color: val(post, "link_flair_background_color"),
|
||||||
|
foreground_color: if val(post, "link_flair_text_color") == "dark" {
|
||||||
"black".to_string()
|
"black".to_string()
|
||||||
} else {
|
} else {
|
||||||
"white".to_string()
|
"white".to_string()
|
||||||
},
|
},
|
||||||
),
|
},
|
||||||
};
|
flags: Flags {
|
||||||
|
nsfw: post["data"]["over_18"].as_bool().unwrap_or(false),
|
||||||
Ok(post)
|
stickied: post["data"]["stickied"].as_bool().unwrap_or(false),
|
||||||
|
},
|
||||||
|
domain: val(post, "domain"),
|
||||||
|
rel_time,
|
||||||
|
created,
|
||||||
|
comments: format_num(post["data"]["num_comments"].as_i64().unwrap_or_default()),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// COMMENTS
|
// COMMENTS
|
||||||
#[async_recursion]
|
#[async_recursion]
|
||||||
async fn parse_comments(json: serde_json::Value) -> Result<Vec<Comment>, &'static str> {
|
async fn parse_comments(json: &serde_json::Value) -> Vec<Comment> {
|
||||||
// Separate the comment JSON into a Vector of comments
|
// Separate the comment JSON into a Vector of comments
|
||||||
let comment_data = json["data"]["children"].as_array().unwrap();
|
let comment_data = match json["data"]["children"].as_array() {
|
||||||
|
Some(f) => f.to_owned(),
|
||||||
|
None => Vec::new(),
|
||||||
|
};
|
||||||
|
|
||||||
let mut comments: Vec<Comment> = Vec::new();
|
let mut comments: Vec<Comment> = Vec::new();
|
||||||
|
|
||||||
// For each comment, retrieve the values to build a Comment object
|
// For each comment, retrieve the values to build a Comment object
|
||||||
for comment in comment_data.iter() {
|
for comment in comment_data {
|
||||||
let unix_time: i64 = comment["data"]["created_utc"].as_f64().unwrap_or(0.0).round() as i64;
|
let unix_time = comment["data"]["created_utc"].as_f64().unwrap_or_default();
|
||||||
if unix_time == 0 {
|
if unix_time == 0.0 {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
let (rel_time, created) = time(unix_time);
|
||||||
|
|
||||||
let score = comment["data"]["score"].as_i64().unwrap_or(0);
|
let score = comment["data"]["score"].as_i64().unwrap_or(0);
|
||||||
let body = markdown_to_html(comment["data"]["body"].as_str().unwrap_or("")).await;
|
let body = rewrite_url(&val(&comment, "body_html"));
|
||||||
|
|
||||||
let replies: Vec<Comment> = if comment["data"]["replies"].is_object() {
|
let replies: Vec<Comment> = if comment["data"]["replies"].is_object() {
|
||||||
parse_comments(comment["data"]["replies"].clone()).await.unwrap_or(Vec::new())
|
parse_comments(&comment["data"]["replies"]).await
|
||||||
} else {
|
} else {
|
||||||
Vec::new()
|
Vec::new()
|
||||||
};
|
};
|
||||||
|
|
||||||
|
dbg!();
|
||||||
|
|
||||||
comments.push(Comment {
|
comments.push(Comment {
|
||||||
body: body,
|
id: val(&comment, "id"),
|
||||||
author: val(comment, "author").await,
|
body,
|
||||||
score: format_num(score),
|
author: Author {
|
||||||
time: Utc.timestamp(unix_time, 0).format("%b %e %Y %H:%M UTC").to_string(),
|
name: val(&comment, "author"),
|
||||||
replies: replies,
|
flair: Flair {
|
||||||
|
flair_parts: parse_rich_flair(
|
||||||
|
val(&comment, "author_flair_type"),
|
||||||
|
comment["data"]["author_flair_richtext"].as_array(),
|
||||||
|
comment["data"]["author_flair_text"].as_str(),
|
||||||
|
),
|
||||||
|
background_color: val(&comment, "author_flair_background_color"),
|
||||||
|
foreground_color: val(&comment, "author_flair_text_color"),
|
||||||
|
},
|
||||||
|
distinguished: val(&comment, "distinguished"),
|
||||||
|
},
|
||||||
|
score: if comment["data"]["score_hidden"].as_bool().unwrap_or_default() {
|
||||||
|
"•".to_string()
|
||||||
|
} else {
|
||||||
|
format_num(score)
|
||||||
|
},
|
||||||
|
rel_time,
|
||||||
|
created,
|
||||||
|
replies,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(comments)
|
comments
|
||||||
}
|
}
|
||||||
|
60
src/proxy.rs
60
src/proxy.rs
@ -1,29 +1,47 @@
|
|||||||
use actix_web::{client::Client, web, Error, HttpResponse, Result};
|
use actix_web::{client::Client, error, web, Error, HttpResponse, Result};
|
||||||
|
use url::Url;
|
||||||
|
|
||||||
#[cfg(feature = "proxy")]
|
|
||||||
use base64::decode;
|
use base64::decode;
|
||||||
|
|
||||||
pub async fn handler(web::Path(url): web::Path<String>) -> Result<HttpResponse> {
|
pub async fn handler(web::Path(b64): web::Path<String>) -> Result<HttpResponse> {
|
||||||
if cfg!(feature = "proxy") {
|
let domains = vec![
|
||||||
let media: String;
|
// THUMBNAILS
|
||||||
|
"a.thumbs.redditmedia.com",
|
||||||
|
"b.thumbs.redditmedia.com",
|
||||||
|
// EMOJI
|
||||||
|
"emoji.redditmedia.com",
|
||||||
|
// ICONS
|
||||||
|
"styles.redditmedia.com",
|
||||||
|
"www.redditstatic.com",
|
||||||
|
// PREVIEWS
|
||||||
|
"preview.redd.it",
|
||||||
|
"external-preview.redd.it",
|
||||||
|
// MEDIA
|
||||||
|
"i.redd.it",
|
||||||
|
"v.redd.it",
|
||||||
|
];
|
||||||
|
|
||||||
#[cfg(not(feature = "proxy"))]
|
let decoded = decode(b64).map(|bytes| String::from_utf8(bytes).unwrap_or_default());
|
||||||
let media = url;
|
|
||||||
|
|
||||||
#[cfg(feature = "proxy")]
|
match decoded {
|
||||||
match decode(url) {
|
Ok(media) => match Url::parse(media.as_str()) {
|
||||||
Ok(bytes) => media = String::from_utf8(bytes).unwrap(),
|
Ok(url) => {
|
||||||
Err(_e) => return Ok(HttpResponse::Ok().body("")),
|
let domain = url.domain().unwrap_or_default();
|
||||||
};
|
|
||||||
|
|
||||||
let client = Client::default();
|
if domains.contains(&domain) {
|
||||||
client
|
Client::default().get(media.replace("&", "&")).send().await.map_err(Error::from).map(|res| {
|
||||||
.get(media.replace("&", "&"))
|
HttpResponse::build(res.status())
|
||||||
.send()
|
.header("Cache-Control", "public, max-age=1209600, s-maxage=86400")
|
||||||
.await
|
.header("Content-Length", res.headers().get("Content-Length").unwrap().to_owned())
|
||||||
.map_err(Error::from)
|
.header("Content-Type", res.headers().get("Content-Type").unwrap().to_owned())
|
||||||
.and_then(|res| Ok(HttpResponse::build(res.status()).streaming(res)))
|
.streaming(res)
|
||||||
} else {
|
})
|
||||||
Ok(HttpResponse::Ok().body(""))
|
} else {
|
||||||
|
Err(error::ErrorForbidden("Resource must be from Reddit"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => Err(error::ErrorBadRequest("Can't parse base64 into URL")),
|
||||||
|
},
|
||||||
|
_ => Err(error::ErrorBadRequest("Can't decode base64")),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
99
src/search.rs
Normal file
99
src/search.rs
Normal file
@ -0,0 +1,99 @@
|
|||||||
|
// CRATES
|
||||||
|
use crate::utils::{cookie, error, fetch_posts, param, prefs, request, val, Post, Preferences};
|
||||||
|
use actix_web::{HttpRequest, HttpResponse};
|
||||||
|
use askama::Template;
|
||||||
|
|
||||||
|
// STRUCTS
|
||||||
|
struct SearchParams {
|
||||||
|
q: String,
|
||||||
|
sort: String,
|
||||||
|
t: String,
|
||||||
|
before: String,
|
||||||
|
after: String,
|
||||||
|
restrict_sr: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
// STRUCTS
|
||||||
|
struct Subreddit {
|
||||||
|
name: String,
|
||||||
|
url: String,
|
||||||
|
description: String,
|
||||||
|
subscribers: i64,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Template)]
|
||||||
|
#[template(path = "search.html", escape = "none")]
|
||||||
|
struct SearchTemplate {
|
||||||
|
posts: Vec<Post>,
|
||||||
|
subreddits: Vec<Subreddit>,
|
||||||
|
sub: String,
|
||||||
|
params: SearchParams,
|
||||||
|
prefs: Preferences,
|
||||||
|
}
|
||||||
|
|
||||||
|
// SERVICES
|
||||||
|
pub async fn find(req: HttpRequest) -> HttpResponse {
|
||||||
|
let nsfw_results = if cookie(&req, "show_nsfw") == "on" { "&include_over_18=on" } else { "" };
|
||||||
|
let path = format!("{}.json?{}{}", req.path(), req.query_string(), nsfw_results);
|
||||||
|
let sub = req.match_info().get("sub").unwrap_or("").to_string();
|
||||||
|
|
||||||
|
let sort = if param(&path, "sort").is_empty() {
|
||||||
|
"relevance".to_string()
|
||||||
|
} else {
|
||||||
|
param(&path, "sort")
|
||||||
|
};
|
||||||
|
|
||||||
|
let subreddits = if param(&path, "restrict_sr").is_empty() {
|
||||||
|
search_subreddits(param(&path, "q")).await
|
||||||
|
} else {
|
||||||
|
Vec::new()
|
||||||
|
};
|
||||||
|
|
||||||
|
match fetch_posts(&path, String::new()).await {
|
||||||
|
Ok((posts, after)) => HttpResponse::Ok().content_type("text/html").body(
|
||||||
|
SearchTemplate {
|
||||||
|
posts,
|
||||||
|
subreddits,
|
||||||
|
sub,
|
||||||
|
params: SearchParams {
|
||||||
|
q: param(&path, "q"),
|
||||||
|
sort,
|
||||||
|
t: param(&path, "t"),
|
||||||
|
before: param(&path, "after"),
|
||||||
|
after,
|
||||||
|
restrict_sr: param(&path, "restrict_sr"),
|
||||||
|
},
|
||||||
|
prefs: prefs(req),
|
||||||
|
}
|
||||||
|
.render()
|
||||||
|
.unwrap(),
|
||||||
|
),
|
||||||
|
Err(msg) => error(msg).await,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn search_subreddits(q: String) -> Vec<Subreddit> {
|
||||||
|
let subreddit_search_path = format!("/subreddits/search.json?q={}&limit=3", q.replace(' ', "+"));
|
||||||
|
|
||||||
|
// Send a request to the url
|
||||||
|
match request(subreddit_search_path).await {
|
||||||
|
// If success, receive JSON in response
|
||||||
|
Ok(response) => {
|
||||||
|
match response["data"]["children"].as_array() {
|
||||||
|
// For each subreddit from subreddit list
|
||||||
|
Some(list) => list
|
||||||
|
.iter()
|
||||||
|
.map(|subreddit| Subreddit {
|
||||||
|
name: val(subreddit, "display_name_prefixed"),
|
||||||
|
url: val(subreddit, "url"),
|
||||||
|
description: val(subreddit, "public_description"),
|
||||||
|
subscribers: subreddit["data"]["subscribers"].as_u64().unwrap_or_default() as i64,
|
||||||
|
})
|
||||||
|
.collect::<Vec<Subreddit>>(),
|
||||||
|
_ => Vec::new(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// If the Reddit API returns an error, exit this function
|
||||||
|
_ => Vec::new(),
|
||||||
|
}
|
||||||
|
}
|
56
src/settings.rs
Normal file
56
src/settings.rs
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
// CRATES
|
||||||
|
use crate::utils::{prefs, Preferences};
|
||||||
|
use actix_web::{cookie::Cookie, web::Form, HttpRequest, HttpResponse};
|
||||||
|
use askama::Template;
|
||||||
|
use time::{Duration, OffsetDateTime};
|
||||||
|
|
||||||
|
// STRUCTS
|
||||||
|
#[derive(Template)]
|
||||||
|
#[template(path = "settings.html")]
|
||||||
|
struct SettingsTemplate {
|
||||||
|
prefs: Preferences,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(serde::Deserialize)]
|
||||||
|
pub struct SettingsForm {
|
||||||
|
theme: Option<String>,
|
||||||
|
front_page: Option<String>,
|
||||||
|
layout: Option<String>,
|
||||||
|
wide: Option<String>,
|
||||||
|
comment_sort: Option<String>,
|
||||||
|
show_nsfw: Option<String>,
|
||||||
|
}
|
||||||
|
|
||||||
|
// FUNCTIONS
|
||||||
|
|
||||||
|
// Retrieve cookies from request "Cookie" header
|
||||||
|
pub async fn get(req: HttpRequest) -> HttpResponse {
|
||||||
|
let s = SettingsTemplate { prefs: prefs(req) }.render().unwrap();
|
||||||
|
HttpResponse::Ok().content_type("text/html").body(s)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set cookies using response "Set-Cookie" header
|
||||||
|
pub async fn set(_req: HttpRequest, form: Form<SettingsForm>) -> HttpResponse {
|
||||||
|
let mut res = HttpResponse::Found();
|
||||||
|
|
||||||
|
let names = vec!["theme", "front_page", "layout", "wide", "comment_sort", "show_nsfw"];
|
||||||
|
let values = vec![&form.theme, &form.front_page, &form.layout, &form.wide, &form.comment_sort, &form.show_nsfw];
|
||||||
|
|
||||||
|
for (i, name) in names.iter().enumerate() {
|
||||||
|
match values[i] {
|
||||||
|
Some(value) => res.cookie(
|
||||||
|
Cookie::build(name.to_owned(), value)
|
||||||
|
.path("/")
|
||||||
|
.http_only(true)
|
||||||
|
.expires(OffsetDateTime::now_utc() + Duration::weeks(52))
|
||||||
|
.finish(),
|
||||||
|
),
|
||||||
|
None => res.del_cookie(&Cookie::named(name.to_owned())),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
res
|
||||||
|
.content_type("text/html")
|
||||||
|
.set_header("Location", "/settings")
|
||||||
|
.body(r#"Redirecting to <a href="/settings">settings</a>..."#)
|
||||||
|
}
|
224
src/subreddit.rs
224
src/subreddit.rs
@ -1,8 +1,8 @@
|
|||||||
// CRATES
|
// CRATES
|
||||||
use crate::utils::{fetch_posts, format_num, format_url, request, val, ErrorTemplate, Params, Post, Subreddit};
|
use crate::utils::*;
|
||||||
use actix_web::{http::StatusCode, web, HttpResponse, Result};
|
use actix_web::{cookie::Cookie, HttpRequest, HttpResponse, Result};
|
||||||
use askama::Template;
|
use askama::Template;
|
||||||
use std::convert::TryInto;
|
use time::{Duration, OffsetDateTime};
|
||||||
|
|
||||||
// STRUCTS
|
// STRUCTS
|
||||||
#[derive(Template)]
|
#[derive(Template)]
|
||||||
@ -10,88 +10,176 @@ use std::convert::TryInto;
|
|||||||
struct SubredditTemplate {
|
struct SubredditTemplate {
|
||||||
sub: Subreddit,
|
sub: Subreddit,
|
||||||
posts: Vec<Post>,
|
posts: Vec<Post>,
|
||||||
sort: String,
|
sort: (String, String),
|
||||||
ends: (String, String),
|
ends: (String, String),
|
||||||
|
prefs: Preferences,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Template)]
|
||||||
|
#[template(path = "wiki.html", escape = "none")]
|
||||||
|
struct WikiTemplate {
|
||||||
|
sub: String,
|
||||||
|
wiki: String,
|
||||||
|
page: String,
|
||||||
|
prefs: Preferences,
|
||||||
}
|
}
|
||||||
|
|
||||||
// SERVICES
|
// SERVICES
|
||||||
#[allow(dead_code)]
|
pub async fn page(req: HttpRequest) -> HttpResponse {
|
||||||
pub async fn page(web::Path(sub): web::Path<String>, params: web::Query<Params>) -> Result<HttpResponse> {
|
let subscribed = cookie(&req, "subscriptions");
|
||||||
render(sub, params.sort.clone(), (params.before.clone(), params.after.clone())).await
|
let front_page = cookie(&req, "front_page");
|
||||||
|
let sort = req.match_info().get("sort").unwrap_or("hot").to_string();
|
||||||
|
|
||||||
|
let sub = req
|
||||||
|
.match_info()
|
||||||
|
.get("sub")
|
||||||
|
.map(String::from)
|
||||||
|
.unwrap_or(if front_page == "default" || front_page.is_empty() {
|
||||||
|
if subscribed.is_empty() {
|
||||||
|
"popular".to_string()
|
||||||
|
} else {
|
||||||
|
subscribed.to_owned()
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
front_page.to_owned()
|
||||||
|
});
|
||||||
|
|
||||||
|
let path = format!("/r/{}/{}.json?{}", sub, sort, req.query_string());
|
||||||
|
|
||||||
|
match fetch_posts(&path, String::new()).await {
|
||||||
|
Ok((posts, after)) => {
|
||||||
|
// If you can get subreddit posts, also request subreddit metadata
|
||||||
|
let sub = if !sub.contains('+') && sub != subscribed && sub != "popular" && sub != "all" {
|
||||||
|
// Regular subreddit
|
||||||
|
subreddit(&sub).await.unwrap_or_default()
|
||||||
|
} else if sub == subscribed {
|
||||||
|
// Subscription feed
|
||||||
|
if req.path().starts_with("/r/") {
|
||||||
|
subreddit(&sub).await.unwrap_or_default()
|
||||||
|
} else {
|
||||||
|
Subreddit::default()
|
||||||
|
}
|
||||||
|
} else if sub.contains('+') {
|
||||||
|
// Multireddit
|
||||||
|
Subreddit {
|
||||||
|
name: sub,
|
||||||
|
..Subreddit::default()
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Subreddit::default()
|
||||||
|
};
|
||||||
|
|
||||||
|
let s = SubredditTemplate {
|
||||||
|
sub,
|
||||||
|
posts,
|
||||||
|
sort: (sort, param(&path, "t")),
|
||||||
|
ends: (param(&path, "after"), after),
|
||||||
|
prefs: prefs(req),
|
||||||
|
}
|
||||||
|
.render()
|
||||||
|
.unwrap();
|
||||||
|
HttpResponse::Ok().content_type("text/html").body(s)
|
||||||
|
}
|
||||||
|
Err(msg) => error(msg).await,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn render(sub_name: String, sort: Option<String>, ends: (Option<String>, Option<String>)) -> Result<HttpResponse> {
|
// Sub or unsub by setting subscription cookie using response "Set-Cookie" header
|
||||||
let sorting = sort.unwrap_or("hot".to_string());
|
pub async fn subscriptions(req: HttpRequest) -> HttpResponse {
|
||||||
let before = ends.1.clone().unwrap_or(String::new()); // If there is an after, there must be a before
|
let mut res = HttpResponse::Found();
|
||||||
|
|
||||||
// Build the Reddit JSON API url
|
let sub = req.match_info().get("sub").unwrap_or_default().to_string();
|
||||||
let url = match ends.0 {
|
let action = req.match_info().get("action").unwrap_or_default().to_string();
|
||||||
Some(val) => format!("https://www.reddit.com/r/{}/{}.json?before={}&count=25", sub_name, sorting, val),
|
let mut sub_list = prefs(req.to_owned()).subs;
|
||||||
None => match ends.1 {
|
|
||||||
Some(val) => format!("https://www.reddit.com/r/{}/{}.json?after={}&count=25", sub_name, sorting, val),
|
// Modify sub list based on action
|
||||||
None => format!("https://www.reddit.com/r/{}/{}.json", sub_name, sorting),
|
if action == "subscribe" && !sub_list.contains(&sub) {
|
||||||
},
|
sub_list.push(sub.to_owned());
|
||||||
|
sub_list.sort();
|
||||||
|
} else if action == "unsubscribe" {
|
||||||
|
sub_list.retain(|s| s != &sub);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Delete cookie if empty, else set
|
||||||
|
if sub_list.is_empty() {
|
||||||
|
res.del_cookie(&Cookie::build("subscriptions", "").path("/").finish());
|
||||||
|
} else {
|
||||||
|
res.cookie(
|
||||||
|
Cookie::build("subscriptions", sub_list.join("+"))
|
||||||
|
.path("/")
|
||||||
|
.http_only(true)
|
||||||
|
.expires(OffsetDateTime::now_utc() + Duration::weeks(52))
|
||||||
|
.finish(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Redirect back to subreddit
|
||||||
|
// check for redirect parameter if unsubscribing from outside sidebar
|
||||||
|
let redirect_path = param(&req.uri().to_string(), "redirect");
|
||||||
|
let path = if !redirect_path.is_empty() && redirect_path.starts_with('/') {
|
||||||
|
redirect_path
|
||||||
|
} else {
|
||||||
|
format!("/r/{}", sub)
|
||||||
};
|
};
|
||||||
|
|
||||||
let sub_result = subreddit(&sub_name).await;
|
res
|
||||||
let items_result = fetch_posts(url, String::new()).await;
|
.content_type("text/html")
|
||||||
|
.set_header("Location", path.to_owned())
|
||||||
|
.body(format!("Redirecting to <a href=\"{0}\">{0}</a>...", path))
|
||||||
|
}
|
||||||
|
|
||||||
if sub_result.is_err() || items_result.is_err() {
|
pub async fn wiki(req: HttpRequest) -> HttpResponse {
|
||||||
let s = ErrorTemplate {
|
let sub = req.match_info().get("sub").unwrap_or("reddit.com").to_string();
|
||||||
message: sub_result.err().unwrap().to_string(),
|
let page = req.match_info().get("page").unwrap_or("index").to_string();
|
||||||
|
let path: String = format!("/r/{}/wiki/{}.json?raw_json=1", sub, page);
|
||||||
|
|
||||||
|
match request(path).await {
|
||||||
|
Ok(res) => {
|
||||||
|
let s = WikiTemplate {
|
||||||
|
sub,
|
||||||
|
wiki: rewrite_url(res["data"]["content_html"].as_str().unwrap_or_default()),
|
||||||
|
page,
|
||||||
|
prefs: prefs(req),
|
||||||
|
}
|
||||||
|
.render()
|
||||||
|
.unwrap();
|
||||||
|
HttpResponse::Ok().content_type("text/html").body(s)
|
||||||
}
|
}
|
||||||
.render()
|
Err(msg) => error(msg).await,
|
||||||
.unwrap();
|
|
||||||
Ok(HttpResponse::Ok().status(StatusCode::NOT_FOUND).content_type("text/html").body(s))
|
|
||||||
} else {
|
|
||||||
let mut sub = sub_result.unwrap();
|
|
||||||
let items = items_result.unwrap();
|
|
||||||
|
|
||||||
sub.icon = if sub.icon != "" {
|
|
||||||
format!(r#"<img class="subreddit_icon" src="{}">"#, sub.icon)
|
|
||||||
} else {
|
|
||||||
String::new()
|
|
||||||
};
|
|
||||||
|
|
||||||
let s = SubredditTemplate {
|
|
||||||
sub: sub,
|
|
||||||
posts: items.0,
|
|
||||||
sort: sorting,
|
|
||||||
ends: (before, items.1),
|
|
||||||
}
|
|
||||||
.render()
|
|
||||||
.unwrap();
|
|
||||||
Ok(HttpResponse::Ok().content_type("text/html").body(s))
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// SUBREDDIT
|
// SUBREDDIT
|
||||||
async fn subreddit(sub: &String) -> Result<Subreddit, &'static str> {
|
async fn subreddit(sub: &str) -> Result<Subreddit, String> {
|
||||||
// Build the Reddit JSON API url
|
// Build the Reddit JSON API url
|
||||||
let url: String = format!("https://www.reddit.com/r/{}/about.json", sub);
|
let path: String = format!("/r/{}/about.json?raw_json=1", sub);
|
||||||
|
|
||||||
// Send a request to the url, receive JSON in response
|
// Send a request to the url
|
||||||
let req = request(url).await;
|
match request(path).await {
|
||||||
|
// If success, receive JSON in response
|
||||||
|
Ok(res) => {
|
||||||
|
// Metadata regarding the subreddit
|
||||||
|
let members: i64 = res["data"]["subscribers"].as_u64().unwrap_or_default() as i64;
|
||||||
|
let active: i64 = res["data"]["accounts_active"].as_u64().unwrap_or_default() as i64;
|
||||||
|
|
||||||
// If the Reddit API returns an error, exit this function
|
// Fetch subreddit icon either from the community_icon or icon_img value
|
||||||
if req.is_err() {
|
let community_icon: &str = res["data"]["community_icon"].as_str().map_or("", |s| s.split('?').collect::<Vec<&str>>()[0]);
|
||||||
return Err(req.err().unwrap());
|
let icon = if community_icon.is_empty() { val(&res, "icon_img") } else { community_icon.to_string() };
|
||||||
|
|
||||||
|
let sub = Subreddit {
|
||||||
|
name: val(&res, "display_name"),
|
||||||
|
title: val(&res, "title"),
|
||||||
|
description: val(&res, "public_description"),
|
||||||
|
info: rewrite_url(&val(&res, "description_html").replace("\\", "")),
|
||||||
|
icon: format_url(icon.as_str()),
|
||||||
|
members: format_num(members),
|
||||||
|
active: format_num(active),
|
||||||
|
wiki: res["data"]["wiki_enabled"].as_bool().unwrap_or_default(),
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(sub)
|
||||||
|
}
|
||||||
|
// If the Reddit API returns an error, exit this function
|
||||||
|
Err(msg) => return Err(msg),
|
||||||
}
|
}
|
||||||
|
|
||||||
// Otherwise, grab the JSON output from the request
|
|
||||||
let res = req.unwrap();
|
|
||||||
|
|
||||||
let members = res["data"]["subscribers"].as_u64().unwrap_or(0);
|
|
||||||
let active = res["data"]["accounts_active"].as_u64().unwrap_or(0);
|
|
||||||
|
|
||||||
let sub = Subreddit {
|
|
||||||
name: val(&res, "display_name").await,
|
|
||||||
title: val(&res, "title").await,
|
|
||||||
description: val(&res, "public_description").await,
|
|
||||||
icon: format_url(val(&res, "icon_img").await.as_str()).await,
|
|
||||||
members: format_num(members.try_into().unwrap()),
|
|
||||||
active: format_num(active.try_into().unwrap()),
|
|
||||||
};
|
|
||||||
|
|
||||||
Ok(sub)
|
|
||||||
}
|
}
|
||||||
|
109
src/user.rs
109
src/user.rs
@ -1,7 +1,8 @@
|
|||||||
// CRATES
|
// CRATES
|
||||||
use crate::utils::{fetch_posts, nested_val, request, ErrorTemplate, Params, Post, User};
|
use crate::utils::{error, fetch_posts, format_url, param, prefs, request, Post, Preferences, User};
|
||||||
use actix_web::{http::StatusCode, web, HttpResponse, Result};
|
use actix_web::{HttpRequest, HttpResponse, Result};
|
||||||
use askama::Template;
|
use askama::Template;
|
||||||
|
use time::OffsetDateTime;
|
||||||
|
|
||||||
// STRUCTS
|
// STRUCTS
|
||||||
#[derive(Template)]
|
#[derive(Template)]
|
||||||
@ -9,65 +10,71 @@ use askama::Template;
|
|||||||
struct UserTemplate {
|
struct UserTemplate {
|
||||||
user: User,
|
user: User,
|
||||||
posts: Vec<Post>,
|
posts: Vec<Post>,
|
||||||
sort: String,
|
sort: (String, String),
|
||||||
|
ends: (String, String),
|
||||||
|
prefs: Preferences,
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn render(username: String, sort: String) -> Result<HttpResponse> {
|
// FUNCTIONS
|
||||||
// Build the Reddit JSON API url
|
pub async fn profile(req: HttpRequest) -> HttpResponse {
|
||||||
let url: String = format!("https://www.reddit.com/user/{}/.json?sort={}", username, sort);
|
// Build the Reddit JSON API path
|
||||||
|
let path = format!("{}.json?{}&raw_json=1", req.path(), req.query_string());
|
||||||
|
|
||||||
let user = user(&username).await;
|
// Retrieve other variables from Libreddit request
|
||||||
let posts = fetch_posts(url, "Comment".to_string()).await;
|
let sort = param(&path, "sort");
|
||||||
|
let username = req.match_info().get("username").unwrap_or("").to_string();
|
||||||
|
|
||||||
if user.is_err() || posts.is_err() {
|
// Request user posts/comments from Reddit
|
||||||
let s = ErrorTemplate {
|
let posts = fetch_posts(&path, "Comment".to_string()).await;
|
||||||
message: user.err().unwrap().to_string(),
|
|
||||||
|
match posts {
|
||||||
|
Ok((posts, after)) => {
|
||||||
|
// If you can get user posts, also request user data
|
||||||
|
let user = user(&username).await.unwrap_or_default();
|
||||||
|
|
||||||
|
let s = UserTemplate {
|
||||||
|
user,
|
||||||
|
posts,
|
||||||
|
sort: (sort, param(&path, "t")),
|
||||||
|
ends: (param(&path, "after"), after),
|
||||||
|
prefs: prefs(req),
|
||||||
|
}
|
||||||
|
.render()
|
||||||
|
.unwrap();
|
||||||
|
HttpResponse::Ok().content_type("text/html").body(s)
|
||||||
}
|
}
|
||||||
.render()
|
// If there is an error show error page
|
||||||
.unwrap();
|
Err(msg) => error(msg).await,
|
||||||
Ok(HttpResponse::Ok().status(StatusCode::NOT_FOUND).content_type("text/html").body(s))
|
|
||||||
} else {
|
|
||||||
let s = UserTemplate {
|
|
||||||
user: user.unwrap(),
|
|
||||||
posts: posts.unwrap().0,
|
|
||||||
sort: sort,
|
|
||||||
}
|
|
||||||
.render()
|
|
||||||
.unwrap();
|
|
||||||
Ok(HttpResponse::Ok().content_type("text/html").body(s))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// SERVICES
|
|
||||||
pub async fn page(web::Path(username): web::Path<String>, params: web::Query<Params>) -> Result<HttpResponse> {
|
|
||||||
match ¶ms.sort {
|
|
||||||
Some(sort) => render(username, sort.to_string()).await,
|
|
||||||
None => render(username, "hot".to_string()).await,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// USER
|
// USER
|
||||||
async fn user(name: &String) -> Result<User, &'static str> {
|
async fn user(name: &str) -> Result<User, String> {
|
||||||
// Build the Reddit JSON API url
|
// Build the Reddit JSON API path
|
||||||
let url: String = format!("https://www.reddit.com/user/{}/about.json", name);
|
let path: String = format!("/user/{}/about.json", name);
|
||||||
|
|
||||||
// Send a request to the url, receive JSON in response
|
// Send a request to the url
|
||||||
let req = request(url).await;
|
match request(path).await {
|
||||||
|
// If success, receive JSON in response
|
||||||
|
Ok(res) => {
|
||||||
|
// Grab creation date as unix timestamp
|
||||||
|
let created: i64 = res["data"]["created"].as_f64().unwrap_or(0.0).round() as i64;
|
||||||
|
|
||||||
// If the Reddit API returns an error, exit this function
|
// nested_val function used to parse JSON from Reddit APIs
|
||||||
if req.is_err() {
|
let about = |item| res["data"]["subreddit"][item].as_str().unwrap_or_default().to_string();
|
||||||
return Err(req.err().unwrap());
|
|
||||||
|
// Parse the JSON output into a User struct
|
||||||
|
Ok(User {
|
||||||
|
name: name.to_string(),
|
||||||
|
title: about("title"),
|
||||||
|
icon: format_url(about("icon_img").as_str()),
|
||||||
|
karma: res["data"]["total_karma"].as_i64().unwrap_or(0),
|
||||||
|
created: OffsetDateTime::from_unix_timestamp(created).format("%b %d '%y"),
|
||||||
|
banner: about("banner_img"),
|
||||||
|
description: about("public_description"),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
// If the Reddit API returns an error, exit this function
|
||||||
|
Err(msg) => return Err(msg),
|
||||||
}
|
}
|
||||||
|
|
||||||
// Otherwise, grab the JSON output from the request
|
|
||||||
let res = req.unwrap();
|
|
||||||
|
|
||||||
// Parse the JSON output into a User struct
|
|
||||||
Ok(User {
|
|
||||||
name: name.to_string(),
|
|
||||||
icon: nested_val(&res, "subreddit", "icon_img").await,
|
|
||||||
karma: res["data"]["total_karma"].as_i64().unwrap(),
|
|
||||||
banner: nested_val(&res, "subreddit", "banner_img").await,
|
|
||||||
description: nested_val(&res, "subreddit", "public_description").await,
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
488
src/utils.rs
488
src/utils.rs
@ -1,215 +1,489 @@
|
|||||||
//
|
//
|
||||||
// CRATES
|
// CRATES
|
||||||
//
|
//
|
||||||
use chrono::{TimeZone, Utc};
|
use actix_web::{cookie::Cookie, HttpRequest, HttpResponse, Result};
|
||||||
use serde_json::{from_str, Value};
|
use askama::Template;
|
||||||
// use surf::{client, get, middleware::Redirect};
|
|
||||||
|
|
||||||
#[cfg(feature = "proxy")]
|
|
||||||
use base64::encode;
|
use base64::encode;
|
||||||
|
use regex::Regex;
|
||||||
|
use serde_json::{from_str, Value};
|
||||||
|
use std::collections::HashMap;
|
||||||
|
use time::{Duration, OffsetDateTime};
|
||||||
|
use url::Url;
|
||||||
|
// use cached::proc_macro::cached;
|
||||||
|
|
||||||
//
|
//
|
||||||
// STRUCTS
|
// STRUCTS
|
||||||
//
|
//
|
||||||
#[allow(dead_code)]
|
|
||||||
// Post flair with text, background color and foreground color
|
|
||||||
pub struct Flair(pub String, pub String, pub String);
|
|
||||||
|
|
||||||
#[allow(dead_code)]
|
// Post flair with content, background color and foreground color
|
||||||
|
pub struct Flair {
|
||||||
|
pub flair_parts: Vec<FlairPart>,
|
||||||
|
pub background_color: String,
|
||||||
|
pub foreground_color: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Part of flair, either emoji or text
|
||||||
|
pub struct FlairPart {
|
||||||
|
pub flair_part_type: String,
|
||||||
|
pub value: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct Author {
|
||||||
|
pub name: String,
|
||||||
|
pub flair: Flair,
|
||||||
|
pub distinguished: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Post flags with nsfw and stickied
|
||||||
|
pub struct Flags {
|
||||||
|
pub nsfw: bool,
|
||||||
|
pub stickied: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct Media {
|
||||||
|
pub url: String,
|
||||||
|
pub width: i64,
|
||||||
|
pub height: i64,
|
||||||
|
}
|
||||||
|
|
||||||
// Post containing content, metadata and media
|
// Post containing content, metadata and media
|
||||||
pub struct Post {
|
pub struct Post {
|
||||||
|
pub id: String,
|
||||||
pub title: String,
|
pub title: String,
|
||||||
pub community: String,
|
pub community: String,
|
||||||
pub body: String,
|
pub body: String,
|
||||||
pub author: String,
|
pub author: Author,
|
||||||
pub url: String,
|
pub permalink: String,
|
||||||
pub score: String,
|
pub score: String,
|
||||||
|
pub upvote_ratio: i64,
|
||||||
pub post_type: String,
|
pub post_type: String,
|
||||||
pub media: String,
|
|
||||||
pub time: String,
|
|
||||||
pub flair: Flair,
|
pub flair: Flair,
|
||||||
|
pub flags: Flags,
|
||||||
|
pub thumbnail: Media,
|
||||||
|
pub media: Media,
|
||||||
|
pub domain: String,
|
||||||
|
pub rel_time: String,
|
||||||
|
pub created: String,
|
||||||
|
pub comments: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[allow(dead_code)]
|
|
||||||
// Comment with content, post, score and data/time that it was posted
|
// Comment with content, post, score and data/time that it was posted
|
||||||
pub struct Comment {
|
pub struct Comment {
|
||||||
|
pub id: String,
|
||||||
pub body: String,
|
pub body: String,
|
||||||
pub author: String,
|
pub author: Author,
|
||||||
pub score: String,
|
pub score: String,
|
||||||
pub time: String,
|
pub rel_time: String,
|
||||||
|
pub created: String,
|
||||||
pub replies: Vec<Comment>,
|
pub replies: Vec<Comment>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[allow(dead_code)]
|
#[derive(Default)]
|
||||||
// User struct containing metadata about user
|
// User struct containing metadata about user
|
||||||
pub struct User {
|
pub struct User {
|
||||||
pub name: String,
|
pub name: String,
|
||||||
|
pub title: String,
|
||||||
pub icon: String,
|
pub icon: String,
|
||||||
pub karma: i64,
|
pub karma: i64,
|
||||||
|
pub created: String,
|
||||||
pub banner: String,
|
pub banner: String,
|
||||||
pub description: String,
|
pub description: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[allow(dead_code)]
|
#[derive(Default)]
|
||||||
// Subreddit struct containing metadata about community
|
// Subreddit struct containing metadata about community
|
||||||
pub struct Subreddit {
|
pub struct Subreddit {
|
||||||
pub name: String,
|
pub name: String,
|
||||||
pub title: String,
|
pub title: String,
|
||||||
pub description: String,
|
pub description: String,
|
||||||
|
pub info: String,
|
||||||
pub icon: String,
|
pub icon: String,
|
||||||
pub members: String,
|
pub members: String,
|
||||||
pub active: String,
|
pub active: String,
|
||||||
|
pub wiki: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
// Parser for query params, used in sorting (eg. /r/rust/?sort=hot)
|
// Parser for query params, used in sorting (eg. /r/rust/?sort=hot)
|
||||||
#[derive(serde::Deserialize)]
|
#[derive(serde::Deserialize)]
|
||||||
pub struct Params {
|
pub struct Params {
|
||||||
|
pub t: Option<String>,
|
||||||
|
pub q: Option<String>,
|
||||||
pub sort: Option<String>,
|
pub sort: Option<String>,
|
||||||
pub after: Option<String>,
|
pub after: Option<String>,
|
||||||
pub before: Option<String>,
|
pub before: Option<String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
// Error template
|
// Error template
|
||||||
#[derive(askama::Template)]
|
#[derive(Template)]
|
||||||
#[template(path = "error.html", escape = "none")]
|
#[template(path = "error.html", escape = "none")]
|
||||||
pub struct ErrorTemplate {
|
pub struct ErrorTemplate {
|
||||||
pub message: String,
|
pub msg: String,
|
||||||
|
pub prefs: Preferences,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Default)]
|
||||||
|
pub struct Preferences {
|
||||||
|
pub theme: String,
|
||||||
|
pub front_page: String,
|
||||||
|
pub layout: String,
|
||||||
|
pub wide: String,
|
||||||
|
pub show_nsfw: String,
|
||||||
|
pub comment_sort: String,
|
||||||
|
pub subs: Vec<String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
// FORMATTING
|
// FORMATTING
|
||||||
//
|
//
|
||||||
|
|
||||||
pub async fn format_url(url: &str) -> String {
|
// Build preferences from cookies
|
||||||
#[cfg(feature = "proxy")]
|
pub fn prefs(req: HttpRequest) -> Preferences {
|
||||||
return "/proxy/".to_string() + encode(url).as_str();
|
Preferences {
|
||||||
|
theme: cookie(&req, "theme"),
|
||||||
#[cfg(not(feature = "proxy"))]
|
front_page: cookie(&req, "front_page"),
|
||||||
return url.to_string();
|
layout: cookie(&req, "layout"),
|
||||||
|
wide: cookie(&req, "wide"),
|
||||||
|
show_nsfw: cookie(&req, "show_nsfw"),
|
||||||
|
comment_sort: cookie(&req, "comment_sort"),
|
||||||
|
subs: cookie(&req, "subscriptions").split('+').map(String::from).filter(|s| !s.is_empty()).collect(),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Grab a query param from a url
|
||||||
|
pub fn param(path: &str, value: &str) -> String {
|
||||||
|
match Url::parse(format!("https://libredd.it/{}", path).as_str()) {
|
||||||
|
Ok(url) => url.query_pairs().into_owned().collect::<HashMap<_, _>>().get(value).unwrap_or(&String::new()).to_owned(),
|
||||||
|
_ => String::new(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parse Cookie value from request
|
||||||
|
pub fn cookie(req: &HttpRequest, name: &str) -> String {
|
||||||
|
actix_web::HttpMessage::cookie(req, name).unwrap_or_else(|| Cookie::new(name, "")).value().to_string()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Direct urls to proxy if proxy is enabled
|
||||||
|
pub fn format_url(url: &str) -> String {
|
||||||
|
if url.is_empty() || url == "self" || url == "default" || url == "nsfw" || url == "spoiler" {
|
||||||
|
String::new()
|
||||||
|
} else {
|
||||||
|
format!("/proxy/{}", encode(url).as_str())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Rewrite Reddit links to Libreddit in body of text
|
||||||
|
pub fn rewrite_url(text: &str) -> String {
|
||||||
|
let re = Regex::new(r#"href="(https://|http://|)(www.|)(reddit).(com)/"#).unwrap();
|
||||||
|
re.replace_all(text, r#"href="/"#).to_string()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Append `m` and `k` for millions and thousands respectively
|
||||||
pub fn format_num(num: i64) -> String {
|
pub fn format_num(num: i64) -> String {
|
||||||
if num > 1000000 {
|
if num >= 1_000_000 {
|
||||||
format!("{}m", num / 1000000)
|
format!("{}m", num / 1_000_000)
|
||||||
} else if num > 1000 {
|
} else if num >= 1000 {
|
||||||
format!("{}k", num / 1000)
|
format!("{}k", num / 1_000)
|
||||||
} else {
|
} else {
|
||||||
num.to_string()
|
num.to_string()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub async fn media(data: &Value) -> (String, Media) {
|
||||||
|
let post_type: &str;
|
||||||
|
// If post is a video, return the video
|
||||||
|
let url = if data["preview"]["reddit_video_preview"]["fallback_url"].is_string() {
|
||||||
|
post_type = "video";
|
||||||
|
format_url(data["preview"]["reddit_video_preview"]["fallback_url"].as_str().unwrap_or_default())
|
||||||
|
} else if data["secure_media"]["reddit_video"]["fallback_url"].is_string() {
|
||||||
|
post_type = "video";
|
||||||
|
format_url(data["secure_media"]["reddit_video"]["fallback_url"].as_str().unwrap_or_default())
|
||||||
|
// Handle images, whether GIFs or pics
|
||||||
|
} else if data["post_hint"].as_str().unwrap_or("") == "image" {
|
||||||
|
let preview = data["preview"]["images"][0].clone();
|
||||||
|
match preview["variants"]["mp4"].as_object() {
|
||||||
|
// Return the mp4 if the media is a gif
|
||||||
|
Some(gif) => {
|
||||||
|
post_type = "gif";
|
||||||
|
format_url(gif["source"]["url"].as_str().unwrap_or_default())
|
||||||
|
}
|
||||||
|
// Return the picture if the media is an image
|
||||||
|
None => {
|
||||||
|
post_type = "image";
|
||||||
|
format_url(preview["source"]["url"].as_str().unwrap_or_default())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if data["is_self"].as_bool().unwrap_or_default() {
|
||||||
|
post_type = "self";
|
||||||
|
data["permalink"].as_str().unwrap_or_default().to_string()
|
||||||
|
} else {
|
||||||
|
post_type = "link";
|
||||||
|
data["url"].as_str().unwrap_or_default().to_string()
|
||||||
|
};
|
||||||
|
|
||||||
|
(
|
||||||
|
post_type.to_string(),
|
||||||
|
Media {
|
||||||
|
url,
|
||||||
|
width: data["preview"]["images"][0]["source"]["width"].as_i64().unwrap_or_default(),
|
||||||
|
height: data["preview"]["images"][0]["source"]["height"].as_i64().unwrap_or_default(),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn parse_rich_flair(flair_type: String, rich_flair: Option<&Vec<Value>>, text_flair: Option<&str>) -> Vec<FlairPart> {
|
||||||
|
// Parse type of flair
|
||||||
|
match flair_type.as_str() {
|
||||||
|
// If flair contains emojis and text
|
||||||
|
"richtext" => match rich_flair {
|
||||||
|
Some(rich) => rich
|
||||||
|
.iter()
|
||||||
|
// For each part of the flair, extract text and emojis
|
||||||
|
.map(|part| {
|
||||||
|
let value = |name: &str| part[name].as_str().unwrap_or_default();
|
||||||
|
FlairPart {
|
||||||
|
flair_part_type: value("e").to_string(),
|
||||||
|
value: match value("e") {
|
||||||
|
"text" => value("t").to_string(),
|
||||||
|
"emoji" => format_url(value("u")),
|
||||||
|
_ => String::new(),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.collect::<Vec<FlairPart>>(),
|
||||||
|
None => Vec::new(),
|
||||||
|
},
|
||||||
|
// If flair contains only text
|
||||||
|
"text" => match text_flair {
|
||||||
|
Some(text) => vec![FlairPart {
|
||||||
|
flair_part_type: "text".to_string(),
|
||||||
|
value: text.to_string(),
|
||||||
|
}],
|
||||||
|
None => Vec::new(),
|
||||||
|
},
|
||||||
|
_ => Vec::new(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn time(created: f64) -> (String, String) {
|
||||||
|
let time = OffsetDateTime::from_unix_timestamp(created.round() as i64);
|
||||||
|
let time_delta = OffsetDateTime::now_utc() - time;
|
||||||
|
|
||||||
|
// If the time difference is more than a month, show full date
|
||||||
|
let rel_time = if time_delta > Duration::days(30) {
|
||||||
|
time.format("%b %d '%y")
|
||||||
|
// Otherwise, show relative date/time
|
||||||
|
} else if time_delta.whole_days() > 0 {
|
||||||
|
format!("{}d ago", time_delta.whole_days())
|
||||||
|
} else if time_delta.whole_hours() > 0 {
|
||||||
|
format!("{}h ago", time_delta.whole_hours())
|
||||||
|
} else {
|
||||||
|
format!("{}m ago", time_delta.whole_minutes())
|
||||||
|
};
|
||||||
|
|
||||||
|
(rel_time, time.format("%b %d %Y, %H:%M UTC"))
|
||||||
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
// JSON PARSING
|
// JSON PARSING
|
||||||
//
|
//
|
||||||
|
|
||||||
#[allow(dead_code)]
|
|
||||||
// val() function used to parse JSON from Reddit APIs
|
// val() function used to parse JSON from Reddit APIs
|
||||||
pub async fn val(j: &serde_json::Value, k: &str) -> String {
|
pub fn val(j: &Value, k: &str) -> String {
|
||||||
String::from(j["data"][k].as_str().unwrap_or(""))
|
j["data"][k].as_str().unwrap_or_default().to_string()
|
||||||
}
|
}
|
||||||
|
|
||||||
#[allow(dead_code)]
|
// Fetch posts of a user or subreddit and return a vector of posts and the "after" value
|
||||||
// nested_val() function used to parse JSON from Reddit APIs
|
pub async fn fetch_posts(path: &str, fallback_title: String) -> Result<(Vec<Post>, String), String> {
|
||||||
pub async fn nested_val(j: &serde_json::Value, n: &str, k: &str) -> String {
|
let res;
|
||||||
String::from(j["data"][n][k].as_str().unwrap())
|
let post_list;
|
||||||
}
|
|
||||||
|
|
||||||
#[allow(dead_code)]
|
// Send a request to the url
|
||||||
pub async fn fetch_posts(url: String, fallback_title: String) -> Result<(Vec<Post>, String), &'static str> {
|
match request(path.to_string()).await {
|
||||||
// Send a request to the url, receive JSON in response
|
// If success, receive JSON in response
|
||||||
let req = request(url).await;
|
Ok(response) => {
|
||||||
|
res = response;
|
||||||
// If the Reddit API returns an error, exit this function
|
}
|
||||||
if req.is_err() {
|
// If the Reddit API returns an error, exit this function
|
||||||
return Err(req.err().unwrap());
|
Err(msg) => return Err(msg),
|
||||||
}
|
}
|
||||||
|
|
||||||
// Otherwise, grab the JSON output from the request
|
|
||||||
let res = req.unwrap();
|
|
||||||
|
|
||||||
// Fetch the list of posts from the JSON response
|
// Fetch the list of posts from the JSON response
|
||||||
let post_list = res["data"]["children"].as_array().unwrap();
|
match res["data"]["children"].as_array() {
|
||||||
|
Some(list) => post_list = list,
|
||||||
|
None => return Err("No posts found".to_string()),
|
||||||
|
}
|
||||||
|
|
||||||
let mut posts: Vec<Post> = Vec::new();
|
let mut posts: Vec<Post> = Vec::new();
|
||||||
|
|
||||||
for post in post_list.iter() {
|
// For each post from posts list
|
||||||
let img = if val(post, "thumbnail").await.starts_with("https:/") {
|
for post in post_list {
|
||||||
format_url(val(post, "thumbnail").await.as_str()).await
|
let (rel_time, created) = time(post["data"]["created_utc"].as_f64().unwrap_or_default());
|
||||||
} else {
|
let score = post["data"]["score"].as_i64().unwrap_or_default();
|
||||||
String::new()
|
let ratio: f64 = post["data"]["upvote_ratio"].as_f64().unwrap_or(1.0) * 100.0;
|
||||||
};
|
let title = val(post, "title");
|
||||||
let unix_time: i64 = post["data"]["created_utc"].as_f64().unwrap().round() as i64;
|
|
||||||
let score = post["data"]["score"].as_i64().unwrap();
|
// Determine the type of media along with the media URL
|
||||||
let title = val(post, "title").await;
|
let (post_type, media) = media(&post["data"]).await;
|
||||||
|
|
||||||
posts.push(Post {
|
posts.push(Post {
|
||||||
|
id: val(post, "id"),
|
||||||
title: if title.is_empty() { fallback_title.to_owned() } else { title },
|
title: if title.is_empty() { fallback_title.to_owned() } else { title },
|
||||||
community: val(post, "subreddit").await,
|
community: val(post, "subreddit"),
|
||||||
body: val(post, "body").await,
|
body: rewrite_url(&val(post, "body_html")),
|
||||||
author: val(post, "author").await,
|
author: Author {
|
||||||
score: format_num(score),
|
name: val(post, "author"),
|
||||||
post_type: "link".to_string(),
|
flair: Flair {
|
||||||
media: img,
|
flair_parts: parse_rich_flair(
|
||||||
url: val(post, "permalink").await,
|
val(post, "author_flair_type"),
|
||||||
time: Utc.timestamp(unix_time, 0).format("%b %e '%y").to_string(),
|
post["data"]["author_flair_richtext"].as_array(),
|
||||||
flair: Flair(
|
post["data"]["author_flair_text"].as_str(),
|
||||||
val(post, "link_flair_text").await,
|
),
|
||||||
val(post, "link_flair_background_color").await,
|
background_color: val(post, "author_flair_background_color"),
|
||||||
if val(post, "link_flair_text_color").await == "dark" {
|
foreground_color: val(post, "author_flair_text_color"),
|
||||||
|
},
|
||||||
|
distinguished: val(post, "distinguished"),
|
||||||
|
},
|
||||||
|
score: if post["data"]["hide_score"].as_bool().unwrap_or_default() {
|
||||||
|
"•".to_string()
|
||||||
|
} else {
|
||||||
|
format_num(score)
|
||||||
|
},
|
||||||
|
upvote_ratio: ratio as i64,
|
||||||
|
post_type,
|
||||||
|
thumbnail: Media {
|
||||||
|
url: format_url(val(post, "thumbnail").as_str()),
|
||||||
|
width: post["data"]["thumbnail_width"].as_i64().unwrap_or_default(),
|
||||||
|
height: post["data"]["thumbnail_height"].as_i64().unwrap_or_default(),
|
||||||
|
},
|
||||||
|
media,
|
||||||
|
domain: val(post, "domain"),
|
||||||
|
flair: Flair {
|
||||||
|
flair_parts: parse_rich_flair(
|
||||||
|
val(post, "link_flair_type"),
|
||||||
|
post["data"]["link_flair_richtext"].as_array(),
|
||||||
|
post["data"]["link_flair_text"].as_str(),
|
||||||
|
),
|
||||||
|
background_color: val(post, "link_flair_background_color"),
|
||||||
|
foreground_color: if val(post, "link_flair_text_color") == "dark" {
|
||||||
"black".to_string()
|
"black".to_string()
|
||||||
} else {
|
} else {
|
||||||
"white".to_string()
|
"white".to_string()
|
||||||
},
|
},
|
||||||
),
|
},
|
||||||
|
flags: Flags {
|
||||||
|
nsfw: post["data"]["over_18"].as_bool().unwrap_or_default(),
|
||||||
|
stickied: post["data"]["stickied"].as_bool().unwrap_or_default(),
|
||||||
|
},
|
||||||
|
permalink: val(post, "permalink"),
|
||||||
|
rel_time,
|
||||||
|
created,
|
||||||
|
comments: format_num(post["data"]["num_comments"].as_i64().unwrap_or_default()),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok((posts, res["data"]["after"].as_str().unwrap_or("").to_string()))
|
Ok((posts, res["data"]["after"].as_str().unwrap_or_default().to_string()))
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
// NETWORKING
|
// NETWORKING
|
||||||
//
|
//
|
||||||
|
|
||||||
|
pub async fn error(msg: String) -> HttpResponse {
|
||||||
|
let body = ErrorTemplate {
|
||||||
|
msg,
|
||||||
|
prefs: Preferences::default(),
|
||||||
|
}
|
||||||
|
.render()
|
||||||
|
.unwrap_or_default();
|
||||||
|
HttpResponse::NotFound().content_type("text/html").body(body)
|
||||||
|
}
|
||||||
|
|
||||||
// Make a request to a Reddit API and parse the JSON response
|
// Make a request to a Reddit API and parse the JSON response
|
||||||
#[allow(dead_code)]
|
// #[cached(size=100,time=60, result = true)]
|
||||||
pub async fn request(url: String) -> Result<serde_json::Value, &'static str> {
|
pub async fn request(path: String) -> Result<Value, String> {
|
||||||
// --- actix-web::client ---
|
let url = format!("https://www.reddit.com{}", path);
|
||||||
// let client = actix_web::client::Client::default();
|
let user_agent = format!("web:libreddit:{}", env!("CARGO_PKG_VERSION"));
|
||||||
// let res = client
|
|
||||||
// .get(url)
|
|
||||||
// .send()
|
|
||||||
// .await?
|
|
||||||
// .body()
|
|
||||||
// .limit(1000000)
|
|
||||||
// .await?;
|
|
||||||
|
|
||||||
// let body = std::str::from_utf8(res.as_ref())?; // .as_ref converts Bytes to [u8]
|
// Send request using awc
|
||||||
|
// async fn send(url: &str) -> Result<String, (bool, String)> {
|
||||||
|
// let client = actix_web::client::Client::default();
|
||||||
|
// let response = client.get(url).header("User-Agent", format!("web:libreddit:{}", env!("CARGO_PKG_VERSION"))).send().await;
|
||||||
|
|
||||||
// --- surf ---
|
// match response {
|
||||||
// let req = get(&url).header("User-Agent", "libreddit");
|
// Ok(mut payload) => {
|
||||||
// let client = client().with(Redirect::new(5));
|
// // Get first number of response HTTP status code
|
||||||
// let mut res = client.send(req).await.unwrap();
|
// match payload.status().to_string().chars().next() {
|
||||||
// let success = res.status().is_success();
|
// // If success
|
||||||
// let body = res.body_string().await.unwrap();
|
// Some('2') => Ok(String::from_utf8(payload.body().limit(20_000_000).await.unwrap_or_default().to_vec()).unwrap_or_default()),
|
||||||
|
// // If redirection
|
||||||
|
// Some('3') => match payload.headers().get("location") {
|
||||||
|
// Some(location) => Err((true, location.to_str().unwrap_or_default().to_string())),
|
||||||
|
// None => Err((false, "Page not found".to_string())),
|
||||||
|
// },
|
||||||
|
// // Otherwise
|
||||||
|
// _ => Err((false, "Page not found".to_string())),
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// Err(e) => { dbg!(e); Err((false, "Couldn't send request to Reddit, this instance may be being rate-limited. Try another.".to_string())) },
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
||||||
// --- reqwest ---
|
// // Print error if debugging then return error based on error message
|
||||||
let res = reqwest::get(&url).await.unwrap();
|
// fn err(url: String, msg: String) -> Result<Value, String> {
|
||||||
// Read the status from the response
|
// // #[cfg(debug_assertions)]
|
||||||
let success = res.status().is_success();
|
// dbg!(format!("{} - {}", url, msg));
|
||||||
// Read the body of the response
|
// Err(msg)
|
||||||
let body = res.text().await.unwrap();
|
// };
|
||||||
|
|
||||||
// Parse the response from Reddit as JSON
|
// // Parse JSON from body. If parsing fails, return error
|
||||||
let json: Value = from_str(body.as_str()).unwrap_or(Value::Null);
|
// fn json(url: String, body: String) -> Result<Value, String> {
|
||||||
|
// match from_str(body.as_str()) {
|
||||||
|
// Ok(json) => Ok(json),
|
||||||
|
// Err(_) => err(url, "Failed to parse page JSON data".to_string()),
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
||||||
if !success {
|
// // Make request to Reddit using send function
|
||||||
println!("! {} - {}", url, "Page not found");
|
// match send(&url).await {
|
||||||
Err("Page not found")
|
// // If success, parse and return body
|
||||||
} else if json == Value::Null {
|
// Ok(body) => json(url, body),
|
||||||
println!("! {} - {}", url, "Failed to parse page JSON data");
|
// // Follow any redirects
|
||||||
Err("Failed to parse page JSON data")
|
// Err((true, location)) => match send(location.as_str()).await {
|
||||||
} else {
|
// // If success, parse and return body
|
||||||
Ok(json)
|
// Ok(body) => json(url, body),
|
||||||
|
// // Follow any redirects again
|
||||||
|
// Err((true, location)) => err(url, location),
|
||||||
|
// // Return errors if request fails
|
||||||
|
// Err((_, msg)) => err(url, msg),
|
||||||
|
// },
|
||||||
|
// // Return errors if request fails
|
||||||
|
// Err((_, msg)) => err(url, msg),
|
||||||
|
// }
|
||||||
|
|
||||||
|
// Send request using ureq
|
||||||
|
match ureq::get(&url).set("User-Agent", user_agent.as_str()).call() {
|
||||||
|
// If response is success
|
||||||
|
Ok(response) => {
|
||||||
|
// Parse the response from Reddit as JSON
|
||||||
|
let json_string = &response.into_string().unwrap_or_default();
|
||||||
|
match from_str(json_string) {
|
||||||
|
Ok(json) => Ok(json),
|
||||||
|
Err(e) => {
|
||||||
|
println!("{} - Failed to parse page JSON data: {} - {}", url, e, json_string);
|
||||||
|
Err("Failed to parse page JSON data".to_string())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// If response is error
|
||||||
|
Err(ureq::Error::Status(_, _)) => {
|
||||||
|
#[cfg(debug_assertions)]
|
||||||
|
dbg!(format!("{} - Page not found", url));
|
||||||
|
Err("Page not found".to_string())
|
||||||
|
}
|
||||||
|
// If failed to send request
|
||||||
|
Err(e) => {
|
||||||
|
println!("{} - Couldn't send request to Reddit: {}", url, e);
|
||||||
|
Err("Couldn't send request to Reddit, this instance may be being rate-limited. Try another.".to_string())
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
BIN
static/favicon.ico
Normal file
BIN
static/favicon.ico
Normal file
Binary file not shown.
After Width: | Height: | Size: 789 B |
@ -1,2 +0,0 @@
|
|||||||
User-Agent: *
|
|
||||||
Disallow: /
|
|
1161
static/style.css
1161
static/style.css
File diff suppressed because it is too large
Load Diff
@ -2,29 +2,41 @@
|
|||||||
<html lang="en">
|
<html lang="en">
|
||||||
<head>
|
<head>
|
||||||
{% block head %}
|
{% block head %}
|
||||||
<title>{% block title %}{% endblock %}</title>
|
<title>{% block title %}Libreddit{% endblock %}</title>
|
||||||
|
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
|
||||||
<meta name="description" content="View on Libreddit, an alternative private front-end to Reddit.">
|
<meta name="description" content="View on Libreddit, an alternative private front-end to Reddit.">
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
|
<link rel="shortcut icon" type="image/x-icon" href="/favicon.ico">
|
||||||
<link rel="stylesheet" href="/style.css">
|
<link rel="stylesheet" type="text/css" href="/style.css">
|
||||||
{% block sortstyle %}
|
|
||||||
<style>
|
|
||||||
#sort > #sort_{{ sort }} {
|
|
||||||
background: aqua;
|
|
||||||
color: black;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
{% endblock %}
|
|
||||||
{% endblock %}
|
|
||||||
</head>
|
|
||||||
<body>
|
|
||||||
{% block header %}
|
|
||||||
<header>
|
|
||||||
<a href="/"><span id="lib">lib</span>reddit. <span id="version">v{{ env!("CARGO_PKG_VERSION") }}</span></a>
|
|
||||||
<a id="github" href="https://github.com/spikecodes/libreddit">GITHUB</a>
|
|
||||||
</header>
|
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
</head>
|
||||||
|
<body class="
|
||||||
|
{% if prefs.layout != "" %}{{ prefs.layout }}{% endif %}
|
||||||
|
{% if prefs.wide == "on" %} wide{% endif %}
|
||||||
|
{% if prefs.theme != "system" %} {{ prefs.theme }}{% endif %}">
|
||||||
|
<!-- NAVIGATION BAR -->
|
||||||
|
<nav>
|
||||||
|
<div id="logo">
|
||||||
|
<a id="libreddit" href="/">
|
||||||
|
<span id="lib">lib</span><span id="reddit">reddit.</span>
|
||||||
|
</a>
|
||||||
|
<span id="version">v{{ env!("CARGO_PKG_VERSION") }}</span>
|
||||||
|
{% block subscriptions %}{% endblock %}
|
||||||
|
</div>
|
||||||
|
{% block search %}{% endblock %}
|
||||||
|
<div id="links">
|
||||||
|
<a id="settings_link" href="/settings">
|
||||||
|
<span>settings</span>
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><circle cx="12" cy="12" r="3"/><path d="M19.4 15a1.65 1.65 0 0 0 .33 1.82l.06.06a2 2 0 0 1 0 2.83 2 2 0 0 1-2.83 0l-.06-.06a1.65 1.65 0 0 0-1.82-.33 1.65 1.65 0 0 0-1 1.51V21a2 2 0 0 1-2 2 2 2 0 0 1-2-2v-.09A1.65 1.65 0 0 0 9 19.4a1.65 1.65 0 0 0-1.82.33l-.06.06a2 2 0 0 1-2.83 0 2 2 0 0 1 0-2.83l.06-.06a1.65 1.65 0 0 0 .33-1.82 1.65 1.65 0 0 0-1.51-1H3a2 2 0 0 1-2-2 2 2 0 0 1 2-2h.09A1.65 1.65 0 0 0 4.6 9a1.65 1.65 0 0 0-.33-1.82l-.06-.06a2 2 0 0 1 0-2.83 2 2 0 0 1 2.83 0l.06.06a1.65 1.65 0 0 0 1.82.33H9a1.65 1.65 0 0 0 1-1.51V3a2 2 0 0 1 2-2 2 2 0 0 1 2 2v.09a1.65 1.65 0 0 0 1 1.51 1.65 1.65 0 0 0 1.82-.33l.06-.06a2 2 0 0 1 2.83 0 2 2 0 0 1 0 2.83l-.06.06a1.65 1.65 0 0 0-.33 1.82V9a1.65 1.65 0 0 0 1.51 1H21a2 2 0 0 1 2 2 2 2 0 0 1-2 2h-.09a1.65 1.65 0 0 0-1.51 1z"/></svg>
|
||||||
|
</a>
|
||||||
|
<a id="code" href="https://github.com/spikecodes/libreddit">
|
||||||
|
<span>code</span>
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="16 18 22 12 16 6"/><polyline points="8 6 2 12 8 18"/></svg>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
</nav>
|
||||||
|
|
||||||
|
<!-- MAIN CONTENT -->
|
||||||
{% block body %}
|
{% block body %}
|
||||||
<main>
|
<main>
|
||||||
{% block content %}
|
{% block content %}
|
||||||
@ -32,4 +44,4 @@
|
|||||||
</main>
|
</main>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{% extends "base.html" %}
|
{% extends "base.html" %}
|
||||||
{% block title %}Error: {{ message }}{% endblock %}
|
{% block title %}Error: {{ msg }}{% endblock %}
|
||||||
{% block sortstyle %}{% endblock %}
|
{% block sortstyle %}{% endblock %}
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<h1 style="text-align: center; font-size: 50px;">{{ message }}</h1>
|
<h1 style="text-align: center; font-size: 50px;">{{ msg }}</h1>
|
||||||
{% endblock %}
|
{% endblock %}
|
@ -1,43 +0,0 @@
|
|||||||
{% extends "base.html" %}
|
|
||||||
{% block title %}Libreddit{% endblock %}
|
|
||||||
{% block content %}
|
|
||||||
<div id="sort">
|
|
||||||
<div id="sort_hot"><a href="?sort=hot">Hot</a></div>
|
|
||||||
<div id="sort_top"><a href="?sort=top">Top</a></div>
|
|
||||||
<div id="sort_new"><a href="?sort=new">New</a></div>
|
|
||||||
<div id="sort_rising"><a href="?sort=rising">Rising</a></div>
|
|
||||||
</div>
|
|
||||||
{% for post in posts %}
|
|
||||||
<div class="post">
|
|
||||||
<div class="post_left">
|
|
||||||
<h3 class="post_score">{{ post.score }}</h3>
|
|
||||||
</div>
|
|
||||||
<div class="post_right">
|
|
||||||
<h4>
|
|
||||||
<b><a class="post_subreddit" href="/r/{{ post.community }}">r/{{ post.community }}</a></b>
|
|
||||||
•
|
|
||||||
Posted by
|
|
||||||
<a class="post_author" href="/u/{{ post.author }}">u/{{ post.author }}</a>
|
|
||||||
<span class="datetime" style="float: right;">{{ post.time }}</span>
|
|
||||||
</h4>
|
|
||||||
<h3 class="post_title">
|
|
||||||
{% if post.flair.0 != "" %}
|
|
||||||
<small style="color:{{ post.flair.2 }}; background:{{ post.flair.1 }}">{{ post.flair.0 }}</small>
|
|
||||||
{% endif %}
|
|
||||||
<a href="{{ post.url }}">{{ post.title }}</a>
|
|
||||||
</h3>
|
|
||||||
</div>
|
|
||||||
<img class="post_thumbnail" src="{{ post.media }}">
|
|
||||||
</div><br>
|
|
||||||
{% endfor %}
|
|
||||||
|
|
||||||
<footer>
|
|
||||||
{% if ends.0 != "" %}
|
|
||||||
<a href="?before={{ ends.0 }}">PREV</a>
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
{% if ends.1 != "" %}
|
|
||||||
<a href="?after={{ ends.1 }}">NEXT</a>
|
|
||||||
{% endif %}
|
|
||||||
</footer>
|
|
||||||
{% endblock %}
|
|
@ -1,76 +1,136 @@
|
|||||||
{% extends "base.html" %}
|
{% extends "base.html" %}
|
||||||
|
{% import "utils.html" as utils %}
|
||||||
|
|
||||||
{% block title %}{{ post.title }} - r/{{ post.community }}{% endblock %}
|
{% block title %}{{ post.title }} - r/{{ post.community }}{% endblock %}
|
||||||
{% block head %}
|
|
||||||
{% call super() %}
|
{% block search %}
|
||||||
<meta name="author" content="u/{{ post.author }}">
|
{% call utils::search(["/r/", post.community.as_str()].concat(), "") %}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
{% macro comment(item) -%}
|
{% block root %}/r/{{ post.community }}{% endblock %}{% block location %}r/{{ post.community }}{% endblock %}
|
||||||
|
{% block head %}
|
||||||
|
{% call super() %}
|
||||||
|
<meta name="author" content="u/{{ post.author.name }}">
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
<div class="comment">
|
{% block subscriptions %}
|
||||||
|
{% call utils::sub_list(post.community.as_str()) %}
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
<!-- OPEN COMMENT MACRO -->
|
||||||
|
{% macro comment(item) -%}
|
||||||
|
<div id="{{ item.id }}" class="comment">
|
||||||
<div class="comment_left">
|
<div class="comment_left">
|
||||||
<h3 class="comment_score">{{ item.score }}</h3>
|
<p class="comment_score">{{ item.score }}</p>
|
||||||
<div class="line"></div>
|
<div class="line"></div>
|
||||||
</div>
|
</div>
|
||||||
<details class="comment_right" open>
|
<details class="comment_right" open>
|
||||||
<summary class="comment_data">
|
<summary class="comment_data">
|
||||||
<a class="comment_author" href="/u/{{ item.author }}">u/{{ item.author }}</a> • <span class="datetime">{{ item.time }}</span>
|
<a class="comment_author {{ item.author.distinguished }} {% if item.author.name == post.author.name %}op{% endif %}" href="/u/{{ item.author.name }}">u/{{ item.author.name }}</a>
|
||||||
|
{% if item.author.flair.flair_parts.len() > 0 %}
|
||||||
|
<small class="author_flair">{% call utils::render_flair(item.author.flair.flair_parts) %}</small>
|
||||||
|
{% endif %}
|
||||||
|
<span class="created" title="{{ post.created }}">{{ item.rel_time }}</span>
|
||||||
</summary>
|
</summary>
|
||||||
<h4 class="comment_body">{{ item.body }}</h4>
|
<div class="comment_body">{{ item.body }}</div>
|
||||||
|
|
||||||
{%- endmacro %}
|
{%- endmacro %}
|
||||||
|
|
||||||
|
<!-- CLOSE COMMENT MACRO -->
|
||||||
|
{% macro close() %}
|
||||||
|
</details></div>
|
||||||
|
{% endmacro %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<div class="post highlighted">
|
<div id="column_one">
|
||||||
<div class="post_left">
|
|
||||||
<h3 class="post_score">{{ post.score }}</h3>
|
<!-- POST CONTENT -->
|
||||||
</div>
|
<div class="post highlighted">
|
||||||
<div class="post_right">
|
<p class="post_header">
|
||||||
<h4>
|
<a class="post_subreddit" href="/r/{{ post.community }}">r/{{ post.community }}</a>
|
||||||
<b><a class="post_subreddit" href="/r/{{ post.community }}">r/{{ post.community }}</a></b>
|
<span class="dot">•</span>
|
||||||
•
|
<a class="post_author" href="/u/{{ post.author.name }}">u/{{ post.author.name }}</a>
|
||||||
Posted by
|
{% if post.author.flair.flair_parts.len() > 0 %}
|
||||||
<a class="post_author" href="/u/{{ post.author }}">u/{{ post.author }}</a>
|
<small class="author_flair">{% call utils::render_flair(post.author.flair.flair_parts) %}</small>
|
||||||
<span class="datetime">{{ post.time }}</span>
|
|
||||||
</h4>
|
|
||||||
<a href="{{ post.url }}" class="post_title">
|
|
||||||
{{ post.title }}
|
|
||||||
{% if post.flair.0 != "" %}
|
|
||||||
<small style="color:{{ post.flair.2 }}; background:{{ post.flair.1 }}">{{ post.flair.0 }}</small>
|
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</a>
|
<span class="dot">•</span>
|
||||||
|
<span class="created" title="{{ post.created }}">{{ post.rel_time }}</span>
|
||||||
|
</p>
|
||||||
|
<p class="post_title">
|
||||||
|
<a href="{{ post.permalink }}">{{ post.title }}</a>
|
||||||
|
{% if post.flair.flair_parts.len() > 0 %}
|
||||||
|
<small class="post_flair" style="color:{{ post.flair.foreground_color }}; background:{{ post.flair.background_color }};">{% call utils::render_flair(post.flair.flair_parts) %}</small>
|
||||||
|
{% endif %}
|
||||||
|
{% if post.flags.nsfw %} <small class="nsfw">NSFW</small>{% endif %}
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<!-- POST MEDIA -->
|
||||||
{% if post.post_type == "image" %}
|
{% if post.post_type == "image" %}
|
||||||
<img class="post_media" src="{{ post.media }}"/>
|
<a href="{{ post.media.url }}" style="display:contents" >
|
||||||
{% else if post.post_type == "video" %}
|
<svg class="post_media"
|
||||||
<video class="post_media" src="{{ post.media }}" controls autoplay loop>
|
width="{{ post.media.width }}px"
|
||||||
|
height="{{ post.media.height }}px"
|
||||||
|
xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<image width="100%" height="100%" href="{{ post.media.url }}"/>
|
||||||
|
<desc>
|
||||||
|
<img alt="Post image" src="{{ post.media.url }}"/>
|
||||||
|
</dev>
|
||||||
|
</svg>
|
||||||
|
</a>
|
||||||
|
{% else if post.post_type == "video" || post.post_type == "gif" %}
|
||||||
|
<video class="post_media" src="{{ post.media.url }}" controls autoplay loop></video>
|
||||||
{% else if post.post_type == "link" %}
|
{% else if post.post_type == "link" %}
|
||||||
<a id="post_url" href="{{ post.media }}">{{ post.media }}</a>
|
<a id="post_url" href="{{ post.media.url }}">{{ post.media.url }}</a>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<h4 class="post_body">{{ post.body }}</h4>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div id="sort">
|
|
||||||
<div id="sort_confidence"><a href="?sort=confidence">Best</a></div>
|
|
||||||
<div id="sort_top"><a href="?sort=top">Top</a></div>
|
|
||||||
<div id="sort_new"><a href="?sort=new">New</a></div>
|
|
||||||
<div id="sort_controversial"><a href="?sort=controversial">Controversial</a></div>
|
|
||||||
<div id="sort_old"><a href="?sort=old">Old</a></div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{% for c in comments -%}
|
<!-- POST BODY -->
|
||||||
|
<div class="post_body">{{ post.body }}</div>
|
||||||
|
<div class="post_score">{{ post.score }}<span class="label"> Upvotes</span></div>
|
||||||
|
<div class="post_footer">
|
||||||
|
<ul id="post_links">
|
||||||
|
<li><a href="/{{ post.id }}">permalink</a></li>
|
||||||
|
<li><a href="https://reddit.com/{{ post.id }}">reddit</a></li>
|
||||||
|
</ul>
|
||||||
|
<p>{{ post.upvote_ratio }}% Upvoted</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- SORT FORM -->
|
||||||
|
<form id="sort">
|
||||||
|
<select name="sort" title="Sort comments by">
|
||||||
|
{% call utils::options(sort, ["confidence", "top", "new", "controversial", "old"], "confidence") %}
|
||||||
|
</select><button id="sort_submit" class="submit">
|
||||||
|
<svg width="15" viewBox="0 0 110 100" fill="none" stroke-width="10" stroke-linecap="round">
|
||||||
|
<path d="M20 50 H100" />
|
||||||
|
<path d="M75 15 L100 50 L75 85" />
|
||||||
|
→
|
||||||
|
</svg>
|
||||||
|
</button>
|
||||||
|
</form>
|
||||||
|
|
||||||
|
<!-- COMMENTS -->
|
||||||
|
{% for c in comments -%}
|
||||||
<div class="thread">
|
<div class="thread">
|
||||||
|
<!-- EACH COMMENT -->
|
||||||
{% call comment(c) %}
|
{% call comment(c) %}
|
||||||
<div class="replies">
|
<blockquote class="replies">{% for reply1 in c.replies %}{% call comment(reply1) %}
|
||||||
{% for reply in c.replies %}
|
<!-- FIRST-LEVEL REPLIES -->
|
||||||
{% call comment(reply) %}
|
<blockquote class="replies">{% for reply2 in reply1.replies %}{% call comment(reply2) %}
|
||||||
<div class="replies">
|
<!-- SECOND-LEVEL REPLIES -->
|
||||||
{% for response in reply.replies %}
|
<blockquote class="replies">{% for reply3 in reply2.replies %}{% call comment(reply3) %}
|
||||||
{% call comment(response) %}</details></div>
|
<!-- THIRD-LEVEL REPLIES -->
|
||||||
{% endfor %}
|
{% if reply3.replies.len() > 0 %}
|
||||||
</div></details></div>
|
<!-- LINK TO CONTINUE REPLIES -->
|
||||||
|
<a class="deeper_replies" href="{{ post.permalink }}{{ reply3.id }}">→ More replies</a>
|
||||||
|
{% endif %}
|
||||||
|
{% call close() %}
|
||||||
|
{% endfor %}
|
||||||
|
</blockquote>{% call close() %}
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</div></details></div>
|
</blockquote>{% call close() %}
|
||||||
|
{% endfor %}
|
||||||
|
</blockquote>{% call close() %}
|
||||||
</div>
|
</div>
|
||||||
{%- endfor %}
|
{%- endfor %}
|
||||||
|
|
||||||
{% endblock %}
|
</div>
|
||||||
|
{% endblock %}
|
||||||
|
130
templates/search.html
Normal file
130
templates/search.html
Normal file
@ -0,0 +1,130 @@
|
|||||||
|
{% extends "base.html" %}
|
||||||
|
{% import "utils.html" as utils %}
|
||||||
|
|
||||||
|
{% block title %}Libreddit: search results - {{ params.q }}{% endblock %}
|
||||||
|
|
||||||
|
{% block subscriptions %}
|
||||||
|
{% call utils::sub_list("") %}
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<div id="column_one">
|
||||||
|
<form id="search_sort">
|
||||||
|
<input id="search" type="text" name="q" placeholder="Search" value="{{ params.q }}" title="Search libreddit">
|
||||||
|
{% if sub != "" %}
|
||||||
|
<div id="inside">
|
||||||
|
<input type="checkbox" name="restrict_sr" id="restrict_sr" {% if params.restrict_sr != "" %}checked{% endif %}>
|
||||||
|
<label for="restrict_sr" class="search_label">in r/{{ sub }}</label>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
<select id="sort_options" name="sort" title="Sort results by">
|
||||||
|
{% call utils::options(params.sort, ["relevance", "hot", "top", "new", "comments"], "") %}
|
||||||
|
</select>{% if params.sort != "new" %}<select id="timeframe" name="t" title="Timeframe">
|
||||||
|
{% call utils::options(params.t, ["hour", "day", "week", "month", "year", "all"], "all") %}
|
||||||
|
</select>{% endif %}<button id="sort_submit" class="submit">
|
||||||
|
<svg width="15" viewBox="0 0 110 100" fill="none" stroke-width="10" stroke-linecap="round">
|
||||||
|
<path d="M20 50 H100" />
|
||||||
|
<path d="M75 15 L100 50 L75 85" />
|
||||||
|
→
|
||||||
|
</svg>
|
||||||
|
</button>
|
||||||
|
</form>
|
||||||
|
|
||||||
|
{% if subreddits.len() > 0 %}
|
||||||
|
<div id="search_subreddits">
|
||||||
|
{% for subreddit in subreddits %}
|
||||||
|
<a href="{{ subreddit.url }}" class="search_subreddit">
|
||||||
|
<p class="search_subreddit_header">
|
||||||
|
<span class="search_subreddit_name">{{ subreddit.name }}</span>
|
||||||
|
<span class="dot">•</span>
|
||||||
|
<span class="search_subreddit_members">{{ subreddit.subscribers }} Members</span>
|
||||||
|
</p>
|
||||||
|
<p class="search_subreddit_description">{{ subreddit.description }}</p>
|
||||||
|
</a>
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
{% for post in posts %}
|
||||||
|
|
||||||
|
{% if post.flags.nsfw && prefs.show_nsfw != "on" %}
|
||||||
|
{% else if post.title != "Comment" %}
|
||||||
|
<div class="post {% if post.flags.stickied %}stickied{% endif %}">
|
||||||
|
<p class="post_header">
|
||||||
|
<a class="post_subreddit" href="/r/{{ post.community }}">r/{{ post.community }}</a>
|
||||||
|
<span class="dot">•</span>
|
||||||
|
<a class="post_author" href="/u/{{ post.author.name }}">u/{{ post.author.name }}</a>
|
||||||
|
<span class="dot">•</span>
|
||||||
|
<span class="created" title="{{ post.created }}">{{ post.rel_time }}</span>
|
||||||
|
</p>
|
||||||
|
<p class="post_title">
|
||||||
|
{% if post.flair.flair_parts.len() > 0 %}
|
||||||
|
<small class="post_flair" style="color:{{ post.flair.foreground_color }}; background:{{ post.flair.background_color }};">{% call utils::render_flair(post.flair.flair_parts) %}</small>
|
||||||
|
{% endif %}
|
||||||
|
<a href="{{ post.permalink }}">{{ post.title }}</a>{% if post.flags.nsfw %} <small class="nsfw">NSFW</small>{% endif %}
|
||||||
|
</p>
|
||||||
|
<!-- POST MEDIA/THUMBNAIL -->
|
||||||
|
{% if (prefs.layout.is_empty() || prefs.layout == "card") && post.post_type == "image" %}
|
||||||
|
<a href="{{ post.media.url }}" style="display:contents" >
|
||||||
|
<svg class="post_media {% if post.media.height / post.media.width < 2 %}short{% endif %}"
|
||||||
|
width="{{ post.media.width }}px"
|
||||||
|
height="{{ post.media.height }}px"
|
||||||
|
xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<image width="100%" height="100%" href="{{ post.media.url }}"/>
|
||||||
|
<desc>
|
||||||
|
<img alt="Post image" src="{{ post.media.url }}"/>
|
||||||
|
</dev>
|
||||||
|
</svg>
|
||||||
|
</a>
|
||||||
|
{% else if post.post_type != "self" %}
|
||||||
|
<a class="post_thumbnail {% if post.thumbnail.url.is_empty() %}no_thumbnail{% endif %}" href="{% if post.post_type == "link" %}{{ post.media.url }}{% else %}{{ post.permalink }}{% endif %}">
|
||||||
|
{% if post.thumbnail.url.is_empty() %}
|
||||||
|
<svg viewBox="0 0 100 106" width="140" height="53" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<title>Thumbnail</title>
|
||||||
|
<path d="M35,15h-15a10,10 0,0,0 0,20h25a10,10 0,0,0 10,-10m-12.5,0a10, 10 0,0,1 10, -10h25a10,10 0,0,1 0,20h-15" fill="none" stroke-width="5" stroke-linecap="round"/>
|
||||||
|
</svg>
|
||||||
|
{% else %}
|
||||||
|
<svg width="{{ post.thumbnail.width }}px" height="{{ post.thumbnail.height }}px" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<image alt="Thumbnail" width="100%" height="100%" href="{{ post.thumbnail.url }}"/>
|
||||||
|
</svg>
|
||||||
|
{% endif %}
|
||||||
|
<span>{% if post.post_type == "link" %}{{ post.domain }}{% else %}{{ post.post_type }}{% endif %}</span>
|
||||||
|
</a>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
<div class="post_score">{{ post.score }}<span class="label"> Upvotes</span></div>
|
||||||
|
<div class="post_footer">
|
||||||
|
<a href="{{ post.permalink }}" class="post_comments">{{ post.comments }} comments</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div class="comment">
|
||||||
|
<div class="comment_left">
|
||||||
|
<p class="comment_score">{{ post.score }}</p>
|
||||||
|
<div class="line"></div>
|
||||||
|
</div>
|
||||||
|
<details class="comment_right" open>
|
||||||
|
<summary class="comment_data">
|
||||||
|
<a class="comment_link" href="{{ post.permalink }}">COMMENT</a>
|
||||||
|
<span class="created" title="{{ post.created }}">{{ post.rel_time }}</span>
|
||||||
|
</summary>
|
||||||
|
<p class="comment_body">{{ post.body }}</p>
|
||||||
|
</details>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
{% endfor %}
|
||||||
|
|
||||||
|
<footer>
|
||||||
|
{% if params.before != "" %}
|
||||||
|
<a href="?q={{ params.q }}&restrict_sr={{ params.restrict_sr }}
|
||||||
|
&sort={{ params.sort }}&t={{ params.t }}
|
||||||
|
&before={{ params.before }}">PREV</a>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if params.after != "" %}
|
||||||
|
<a href="?q={{ params.q }}&restrict_sr={{ params.restrict_sr }}
|
||||||
|
&sort={{ params.sort }}&t={{ params.t }}
|
||||||
|
&after={{ params.after }}">NEXT</a>
|
||||||
|
{% endif %}
|
||||||
|
</footer>
|
||||||
|
</div>
|
||||||
|
{% endblock %}
|
70
templates/settings.html
Normal file
70
templates/settings.html
Normal file
@ -0,0 +1,70 @@
|
|||||||
|
{% extends "base.html" %}
|
||||||
|
{% import "utils.html" as utils %}
|
||||||
|
|
||||||
|
{% block title %}Libreddit Settings{% endblock %}
|
||||||
|
|
||||||
|
{% block search %}
|
||||||
|
{% call utils::search("".to_owned(), "", "") %}
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<div id="settings">
|
||||||
|
<form action="/settings" method="POST">
|
||||||
|
<div class="prefs">
|
||||||
|
<p>Appearance</p>
|
||||||
|
<div id="theme">
|
||||||
|
<label for="theme">Theme:</label>
|
||||||
|
<select name="theme">
|
||||||
|
{% call utils::options(prefs.theme, ["system", "light", "dark"], "system") %}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
<p>Interface</p>
|
||||||
|
<div id="front_page">
|
||||||
|
<label for="front_page">Front page:</label>
|
||||||
|
<select name="front_page">
|
||||||
|
{% call utils::options(prefs.front_page, ["default", "popular", "all"], "default") %}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
<div id="layout">
|
||||||
|
<label for="layout">Layout:</label>
|
||||||
|
<select name="layout">
|
||||||
|
{% call utils::options(prefs.layout, ["card", "clean", "compact"], "card") %}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
<div id="wide">
|
||||||
|
<label for="wide">Wide UI:</label>
|
||||||
|
<input type="checkbox" name="wide" {% if prefs.wide == "on" %}checked{% endif %}>
|
||||||
|
</div>
|
||||||
|
<p>Content</p>
|
||||||
|
<div id="comment_sort">
|
||||||
|
<label for="comment_sort">Default comment sort:</label>
|
||||||
|
<select name="comment_sort">
|
||||||
|
{% call utils::options(prefs.comment_sort, ["confidence", "top", "new", "controversial", "old"], "confidence") %}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
<div id="show_nsfw">
|
||||||
|
<label for="show_nsfw">Show NSFW posts:</label>
|
||||||
|
<input type="checkbox" name="show_nsfw" {% if prefs.show_nsfw == "on" %}checked{% endif %}>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<p id="settings_note"><b>Note:</b> settings are saved in browser cookies. Clearing your cookie data will reset them.</p>
|
||||||
|
<input id="save" type="submit" value="Save">
|
||||||
|
</form>
|
||||||
|
{% if prefs.subs.len() > 0 %}
|
||||||
|
<aside class="prefs">
|
||||||
|
<p>Subscribed Subreddits</p>
|
||||||
|
<ul id="settings_subs">
|
||||||
|
{% for sub in prefs.subs %}
|
||||||
|
<li>
|
||||||
|
<span>{{ sub }}</span>
|
||||||
|
<form action="/r/{{ sub }}/unsubscribe/?redirect=/settings" method="POST">
|
||||||
|
<button class="unsubscribe">Unsubscribe</button>
|
||||||
|
</form>
|
||||||
|
</li>
|
||||||
|
{% endfor %}
|
||||||
|
</ul>
|
||||||
|
</aside>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{% endblock %}
|
@ -1,56 +1,148 @@
|
|||||||
{% extends "base.html" %}
|
{% extends "base.html" %}
|
||||||
{% block title %}r/{{ sub.name }}: {{ sub.description }}{% endblock %}
|
{% import "utils.html" as utils %}
|
||||||
|
|
||||||
|
{% block title %}
|
||||||
|
{% if sub.title != "" %}{{ sub.title }}
|
||||||
|
{% else if sub.name != "" %}{{ sub.name }}
|
||||||
|
{% else %}Libreddit{% endif %}
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block search %}
|
||||||
|
{% call utils::search(["/r/", sub.name.as_str()].concat(), "") %}
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block subscriptions %}
|
||||||
|
{% call utils::sub_list(sub.name.as_str(), "wide") %}
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
{% block body %}
|
{% block body %}
|
||||||
<div id="about">
|
|
||||||
<div class="subreddit">
|
|
||||||
<div class="subreddit_left">
|
|
||||||
{{ sub.icon }}
|
|
||||||
</div>
|
|
||||||
<div class="subreddit_right">
|
|
||||||
<h2 class="subreddit_name">r/{{ sub.name }}</h2>
|
|
||||||
<p class="subreddit_description">{{ sub.description }}</p>
|
|
||||||
<div id="stats">👤 {{ sub.members }} 🟢 {{ sub.active }}</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<main>
|
<main>
|
||||||
<div id="sort">
|
<div id="column_one">
|
||||||
<div id="sort_hot"><a href="?sort=hot">Hot</a></div>
|
<form id="sort">
|
||||||
<div id="sort_top"><a href="?sort=top">Top</a></div>
|
<div id="sort_options">
|
||||||
<div id="sort_new"><a href="?sort=new">New</a></div>
|
{% if sub.name.is_empty() %}
|
||||||
</div>
|
{% call utils::sort("", ["hot", "new", "top", "rising", "controversial"], sort.0) %}
|
||||||
{% for post in posts %}
|
{% else %}
|
||||||
<div class="post">
|
{% call utils::sort(["/r/", sub.name.as_str()].concat(), ["hot", "new", "top", "rising", "controversial"], sort.0) %}
|
||||||
<div class="post_left">
|
|
||||||
<h3 class="post_score">{{ post.score }}</h3>
|
|
||||||
</div>
|
|
||||||
<div class="post_right">
|
|
||||||
<h4>
|
|
||||||
<b><a class="post_subreddit" href="/r/{{ post.community }}">r/{{ sub.name }}</a></b>
|
|
||||||
•
|
|
||||||
Posted by
|
|
||||||
<a class="post_author" href="/u/{{ post.author }}">u/{{ post.author }}</a>
|
|
||||||
<span class="datetime">{{ post.time }}</span>
|
|
||||||
</h4>
|
|
||||||
<h3 class="post_title">
|
|
||||||
{% if post.flair.0 != "" %}
|
|
||||||
<small style="color:{{ post.flair.2 }}; background:{{ post.flair.1 }}">{{ post.flair.0 }}</small>
|
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<a href="{{ post.url }}">{{ post.title }}</a>
|
</div>
|
||||||
</h3>
|
{% if sort.0 == "top" || sort.0 == "controversial" %}<select id="timeframe" name="t" title="Timeframe">
|
||||||
|
{% call utils::options(sort.1, ["hour", "day", "week", "month", "year", "all"], "day") %}
|
||||||
|
</select>
|
||||||
|
<button id="sort_submit" class="submit">
|
||||||
|
<svg width="15" viewBox="0 0 110 100" fill="none" stroke-width="10" stroke-linecap="round">
|
||||||
|
<path d="M20 50 H100" />
|
||||||
|
<path d="M75 15 L100 50 L75 85" />
|
||||||
|
→
|
||||||
|
</svg>
|
||||||
|
</button>
|
||||||
|
{% endif %}
|
||||||
|
</form>
|
||||||
|
|
||||||
|
<div id="posts">
|
||||||
|
{% for post in posts %}
|
||||||
|
{% if !(post.flags.nsfw && prefs.show_nsfw != "on") %}
|
||||||
|
<hr class="sep" />
|
||||||
|
<div class="post {% if post.flags.stickied %}stickied{% endif %}">
|
||||||
|
<p class="post_header">
|
||||||
|
<a class="post_subreddit" href="/r/{{ post.community }}">r/{{ post.community }}</a>
|
||||||
|
<span class="dot">•</span>
|
||||||
|
<a class="post_author" href="/u/{{ post.author.name }}">u/{{ post.author.name }}</a>
|
||||||
|
<span class="dot">•</span>
|
||||||
|
<span class="created" title="{{ post.created }}">{{ post.rel_time }}</span>
|
||||||
|
</p>
|
||||||
|
<p class="post_title">
|
||||||
|
{% if post.flair.flair_parts.len() > 0 %}
|
||||||
|
<small class="post_flair" style="color:{{ post.flair.foreground_color }}; background:{{ post.flair.background_color }};">{% call utils::render_flair(post.flair.flair_parts) %}</small>
|
||||||
|
{% endif %}
|
||||||
|
<a href="{{ post.permalink }}">{{ post.title }}</a>{% if post.flags.nsfw %} <small class="nsfw">NSFW</small>{% endif %}
|
||||||
|
</p>
|
||||||
|
<!-- POST MEDIA/THUMBNAIL -->
|
||||||
|
{% if (prefs.layout.is_empty() || prefs.layout == "card") && post.post_type == "image" %}
|
||||||
|
<a href="{{ post.media.url }}" style="display:contents" >
|
||||||
|
<svg class="post_media {% if post.media.height / post.media.width < 2 %}short{% endif %}"
|
||||||
|
width="{{ post.media.width }}px"
|
||||||
|
height="{{ post.media.height }}px"
|
||||||
|
xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<image width="100%" height="100%" href="{{ post.media.url }}"/>
|
||||||
|
<desc>
|
||||||
|
<img alt="Post image" src="{{ post.media.url }}"/>
|
||||||
|
</dev>
|
||||||
|
</svg>
|
||||||
|
</a>
|
||||||
|
{% else if post.post_type != "self" %}
|
||||||
|
<a class="post_thumbnail {% if post.thumbnail.url.is_empty() %}no_thumbnail{% endif %}" href="{% if post.post_type == "link" %}{{ post.media.url }}{% else %}{{ post.permalink }}{% endif %}">
|
||||||
|
{% if post.thumbnail.url.is_empty() %}
|
||||||
|
<svg viewBox="0 0 100 106" width="140" height="53" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<title>Thumbnail</title>
|
||||||
|
<path d="M35,15h-15a10,10 0,0,0 0,20h25a10,10 0,0,0 10,-10m-12.5,0a10, 10 0,0,1 10, -10h25a10,10 0,0,1 0,20h-15" fill="none" stroke-width="5" stroke-linecap="round"/>
|
||||||
|
</svg>
|
||||||
|
{% else %}
|
||||||
|
<svg width="{{ post.thumbnail.width }}px" height="{{ post.thumbnail.height }}px" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<image alt="Thumbnail" width="100%" height="100%" href="{{ post.thumbnail.url }}"/>
|
||||||
|
</svg>
|
||||||
|
{% endif %}
|
||||||
|
<span>{% if post.post_type == "link" %}{{ post.domain }}{% else %}{{ post.post_type }}{% endif %}</span>
|
||||||
|
</a>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
<div class="post_score">{{ post.score }}<span class="label"> Upvotes</span></div>
|
||||||
|
<div class="post_footer">
|
||||||
|
<a href="{{ post.permalink }}" class="post_comments">{{ post.comments }} comments</a>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<img class="post_thumbnail" src="{{ post.media }}">
|
|
||||||
</div><br>
|
|
||||||
{% endfor %}
|
|
||||||
|
|
||||||
<footer>
|
|
||||||
{% if ends.0 != "" %}
|
|
||||||
<a href="?before={{ ends.0 }}">PREV</a>
|
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
|
||||||
{% if ends.1 != "" %}
|
<footer>
|
||||||
<a href="?after={{ ends.1 }}">NEXT</a>
|
{% if ends.0 != "" %}
|
||||||
{% endif %}
|
<a href="?sort={{ sort.0 }}&t={{ sort.1 }}&before={{ ends.0 }}">PREV</a>
|
||||||
</footer>
|
{% endif %}
|
||||||
|
|
||||||
|
{% if ends.1 != "" %}
|
||||||
|
<a href="?sort={{ sort.0 }}&t={{ sort.1 }}&after={{ ends.1 }}">NEXT</a>
|
||||||
|
{% endif %}
|
||||||
|
</footer>
|
||||||
|
</div>
|
||||||
|
{% if sub.name != "" && !sub.name.contains("+") %}
|
||||||
|
<aside>
|
||||||
|
<div class="panel" id="subreddit">
|
||||||
|
{% if sub.wiki %}
|
||||||
|
<div id="top">
|
||||||
|
<div>Posts</div>
|
||||||
|
<a href="/r/{{ sub.name }}/wiki/index">Wiki</a>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
<div id="sub_meta">
|
||||||
|
<img id="sub_icon" src="{{ sub.icon }}" alt="Icon for r/{{ sub.name }}">
|
||||||
|
<p id="sub_title">{{ sub.title }}</p>
|
||||||
|
<p id="sub_name">r/{{ sub.name }}</p>
|
||||||
|
<p id="sub_description">{{ sub.description }}</p>
|
||||||
|
<div id="sub_details">
|
||||||
|
<label>Members</label>
|
||||||
|
<label>Active</label>
|
||||||
|
<div>{{ sub.members }}</div>
|
||||||
|
<div>{{ sub.active }}</div>
|
||||||
|
</div>
|
||||||
|
<div id="sub_subscription">
|
||||||
|
{% if prefs.subs.contains(sub.name) %}
|
||||||
|
<form action="/r/{{ sub.name }}/unsubscribe" method="POST">
|
||||||
|
<button class="unsubscribe">Unsubscribe</button>
|
||||||
|
</form>
|
||||||
|
{% else %}
|
||||||
|
<form action="/r/{{ sub.name }}/subscribe" method="POST">
|
||||||
|
<button class="subscribe">Subscribe</button>
|
||||||
|
</form>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<details class="panel" id="sidebar">
|
||||||
|
<summary id="sidebar_label">Sidebar</summary>
|
||||||
|
<div id="sidebar_contents">{{ sub.info }}</div>
|
||||||
|
</details>
|
||||||
|
</aside>
|
||||||
|
{% endif %}
|
||||||
</main>
|
</main>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
@ -1,62 +1,128 @@
|
|||||||
{% extends "base.html" %}
|
{% extends "base.html" %}
|
||||||
{% block title %}Libreddit: u/{{ user.name }}{% endblock %}
|
{% import "utils.html" as utils %}
|
||||||
|
|
||||||
|
{% block search %}
|
||||||
|
{% call utils::search("".to_owned(), "", "") %}
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block title %}{{ user.name.replace("u/", "") }} (u/{{ user.name }}) - Libreddit{% endblock %}
|
||||||
|
|
||||||
|
{% block subscriptions %}
|
||||||
|
{% call utils::sub_list("") %}
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
{% block body %}
|
{% block body %}
|
||||||
<div id="about">
|
|
||||||
<div class="user">
|
|
||||||
<div class="user_left">
|
|
||||||
<img class="user_icon" src="{{ user.icon }}">
|
|
||||||
</div>
|
|
||||||
<div class="user_right">
|
|
||||||
<h2 class="user_name">u/{{ user.name }}</h2>
|
|
||||||
<p class="user_description"><span>Karma:</span> {{ user.karma }} | <span>Description:</span> "{{ user.description }}"</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<main>
|
<main>
|
||||||
<div id="sort">
|
<div id="column_one">
|
||||||
<div id="sort_hot"><a href="?sort=hot">Hot</a></div>
|
<form id="sort">
|
||||||
<div id="sort_top"><a href="?sort=top">Top</a></div>
|
<select name="sort">
|
||||||
<div id="sort_new"><a href="?sort=new">New</a></div>
|
{% call utils::options(sort.0, ["hot", "new", "top"], "") %}
|
||||||
</div>
|
</select>{% if sort.0 == "top" %}<select id="timeframe" name="t">
|
||||||
{% for post in posts %}
|
{% call utils::options(sort.1, ["hour", "day", "week", "month", "year", "all"], "all") %}
|
||||||
{% if post.title != "Comment" %}
|
</select>{% endif %}<button id="sort_submit" class="submit">
|
||||||
<div class='post'>
|
<svg width="15" viewBox="0 0 110 100" fill="none" stroke-width="10" stroke-linecap="round">
|
||||||
<div class="post_left">
|
<path d="M20 50 H100" />
|
||||||
<h3 class="post_score">{{ post.score }}</h3>
|
<path d="M75 15 L100 50 L75 85" />
|
||||||
</div>
|
→
|
||||||
<div class="post_right">
|
</svg>
|
||||||
<h4>
|
</button>
|
||||||
<b><a class="post_subreddit" href="/r/{{ post.community }}">r/{{ post.community }}</a></b>
|
</form>
|
||||||
•
|
|
||||||
Posted by
|
<div id="posts">
|
||||||
<a class="post_author" href="/u/{{ post.author }}">u/{{ post.author }}</a>
|
{% for post in posts %}
|
||||||
<span class="datetime" style="float: right;">{{ post.time }}</span>
|
|
||||||
</h4>
|
{% if post.flags.nsfw && prefs.show_nsfw != "on" %}
|
||||||
<h3 class="post_title">
|
{% else if post.title != "Comment" %}
|
||||||
{% if post.flair.0 == "Comment" %}
|
<div class="post {% if post.flags.stickied %}stickied{% endif %}">
|
||||||
{% else if post.flair.0 == "" %}
|
<p class="post_header">
|
||||||
{% else %}
|
<a class="post_subreddit" href="/r/{{ post.community }}">r/{{ post.community }}</a>
|
||||||
<small style="color:{{ post.flair.2 }}; background:{{ post.flair.1 }}">{{ post.flair.0 }}</small>
|
<span class="dot">•</span>
|
||||||
|
<a class="post_author" href="/u/{{ post.author.name }}">u/{{ post.author.name }}</a>
|
||||||
|
<span class="dot">•</span>
|
||||||
|
<span class="created" title="{{ post.created }}">{{ post.rel_time }}</span>
|
||||||
|
</p>
|
||||||
|
<p class="post_title">
|
||||||
|
{% if post.flair.flair_parts.len() > 0 %}
|
||||||
|
<small class="post_flair" style="color:{{ post.flair.foreground_color }}; background:{{ post.flair.background_color }};">{% call utils::render_flair(post.flair.flair_parts) %}</small>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<a href="{{ post.url }}">{{ post.title }}</a>
|
<a href="{{ post.permalink }}">{{ post.title }}</a>{% if post.flags.nsfw %} <small class="nsfw">NSFW</small>{% endif %}
|
||||||
</h3>
|
</p>
|
||||||
|
<!-- POST MEDIA/THUMBNAIL -->
|
||||||
|
{% if (prefs.layout.is_empty() || prefs.layout == "card") && post.post_type == "image" %}
|
||||||
|
<a href="{{ post.media.url }}" style="display:contents" >
|
||||||
|
<svg class="post_media {% if post.media.height / post.media.width < 2 %}short{% endif %}"
|
||||||
|
width="{{ post.media.width }}px"
|
||||||
|
height="{{ post.media.height }}px"
|
||||||
|
xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<image width="100%" height="100%" href="{{ post.media.url }}"/>
|
||||||
|
<desc>
|
||||||
|
<img alt="Post image" src="{{ post.media.url }}"/>
|
||||||
|
</dev>
|
||||||
|
</svg>
|
||||||
|
</a>
|
||||||
|
{% else if post.post_type != "self" %}
|
||||||
|
<a class="post_thumbnail {% if post.thumbnail.url.is_empty() %}no_thumbnail{% endif %}" href="{% if post.post_type == "link" %}{{ post.media.url }}{% else %}{{ post.permalink }}{% endif %}">
|
||||||
|
{% if post.thumbnail.url.is_empty() %}
|
||||||
|
<svg viewBox="0 0 100 106" width="140" height="53" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<title>Thumbnail</title>
|
||||||
|
<path d="M35,15h-15a10,10 0,0,0 0,20h25a10,10 0,0,0 10,-10m-12.5,0a10, 10 0,0,1 10, -10h25a10,10 0,0,1 0,20h-15" fill="none" stroke-width="5" stroke-linecap="round"/>
|
||||||
|
</svg>
|
||||||
|
{% else %}
|
||||||
|
<svg width="{{ post.thumbnail.width }}px" height="{{ post.thumbnail.height }}px" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<image alt="Thumbnail" width="100%" height="100%" href="{{ post.thumbnail.url }}"/>
|
||||||
|
</svg>
|
||||||
|
{% endif %}
|
||||||
|
<span>{% if post.post_type == "link" %}{{ post.domain }}{% else %}{{ post.post_type }}{% endif %}</span>
|
||||||
|
</a>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
<div class="post_score">{{ post.score }}<span class="label"> Upvotes</span></div>
|
||||||
|
<div class="post_footer">
|
||||||
|
<a href="{{ post.permalink }}" class="post_comments">{{ post.comments }} comments</a>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<img class="post_thumbnail" src="{{ post.media }}">
|
{% else %}
|
||||||
</div><br>
|
<div class="comment">
|
||||||
{% else %}
|
<div class="comment_left">
|
||||||
<div class="comment">
|
<p class="comment_score">{{ post.score }}</p>
|
||||||
<div class="comment_left">
|
<div class="line"></div>
|
||||||
<h3 class="comment_score">{{ post.score }}</h3>
|
</div>
|
||||||
|
<details class="comment_right" open>
|
||||||
|
<summary class="comment_data">
|
||||||
|
<a class="comment_link" href="{{ post.permalink }}">COMMENT</a>
|
||||||
|
<span class="created" title="{{ post.created }}">{{ post.rel_time }}</span>
|
||||||
|
</summary>
|
||||||
|
<p class="comment_body">{{ post.body }}</p>
|
||||||
|
</details>
|
||||||
</div>
|
</div>
|
||||||
<div class="comment_right">
|
{% endif %}
|
||||||
<h4>
|
|
||||||
COMMENT
|
{% endfor %}
|
||||||
<span class="datetime">{{ post.time }}</span>
|
|
||||||
</h4>
|
|
||||||
<h4 class="comment_body">{{ post.body }}</h4>
|
|
||||||
</div>
|
</div>
|
||||||
</div><br>
|
|
||||||
{% endif %}
|
<footer>
|
||||||
{% endfor %}
|
{% if ends.0 != "" %}
|
||||||
|
<a href="?sort={{ sort.0 }}&t={{ sort.1 }}&before={{ ends.0 }}">PREV</a>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if ends.1 != "" %}
|
||||||
|
<a href="?sort={{ sort.0 }}&t={{ sort.1 }}&after={{ ends.1 }}">NEXT</a>
|
||||||
|
{% endif %}
|
||||||
|
</footer>
|
||||||
|
</div>
|
||||||
|
<aside>
|
||||||
|
<div class="panel" id="user">
|
||||||
|
<img id="user_icon" src="{{ user.icon }}">
|
||||||
|
<p id="user_title">{{ user.title }}</p>
|
||||||
|
<p id="user_name">u/{{ user.name }}</p>
|
||||||
|
<div id="user_description">{{ user.description }}</div>
|
||||||
|
<div id="user_details">
|
||||||
|
<label>Karma</label>
|
||||||
|
<label>Created</label>
|
||||||
|
<div>{{ user.karma }}</div>
|
||||||
|
<div>{{ user.created }}</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</aside>
|
||||||
</main>
|
</main>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
54
templates/utils.html
Normal file
54
templates/utils.html
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
{% macro options(current, values, default) -%}
|
||||||
|
{% for value in values %}
|
||||||
|
<option value="{{ value }}" {% if current == value || (current == "" && value == default) %}selected{% endif %}>
|
||||||
|
{{ format!("{}{}", value.get(0..1).unwrap().to_uppercase(), value.get(1..).unwrap()) }}
|
||||||
|
</option>
|
||||||
|
{% endfor %}
|
||||||
|
{%- endmacro %}
|
||||||
|
|
||||||
|
{% macro sort(root, methods, selected) -%}
|
||||||
|
{% for method in methods %}
|
||||||
|
<a {% if method == selected %}class="selected"{% endif %} href="{{ root }}/{{ method }}">
|
||||||
|
{{ format!("{}{}", method.get(0..1).unwrap().to_uppercase(), method.get(1..).unwrap()) }}
|
||||||
|
</a>
|
||||||
|
{% endfor %}
|
||||||
|
{%- endmacro %}
|
||||||
|
|
||||||
|
{% macro search(root, search) -%}
|
||||||
|
<form action="{% if root != "/r/" && !root.is_empty() %}{{ root }}{% endif %}/search/" id="searchbox">
|
||||||
|
<input id="search" type="text" name="q" placeholder="Search" title="Search libreddit" value="{{ search }}">
|
||||||
|
{% if root != "/r/" && !root.is_empty() %}
|
||||||
|
<div id="inside">
|
||||||
|
<input type="checkbox" name="restrict_sr" id="restrict_sr">
|
||||||
|
<label for="restrict_sr" class="search_label">in {{ root }}</label>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
<button class="submit">
|
||||||
|
<svg width="15" viewBox="0 0 110 100" fill="none" stroke-width="10" stroke-linecap="round">
|
||||||
|
<path d="M20 50 H100" />
|
||||||
|
<path d="M75 15 L100 50 L75 85" />
|
||||||
|
→
|
||||||
|
</svg>
|
||||||
|
</button>
|
||||||
|
</form>
|
||||||
|
{%- endmacro %}
|
||||||
|
|
||||||
|
{% macro render_flair(flair) -%}
|
||||||
|
{% for flair_part in flair %}
|
||||||
|
{% if flair_part.flair_part_type == "emoji" %}<span class="emoji" style="background-image:url('{{ flair_part.value }}');"></span>
|
||||||
|
{% else if flair_part.flair_part_type == "text" %}<span>{{ flair_part.value }}</span>{% endif %}
|
||||||
|
{% endfor %}
|
||||||
|
{%- endmacro %}
|
||||||
|
|
||||||
|
{% macro sub_list(current) -%}
|
||||||
|
{% if prefs.subs.len() > 0 %}
|
||||||
|
<details id="subscriptions">
|
||||||
|
<summary>Subscriptions</summary>
|
||||||
|
<div id="sub_list">
|
||||||
|
{% for sub in prefs.subs %}
|
||||||
|
<a href="/r/{{ sub }}" {% if sub == current %}class="selected"{% endif %}>{{ sub }}</a>
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
</details>
|
||||||
|
{% endif %}
|
||||||
|
{%- endmacro %}
|
29
templates/wiki.html
Normal file
29
templates/wiki.html
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
{% extends "base.html" %}
|
||||||
|
{% import "utils.html" as utils %}
|
||||||
|
|
||||||
|
{% block title %}
|
||||||
|
{% if sub != "" %}{{ page }} - {{ sub }}
|
||||||
|
{% else %}Libreddit{% endif %}
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block search %}
|
||||||
|
{% call utils::search(["/r/", sub.as_str()].concat(), "") %}
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block subscriptions %}
|
||||||
|
{% call utils::sub_list(sub.as_str()) %}
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block body %}
|
||||||
|
<main>
|
||||||
|
<div class="panel" id="column_one">
|
||||||
|
<div id="top">
|
||||||
|
<a href="/r/{{ sub }}">Posts</a>
|
||||||
|
<div>Wiki</div>
|
||||||
|
</div>
|
||||||
|
<div id="wiki">
|
||||||
|
{{ wiki }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</main>
|
||||||
|
{% endblock %}
|
Reference in New Issue
Block a user