X-Git-Url: https://njoseph.me/gitweb/nimcoon.git/blobdiff_plain/68fd8100bc36465b29ae76b9f41d9eb7fa052e7a..121e06b2f5cb4a48cd9190d4a6b88ca756512d6d:/clitube.nim diff --git a/clitube.nim b/clitube.nim index a1ba868..b15d5a9 100644 --- a/clitube.nim +++ b/clitube.nim @@ -1,51 +1,99 @@ -import htmlparser -import httpClient -import os -import osproc -import sequtils, sugar -import strtabs -import strutils -import uri -import xmltree +import + htmlparser, + httpClient, + parseopt, + osproc, + sequtils, + sugar, + strformat, + std/[terminal], + strtabs, + strutils, + uri, + xmltree -# Supported video players in order of preference -let supportedPlayers = @["mpv", "mplayer", "vlc"] +import preferences -# Hard-coded terminal colors -proc reset(): string {.procvar.} = "\e[0m" -proc bold*(s: string): string {.procvar.} = "\e[1m" & s & reset() - -proc fgMagenta*(s: string): string {.procvar.} = "\e[35m" & s & reset() -proc fgCyan*(s: string): string {.procvar.} = "\e[36m" & s & reset() - -type SearchResult = tuple[title: string, url: string] +type + SearchResult = tuple[title: string, url: string] + CommandLineOptions = tuple[searchQuery: string, musicOnly: bool, feelingLucky: bool] proc selectMediaPlayer(): string = let availablePlayers = filterIt(supportedPlayers, execProcess("which " & it).len != 0) if len(availablePlayers) == 0: - echo "Please install one of the supported media players: ", supportedPlayers + stderr.writeLine &"Please install one of the supported media players: {supportedPlayers}" raise newException(OSError, "No supported media player found") else: return availablePlayers[0] +proc parseOptions(): CommandLineOptions = + var + searchQuery = "" + musicOnly = false + feelingLucky = false + + for kind, key, value in getopt(): + case kind + of cmdArgument: + searchQuery = key + of cmdShortOption, cmdLongOption: + case key + of "m", "music": musicOnly = true + of "l", "lucky": feelingLucky = true + of cmdEnd: + discard + + return (searchQuery, musicOnly, feelingLucky) + proc getYoutubePage(searchQuery: string): string = let queryParam = encodeUrl(searchQuery) - var client = newHttpClient() - let response = get(client, "https://www.youtube.com/results?hl=en&search_query=" & queryParam) - # TODO Get rid of temp file or make one temp file per user - writeFile("/tmp/cli-tube-page.html", response.body) - return "/tmp/cli-tube-page.html" + let client = newHttpClient() + let response = get(client, &"https://www.youtube.com/results?hl=en&search_query={queryParam}") + return $response.body proc extractTitlesAndUrls(htmlFile: string): seq[SearchResult] = - loadHtml(htmlFile).findAll("a"). + parseHtml(htmlFile).findAll("a"). filter(a => "watch" in a.attrs["href"] and a.attrs.hasKey "title"). - map(a => (a.attrs["title"], "https://www.youtube.com" & a.attrs["href"])) + map(a => (a.attrs["title"], "https://www.youtube.com" & a.attrs["href"]))[..(limit-1)] proc presentVideoOptions(searchResults: seq[SearchResult]) = + echo "" for index, (title, url) in searchResults: - echo index, ". ", title.bold.fgMagenta, "\n", url.fgCyan, "\n" + styledEcho $index, ". ", styleBright, fgMagenta, title, "\n", resetStyle, fgCyan, url, "\n" + +proc directPlay(searchQuery: string, player: string) = + styledEcho "\n", fgGreen, "Playing ", styleBright, fgMagenta, searchQuery + if "watch?" in searchQuery or "videos/watch" in searchQuery : + discard execProcess(&"{player} {searchQuery}") + quit(0) + elif searchQuery.startswith("magnet:"): + discard execProcess(&"peerflix \"{searchQuery}\" --{player}") + quit(0) + +proc main() = + let + player = selectMediaPlayer() + (searchQuery, musicOnly, feelingLucky) = parseOptions() + + directPlay(searchQuery, player) + + let searchResults = extractTitlesAndUrls(getYoutubePage(searchQuery)) + + let number = + if feelingLucky: 0 + else: + presentVideoOptions(searchResults) + stdout.styledWrite(fgYellow, "Choose video number: ") + parseInt(readLine(stdin)) + + styledEcho "\n", fgGreen, "Playing ", styleBright, fgMagenta, searchResults[number].title + + var command = @[player, searchResults[number].url] + + if musicOnly: + command.add("--no-video") + + # Play the video using the preferred/available media player + discard execProcess(command.join(" ")) -presentVideoOptions( - extractTitlesAndUrls( - getYoutubePage( - paramStr(1)))) +main()