Initial commit
This commit is contained in:
commit
a61f3468ee
|
@ -0,0 +1 @@
|
||||||
|
omordl
|
|
@ -0,0 +1,10 @@
|
||||||
|
package main
|
||||||
|
|
||||||
|
const (
|
||||||
|
CONFIG_PLACEHOLDER string = "https://old.reddit.com/prefs/apps"
|
||||||
|
USER_AGENT string = "linux:omordl:v0.1.0 (by u/the_blank_x)"
|
||||||
|
EXAMPLE_CONFIG string = `{
|
||||||
|
"client_id": "https://old.reddit.com/prefs/apps",
|
||||||
|
"client_secret": "https://old.reddit.com/prefs/apps"
|
||||||
|
}`
|
||||||
|
)
|
|
@ -0,0 +1,5 @@
|
||||||
|
module omordl
|
||||||
|
|
||||||
|
go 1.17
|
||||||
|
|
||||||
|
require golang.org/x/sys v0.0.0-20211116061358-0a5406a5449c // indirect
|
|
@ -0,0 +1,2 @@
|
||||||
|
golang.org/x/sys v0.0.0-20211116061358-0a5406a5449c h1:DHcbWVXeY+0Y8HHKR+rbLwnoh2F4tNCY7rTiHJ30RmA=
|
||||||
|
golang.org/x/sys v0.0.0-20211116061358-0a5406a5449c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
|
@ -0,0 +1,153 @@
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bufio"
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"net/url"
|
||||||
|
"os"
|
||||||
|
"os/exec"
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"golang.org/x/sys/unix"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
if len(os.Args) != 2 {
|
||||||
|
fmt.Fprintf(os.Stderr, "Usage: %s <submission id/url>\n", os.Args[0])
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
submissionUrl, err := url.Parse(os.Args[1])
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "Error when parsing submission url: %s\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
submissionId := os.Args[1]
|
||||||
|
if submissionUrl.Hostname() == "redd.it" {
|
||||||
|
submissionId = submissionUrl.EscapedPath()[1:]
|
||||||
|
} else if submissionUrl.Hostname() != "" {
|
||||||
|
split := strings.SplitN(submissionUrl.EscapedPath(), "/", 6)
|
||||||
|
if len(split) < 5 {
|
||||||
|
fmt.Fprintln(os.Stderr, "URL passed does not have enough path seperators")
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
submissionId = split[4]
|
||||||
|
}
|
||||||
|
if submissionId == "" {
|
||||||
|
fmt.Fprintln(os.Stderr, "Submission ID is empty")
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
client := &http.Client{
|
||||||
|
Timeout: time.Duration(30 * time.Second),
|
||||||
|
}
|
||||||
|
err = LoadConfigAndData()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "Failed to load config and data: %s\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
token, err := GetToken(client)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "Failed to get token: %s\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
submission, err := GetSubmission(client, token, submissionId)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "Failed to get submission: %s\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
filename := strings.ReplaceAll(submission.Title, "/", "_") + "-" + submissionId
|
||||||
|
if filename[0] == '.' {
|
||||||
|
filename = "_" + filename[1:]
|
||||||
|
}
|
||||||
|
files, err := os.ReadDir(".")
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "Failed to list files: %s\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
stdin := bufio.NewReader(os.Stdin)
|
||||||
|
for _, i := range files {
|
||||||
|
if strings.HasPrefix(i.Name(), filename) {
|
||||||
|
fmt.Printf("A file that starts with %s exists (%s), potentially overwrite (y/N)? ", filename, i.Name())
|
||||||
|
b, err := stdin.ReadByte()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "Error when reading stdin: %s\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
if b != 'y' && b != 'Y' {
|
||||||
|
fmt.Println("Not overwriting")
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if submission.CrosspostParent != "" && len(submission.CrosspostParentList) > 0 {
|
||||||
|
submission, err = GetSubmission(client, token, submission.CrosspostParent[3:])
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "Failed to get original submission: %s\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if submission.IsSelf {
|
||||||
|
fmt.Fprintln(os.Stderr, "Cannot download selfposts")
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
urls := make([]string, 1)
|
||||||
|
urls[0] = submission.Url
|
||||||
|
if submission.IsVideo {
|
||||||
|
ffmpegPath, err := exec.LookPath("ffmpeg")
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "Failed to find ffmpeg: %s\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
ffmpegUrl := submission.SecureMedia.RedditVideo.HlsUrl
|
||||||
|
if ffmpegUrl == "" {
|
||||||
|
ffmpegUrl = submission.SecureMedia.RedditVideo.DashUrl
|
||||||
|
}
|
||||||
|
if ffmpegUrl != "" {
|
||||||
|
client.CloseIdleConnections()
|
||||||
|
err = unix.Exec(ffmpegPath, []string{"ffmpeg", "-nostdin", "-i", ffmpegUrl, "-c", "copy", "--", filename + ".mp4"}, os.Environ())
|
||||||
|
fmt.Fprintf(os.Stderr, "Failed to exec as ffmpeg: %s\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
fallbackUrl := submission.SecureMedia.RedditVideo.FallbackUrl
|
||||||
|
if fallbackUrl != "" {
|
||||||
|
urls[0] = fallbackUrl
|
||||||
|
}
|
||||||
|
} else if submission.IsGallery {
|
||||||
|
var galleryKeys []string
|
||||||
|
if submission.GalleryData != nil {
|
||||||
|
sort.Sort(submission.GalleryData)
|
||||||
|
for _, i := range submission.GalleryData.Items {
|
||||||
|
galleryKeys = append(galleryKeys, i.MediaId)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
for i, _ := range submission.MediaMetadata {
|
||||||
|
galleryKeys = append(galleryKeys, i)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
urls = nil
|
||||||
|
for _, i := range galleryKeys {
|
||||||
|
mediaMetadataItem := submission.MediaMetadata[i]
|
||||||
|
if mediaMetadataItem.Status != "valid" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if mediaMetadataItem.S.U != "" {
|
||||||
|
urls = append(urls, mediaMetadataItem.S.U)
|
||||||
|
} else if mediaMetadataItem.S.Mp4 != "" {
|
||||||
|
urls = append(urls, mediaMetadataItem.S.Mp4)
|
||||||
|
} else if mediaMetadataItem.S.Gif != "" {
|
||||||
|
urls = append(urls, mediaMetadataItem.S.Gif)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fmt.Println(urls)
|
||||||
|
i, err := InteractivelyAskIndex(stdin, urls)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "Failed to get index: %s\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
url := urls[i]
|
||||||
|
fmt.Println(url)
|
||||||
|
}
|
|
@ -0,0 +1,105 @@
|
||||||
|
package main
|
||||||
|
|
||||||
|
type Config struct {
|
||||||
|
ClientId string `json:"client_id"`
|
||||||
|
ClientSecret string `json:"client_secret"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Data struct {
|
||||||
|
AuthorizationHeader string `json:"authorization_header"`
|
||||||
|
AuthorizationExpiry int64 `json:"authorization_expiry"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type TokenResponse struct {
|
||||||
|
AccessToken string `json:"access_token"`
|
||||||
|
TokenType string `json:"token_type"`
|
||||||
|
ExpiresIn int64 `json:"expires_in"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type RedditVideo struct {
|
||||||
|
HlsUrl string `json:"hls_url"`
|
||||||
|
DashUrl string `json:"dash_url"`
|
||||||
|
FallbackUrl string `json:"fallback_url"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type SecureMedia struct {
|
||||||
|
RedditVideo *RedditVideo `json:"reddit_video"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type GalleryDataItem struct {
|
||||||
|
Id int `json:"id"`
|
||||||
|
MediaId string `json:"media_id"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type GalleryData struct {
|
||||||
|
Items []GalleryDataItem `json:"items"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s GalleryData) Len() int {
|
||||||
|
return len(s.Items)
|
||||||
|
}
|
||||||
|
func (s GalleryData) Swap(i, j int) {
|
||||||
|
s.Items[i], s.Items[j] = s.Items[j], s.Items[i]
|
||||||
|
}
|
||||||
|
func (s GalleryData) Less(i, j int) bool {
|
||||||
|
return s.Items[i].Id < s.Items[j].Id
|
||||||
|
}
|
||||||
|
|
||||||
|
// despite the name, it is not several items, but one
|
||||||
|
// ???
|
||||||
|
type MediaMetadataItemS struct {
|
||||||
|
U string `json:"u"`
|
||||||
|
Mp4 string `json:"mp4"`
|
||||||
|
Gif string `json:"gif"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type MediaMetadataItem struct {
|
||||||
|
Status string `json:"status"`
|
||||||
|
S *MediaMetadataItemS `json:"s"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type PreviewSource struct {
|
||||||
|
Url string `json:"url"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type PreviewVariants struct {
|
||||||
|
Mp4 *PreviewSource `json:"mp4"`
|
||||||
|
Gif *PreviewSource `json:"gif"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type PreviewImage struct {
|
||||||
|
Variants *PreviewVariants `json:"variants"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Preview struct {
|
||||||
|
Images []PreviewImage `json:"images"`
|
||||||
|
Source *PreviewSource `json:"source"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Submission struct {
|
||||||
|
CrosspostParent string `json:"crosspost_parent"`
|
||||||
|
// we don't care about the list, just does it exist or not
|
||||||
|
CrosspostParentList []interface{} `json:"crosspost_parent_list"`
|
||||||
|
Url string `json:"url"`
|
||||||
|
IsVideo bool `json:"is_video"`
|
||||||
|
SecureMedia *SecureMedia `json:"secure_media"`
|
||||||
|
IsGallery bool `json:"is_gallery"`
|
||||||
|
GalleryData *GalleryData `json:"gallery_data"`
|
||||||
|
MediaMetadata map[string]MediaMetadataItem `json:"media_metadata"`
|
||||||
|
IsRedditMediaDomain bool `json:"is_reddit_media_domain"`
|
||||||
|
Preview *Preview `json:"preview"`
|
||||||
|
IsSelf bool `json:"is_self"`
|
||||||
|
Title string `json:"title"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type SubmissionChild struct {
|
||||||
|
Data *Submission `json:"data"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type SubmissionResponseItemData struct {
|
||||||
|
Children []SubmissionChild `json:"children"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type SubmissionResponseItem struct {
|
||||||
|
Data SubmissionResponseItemData `json:"data"`
|
||||||
|
}
|
|
@ -0,0 +1,174 @@
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bufio"
|
||||||
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"io/fs"
|
||||||
|
"net/http"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
configDir string
|
||||||
|
config *Config
|
||||||
|
data *Data
|
||||||
|
)
|
||||||
|
|
||||||
|
func LoadConfigAndData() error {
|
||||||
|
iHateBugs, err := os.UserConfigDir()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("Cannot determine config directory: %s", err)
|
||||||
|
}
|
||||||
|
configDir = filepath.Join(iHateBugs, "omordl")
|
||||||
|
err = os.MkdirAll(configDir, 0o700)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("Cannot create config directory: %s", err)
|
||||||
|
}
|
||||||
|
configFile, err := os.OpenFile(filepath.Join(configDir, "config.json"), os.O_RDONLY, 0o600)
|
||||||
|
if err == nil {
|
||||||
|
contents, err := io.ReadAll(configFile)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("Failed to read config file: %s", err)
|
||||||
|
}
|
||||||
|
err = json.Unmarshal(contents, &config)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("Failed to parse config file: %s", err)
|
||||||
|
}
|
||||||
|
} else if errors.Is(err, fs.ErrNotExist) {
|
||||||
|
configFile, err = os.OpenFile(filepath.Join(configDir, "config.json"), os.O_WRONLY|os.O_CREATE|os.O_EXCL, 0o600)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("Failed to open config file for writing default template: %s", err)
|
||||||
|
}
|
||||||
|
_, err = configFile.WriteString(EXAMPLE_CONFIG)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("Failed to write to config file: %s", err)
|
||||||
|
}
|
||||||
|
return fmt.Errorf("Please fill out the default template at %s", filepath.Join(configDir, "config.json"))
|
||||||
|
} else {
|
||||||
|
return fmt.Errorf("Failed to open config file: %s", err)
|
||||||
|
}
|
||||||
|
if config.ClientId == CONFIG_PLACEHOLDER || config.ClientSecret == CONFIG_PLACEHOLDER {
|
||||||
|
return fmt.Errorf("Please fill out the default template at %s", filepath.Join(configDir, "config.json"))
|
||||||
|
}
|
||||||
|
dataFile, err := os.OpenFile(filepath.Join(configDir, "data.json"), os.O_RDONLY, 0o600)
|
||||||
|
if err == nil {
|
||||||
|
contents, err := io.ReadAll(dataFile)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("Failed to read data file: %s", err)
|
||||||
|
}
|
||||||
|
err = json.Unmarshal(contents, &data)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("Failed to parse data file: %s", err)
|
||||||
|
}
|
||||||
|
} else if !errors.Is(err, fs.ErrNotExist) {
|
||||||
|
return fmt.Errorf("Failed to open data file: %s", err)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func WriteData() error {
|
||||||
|
contents, err := json.Marshal(data)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
file, err := os.OpenFile(filepath.Join(configDir, "data.json"), os.O_WRONLY|os.O_CREATE, 0o600)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
_, err = file.Write(contents)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetToken(client *http.Client) (string, error) {
|
||||||
|
if data.AuthorizationHeader != "" && data.AuthorizationExpiry != 0 && data.AuthorizationExpiry > time.Now().Unix() {
|
||||||
|
return data.AuthorizationHeader, nil
|
||||||
|
}
|
||||||
|
request, err := http.NewRequest("POST", "https://www.reddit.com/api/v1/access_token?grant_type=client_credentials", nil)
|
||||||
|
if err != nil {
|
||||||
|
return "", fmt.Errorf("Failed to create request: %s", err)
|
||||||
|
}
|
||||||
|
request.Header.Add("User-Agent", USER_AGENT)
|
||||||
|
request.SetBasicAuth(config.ClientId, config.ClientSecret)
|
||||||
|
response, err := client.Do(request)
|
||||||
|
if err != nil {
|
||||||
|
return "", fmt.Errorf("Failed to get response: %s", err)
|
||||||
|
}
|
||||||
|
contents, err := io.ReadAll(response.Body)
|
||||||
|
response.Body.Close()
|
||||||
|
if err != nil {
|
||||||
|
return "", fmt.Errorf("Failed to read response body: %s", err)
|
||||||
|
}
|
||||||
|
if response.StatusCode != 200 {
|
||||||
|
return "", fmt.Errorf("Response returned status code %d, body: %s", response.StatusCode, contents)
|
||||||
|
}
|
||||||
|
var tokenResponse TokenResponse
|
||||||
|
err = json.Unmarshal(contents, &tokenResponse)
|
||||||
|
if err != nil {
|
||||||
|
return "", fmt.Errorf("Failed to parse response: %s", err)
|
||||||
|
}
|
||||||
|
s := []string{tokenResponse.TokenType, tokenResponse.AccessToken}
|
||||||
|
data.AuthorizationHeader = strings.Join(s, " ")
|
||||||
|
data.AuthorizationExpiry = time.Now().Unix() + tokenResponse.ExpiresIn - 5
|
||||||
|
err = WriteData()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "Warning: Failed to save token to disk: %s\n", err)
|
||||||
|
}
|
||||||
|
return data.AuthorizationHeader, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetSubmission(client *http.Client, token, submissionId string) (*Submission, error) {
|
||||||
|
request, err := http.NewRequest("GET", "https://oauth.reddit.com/comments/"+submissionId+"/?raw_json=1&limit=1", nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("Failed to create request: %s", err)
|
||||||
|
}
|
||||||
|
request.Header.Add("User-Agent", USER_AGENT)
|
||||||
|
request.Header.Add("Authorization", data.AuthorizationHeader)
|
||||||
|
response, err := client.Do(request)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("Failed to get response: %s", err)
|
||||||
|
}
|
||||||
|
contents, err := io.ReadAll(response.Body)
|
||||||
|
response.Body.Close()
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("Failed to read response body: %s", err)
|
||||||
|
}
|
||||||
|
if response.StatusCode != 200 {
|
||||||
|
return nil, fmt.Errorf("Response returned status code %d, body: %s", response.StatusCode, contents)
|
||||||
|
}
|
||||||
|
var submission []SubmissionResponseItem
|
||||||
|
err = json.Unmarshal(contents, &submission)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("Failed to parse response: %s", err)
|
||||||
|
}
|
||||||
|
return submission[0].Data.Children[0].Data, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func InteractivelyAskIndex(stdin *bufio.Reader, items []string) (int, error) {
|
||||||
|
if len(items) == 1 {
|
||||||
|
return 0, nil
|
||||||
|
}
|
||||||
|
fmt.Printf("Select an item (from 1 to %d): ", len(items))
|
||||||
|
str, err := stdin.ReadString('\n')
|
||||||
|
if err != nil {
|
||||||
|
return 0, fmt.Errorf("Failed to read stdin: %s", err)
|
||||||
|
}
|
||||||
|
i, err := strconv.Atoi(strings.TrimSpace(str))
|
||||||
|
if err != nil {
|
||||||
|
return 0, fmt.Errorf("Failed to parse stdin: %s", err)
|
||||||
|
}
|
||||||
|
i--
|
||||||
|
if i < 0 {
|
||||||
|
return i, errors.New("Index is under or equal to 0")
|
||||||
|
}
|
||||||
|
if i >= len(items) {
|
||||||
|
return i, errors.New("Index is bigger than items available")
|
||||||
|
}
|
||||||
|
return i, nil
|
||||||
|
}
|
Reference in New Issue