Add watchlist, enable migrations
This commit is contained in:
parent
f8aeed318f
commit
db4a42171d
|
@ -1,17 +1,22 @@
|
||||||
@page
|
@page
|
||||||
|
@using global::c3stream.DataModels
|
||||||
@model IndexModel
|
@model IndexModel
|
||||||
@{
|
@{
|
||||||
ViewData["Title"] = "Home";
|
ViewData["Title"] = "Home";
|
||||||
var cookie = c3stream.UpdateCookie(Request, Response, "/");
|
var cookie = c3stream.UpdateCookie(Request, Response, "/");
|
||||||
|
var marked = new Database.DbConn().States.Any(p => p.UserId == cookie && p.State == "marked");
|
||||||
}
|
}
|
||||||
|
|
||||||
<div style="text-align: center">
|
<div style="text-align: center">
|
||||||
<h1>Welcome to c3stream!</h1>
|
<h1>Welcome to c3stream!</h1>
|
||||||
Your bookmark link:<br/>
|
Your bookmark link:<br/>
|
||||||
<code onclick="copyToClipboard(this)">https://@Request.Host.Value?bookmark=@cookie</code><br/><br/>
|
<code onclick="copyToClipboard(this)">https://@Request.Host.Value?bookmark=@cookie</code><br/><br/>
|
||||||
<div class="btn-group">
|
<div class="btn-group">
|
||||||
@foreach (var conf in c3stream.Conferences) {
|
@if (marked) {
|
||||||
<a role="button" class="btn btn-primary" href="/Conference?c=@conf.Acronym">@conf.Acronym</a>
|
<a role="button" class="btn btn-primary" href="/Watchlist">watchlist</a>
|
||||||
}
|
}
|
||||||
</div>
|
@foreach (var conf in c3stream.Conferences) {
|
||||||
|
<a role="button" class="btn btn-primary" href="/Conference?c=@conf.Acronym">@conf.Acronym</a>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
110
Pages/Watchlist.cshtml
Normal file
110
Pages/Watchlist.cshtml
Normal file
|
@ -0,0 +1,110 @@
|
||||||
|
@page
|
||||||
|
@model WatchlistModel
|
||||||
|
@using System.Net
|
||||||
|
@using global::c3stream.DataModels
|
||||||
|
@using static WatchlistModel
|
||||||
|
@{
|
||||||
|
var cookie = c3stream.UpdateCookie(Request, Response, $"/Watchlist");
|
||||||
|
ViewData["Title"] = "Watchlist";
|
||||||
|
await using var db = new Database.DbConn();
|
||||||
|
var states = db.States.ToList();
|
||||||
|
var marked = db.States.Where(p => p.UserId == cookie && p.State == "marked").Select(p => p.TalkId).ToList();
|
||||||
|
var watchlist = c3stream.GetEventsByGuid(marked);
|
||||||
|
}
|
||||||
|
|
||||||
|
<table class="table">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th scope="col">Conference</th>
|
||||||
|
<th scope="col">Event</th>
|
||||||
|
<th scope="col">
|
||||||
|
@Html.Raw(Request.Query["orderby"] == "published" ? $"<a href=\"/Conference?c={Request.Query["c"]}\">Published" : $"<a href=\"/Conference?c={Request.Query["c"]}&orderby=published\">Date")
|
||||||
|
</th>
|
||||||
|
<th scope="col">Category</th>
|
||||||
|
<th scope="col">Title</th>
|
||||||
|
<th scope="col">Speaker(s)</th>
|
||||||
|
<th scope="col">Lang</th>
|
||||||
|
<th scope="col">Actions</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
@foreach (var talk in Request.Query["orderby"] == "published" ? watchlist.OrderByDescending(p => p.ReleaseDate) : watchlist.OrderBy(p => p.Date)) {
|
||||||
|
var state = states.FirstOrDefault(p => p.TalkId == talk.Guid && p.UserId == cookie)?.State;
|
||||||
|
var isWatched = state == "watched";
|
||||||
|
var isMarked = state == "marked";
|
||||||
|
var file = $"{talk.Slug}.mp4";
|
||||||
|
var conference = c3stream.GetConferenceByEventGuid(talk.Guid);
|
||||||
|
var eventName = talk.Tags.Count <= 1 ? conference.Acronym : talk.Tags[0].Replace("-", "-<br/>");
|
||||||
|
var category = talk.Tags.Count switch {
|
||||||
|
0 => "<no category>",
|
||||||
|
1 => talk.Tags[0],
|
||||||
|
2 => "<no category>",
|
||||||
|
3 => talk.Tags[2],
|
||||||
|
4 => talk.Tags[3],
|
||||||
|
5 => talk.Tags[3],
|
||||||
|
6 => talk.Tags[3], // rc3: is this correct?
|
||||||
|
_ => "<unknown tag format>"
|
||||||
|
};
|
||||||
|
<tr>
|
||||||
|
<td>@Html.Raw(eventName)</td>
|
||||||
|
<td>@(Request.Query["orderby"] == "published" ? talk.ReleaseDate?.Date.ToShortDateString() : talk.Date?.Date.ToShortDateString())</td>
|
||||||
|
<td>@category</td>
|
||||||
|
@if (isWatched) {
|
||||||
|
<td style="color: #95cb7a">@talk.Title</td>
|
||||||
|
}
|
||||||
|
else if (isMarked) {
|
||||||
|
<td style="color: #da7d4f">@talk.Title</td>
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
<td>@talk.Title</td>
|
||||||
|
}
|
||||||
|
<td>@(talk.Persons.Any() ? talk.Persons.Aggregate((s, s1) => $"{s}, {s1}") : "<no speakers>")</td>
|
||||||
|
<td>@talk.OriginalLanguage</td>
|
||||||
|
<td>
|
||||||
|
<div class="btn-group" role="group">
|
||||||
|
<a href="@talk.FrontendLink.AbsoluteUri" role="button" class="btn btn-primary btn-c3saction w-100" data-toggle="tooltip" data-placement="top" title="Play">
|
||||||
|
<i class="fas fa-play-circle"></i>
|
||||||
|
</a>
|
||||||
|
@if (System.IO.File.Exists(System.IO.Path.Combine(c3stream.CachePath, conference.Acronym, file))) {
|
||||||
|
<a href="@(c3stream.CacheUrl + $"{conference.Acronym}/{file}")" role="button" class="btn btn-primary btn-c3saction w-100" data-toggle="tooltip" data-placement="top" title="Mirror">
|
||||||
|
<i class="fas fa-cloud-download"></i>
|
||||||
|
</a>
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
<a href="/" role="button" class="btn btn-primary btn-c3saction disabled">
|
||||||
|
<i class="fas fa-cloud-download"></i>
|
||||||
|
</a>
|
||||||
|
}
|
||||||
|
<a href="/Info?guid=@talk.Guid" role="button" class="btn btn-primary btn-c3saction w-100" data-toggle="tooltip" data-placement="top" title="Info">
|
||||||
|
<i class="fas fa-info-circle"></i>
|
||||||
|
</a>
|
||||||
|
@if (isWatched) {
|
||||||
|
<button onclick="SetState('@talk.Guid', 'unwatched')" class="btn btn-primary btn-c3saction w-100" data-toggle="tooltip" data-placement="top" title="Mark unwatched">
|
||||||
|
<i class="fas fa-times"></i>
|
||||||
|
</button>
|
||||||
|
<button class="btn btn-primary btn-c3saction disabled">
|
||||||
|
<i class="fas fa-clock"></i>
|
||||||
|
</button>
|
||||||
|
}
|
||||||
|
else if (isMarked) {
|
||||||
|
<button onclick="SetState('@talk.Guid', 'watched')" class="btn btn-primary btn-c3saction w-100" data-toggle="tooltip" data-placement="top" title="Mark watched">
|
||||||
|
<i class="fas fa-check"></i>
|
||||||
|
</button>
|
||||||
|
<button onclick="SetState('@talk.Guid', 'unwatched')" class="btn btn-primary btn-c3saction w-100" data-toggle="tooltip" data-placement="top" title="Remove from watch later">
|
||||||
|
<i class="fas fa-undo-alt"></i>
|
||||||
|
</button>
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
<button onclick="SetState('@talk.Guid', 'watched')" class="btn btn-primary btn-c3saction w-100" data-toggle="tooltip" data-placement="top" title="Mark watched">
|
||||||
|
<i class="fas fa-check"></i>
|
||||||
|
</button>
|
||||||
|
<button onclick="SetState('@talk.Guid', 'marked')" class="btn btn-primary btn-c3saction w-100" data-toggle="tooltip" data-placement="top" title="Add to watch later">
|
||||||
|
<i class="fas fa-clock"></i>
|
||||||
|
</button>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
39
Pages/Watchlist.cshtml.cs
Normal file
39
Pages/Watchlist.cshtml.cs
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using c3stream.DataModels;
|
||||||
|
using c3stream.DataModels.Tables;
|
||||||
|
using LinqToDB;
|
||||||
|
using Microsoft.AspNetCore.Mvc.RazorPages;
|
||||||
|
using Microsoft.Extensions.Logging;
|
||||||
|
using Newtonsoft.Json;
|
||||||
|
|
||||||
|
namespace c3stream.Pages {
|
||||||
|
public class WatchlistModel : PageModel {
|
||||||
|
private readonly ILogger<ConferenceModel> _logger;
|
||||||
|
|
||||||
|
public WatchlistModel(ILogger<ConferenceModel> logger) => _logger = logger;
|
||||||
|
|
||||||
|
public void OnGet() {
|
||||||
|
var guid = Request.Query["guid"].ToString();
|
||||||
|
var state = Request.Query["state"].ToString();
|
||||||
|
var userid = Request.Cookies["bookmark"];
|
||||||
|
if (string.IsNullOrWhiteSpace(guid) || string.IsNullOrWhiteSpace(state) || !Request.Cookies.ContainsKey("bookmark"))
|
||||||
|
return;
|
||||||
|
|
||||||
|
using var db = new Database.DbConn();
|
||||||
|
var existing = db.States.FirstOrDefault(p => p.TalkId == guid && p.UserId == userid);
|
||||||
|
if (existing != null)
|
||||||
|
if (state == "unwatched")
|
||||||
|
db.States.Delete(p => p == existing);
|
||||||
|
else {
|
||||||
|
existing.State = state;
|
||||||
|
db.Update(existing);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
db.Insert(new States {
|
||||||
|
State = state, TalkId = guid, UserId = userid
|
||||||
|
});
|
||||||
|
Response.Redirect("/");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
210
c3stream.cs
210
c3stream.cs
|
@ -9,127 +9,129 @@ using Microsoft.AspNetCore.Hosting;
|
||||||
using Microsoft.AspNetCore.Http;
|
using Microsoft.AspNetCore.Http;
|
||||||
using Microsoft.Extensions.Hosting;
|
using Microsoft.Extensions.Hosting;
|
||||||
|
|
||||||
namespace c3stream {
|
namespace c3stream;
|
||||||
public static class c3stream {
|
|
||||||
public const string DataPath = "data";
|
|
||||||
public const string DbFile = "c3stream.sqlite";
|
|
||||||
public const string CachePath = "/mnt/storage/archive/Video/congress/";
|
|
||||||
public const string CacheUrl = "https://mirror.c3stream.de/";
|
|
||||||
public static object Lock = new();
|
|
||||||
public static string DbPath = Path.Combine(DataPath, DbFile);
|
|
||||||
|
|
||||||
public static readonly List<ConferenceObject> Conferences = new() {
|
public static class c3stream {
|
||||||
new ConferenceObject("rc3-2021", true),
|
public const string DataPath = "data";
|
||||||
new ConferenceObject("rc3"),
|
public const string DbFile = "c3stream.sqlite";
|
||||||
new ConferenceObject("36c3"),
|
public const string CachePath = "/mnt/storage/archive/Video/congress/";
|
||||||
new ConferenceObject("camp2019"),
|
public const string CacheUrl = "https://mirror.c3stream.de/";
|
||||||
new ConferenceObject("gpn19"),
|
public static object Lock = new();
|
||||||
new ConferenceObject("35c3"),
|
public static string DbPath = Path.Combine(DataPath, DbFile);
|
||||||
new ConferenceObject("34c3"),
|
|
||||||
new ConferenceObject("33c3"),
|
|
||||||
new ConferenceObject("32c3"),
|
|
||||||
new ConferenceObject("31c3"),
|
|
||||||
new ConferenceObject("30c3")
|
|
||||||
};
|
|
||||||
|
|
||||||
public static void Main(string[] args) {
|
public static readonly List<ConferenceObject> Conferences = new() {
|
||||||
if (!Directory.Exists(DataPath))
|
new ConferenceObject("rc3-2021", true),
|
||||||
Directory.CreateDirectory(DataPath);
|
new ConferenceObject("rc3"),
|
||||||
if (!File.Exists(DbPath))
|
new ConferenceObject("36c3"),
|
||||||
File.Copy(Path.Combine(DataPath, "database.init.sqlite"), DbPath);
|
new ConferenceObject("camp2019"),
|
||||||
|
new ConferenceObject("gpn19"),
|
||||||
|
new ConferenceObject("35c3"),
|
||||||
|
new ConferenceObject("34c3"),
|
||||||
|
new ConferenceObject("33c3"),
|
||||||
|
new ConferenceObject("32c3"),
|
||||||
|
new ConferenceObject("31c3"),
|
||||||
|
new ConferenceObject("30c3")
|
||||||
|
};
|
||||||
|
|
||||||
DataConnection.DefaultSettings = new Database.Settings();
|
public static void Main(string[] args) {
|
||||||
|
if (!Directory.Exists(DataPath))
|
||||||
|
Directory.CreateDirectory(DataPath);
|
||||||
|
if (!File.Exists(DbPath))
|
||||||
|
File.Copy(Path.Combine(DataPath, "database.init.sqlite"), DbPath);
|
||||||
|
|
||||||
foreach (var conference in Conferences)
|
DataConnection.DefaultSettings = new Database.Settings();
|
||||||
UpdateConference(conference);
|
Migrations.RunMigrations();
|
||||||
|
|
||||||
if (args.Length != 0) {
|
foreach (var conference in Conferences)
|
||||||
if (args[0] == "logo")
|
UpdateConference(conference);
|
||||||
foreach (var conference in Conferences) {
|
|
||||||
Console.WriteLine($"wget {conference.LogoUri} -O {Path.Combine(CachePath, conference.Acronym, "logo.png")}");
|
if (args.Length != 0) {
|
||||||
}
|
if (args[0] == "logo")
|
||||||
else if (Conferences.All(p => p.Acronym != args[0]))
|
foreach (var conference in Conferences)
|
||||||
Console.WriteLine("No matching conference found.");
|
Console.WriteLine($"wget {conference.LogoUri} -O {Path.Combine(CachePath, conference.Acronym, "logo.png")}");
|
||||||
else
|
else if (Conferences.All(p => p.Acronym != args[0]))
|
||||||
foreach (var talk in Conferences.First(p => p.Acronym == args[0]).Talks)
|
Console.WriteLine("No matching conference found.");
|
||||||
Console.WriteLine($"youtube-dl -f \"best[ext = mp4]\" {talk.FrontendLink} -o \"{Path.Combine(CachePath, args[0], talk.Slug)}.mp4\"");
|
else
|
||||||
}
|
foreach (var talk in Conferences.First(p => p.Acronym == args[0]).Talks)
|
||||||
else {
|
Console.WriteLine($"youtube-dl -f \"best[ext = mp4]\" {talk.FrontendLink} -o \"{Path.Combine(CachePath, args[0], talk.Slug)}.mp4\"");
|
||||||
CreateHostBuilder(args).Build().Run();
|
}
|
||||||
}
|
else {
|
||||||
|
CreateHostBuilder(args).Build().Run();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//TODO: move this to the database as well
|
||||||
|
public static void UpdateConference(ConferenceObject conference) {
|
||||||
|
using var httpc = new HttpClient();
|
||||||
|
|
||||||
|
var jsonpath = Path.Combine(DataPath, conference.Acronym + "_index.json");
|
||||||
|
var json = "";
|
||||||
|
if (!File.Exists(jsonpath)) {
|
||||||
|
json = httpc.GetStringAsync($"https://api.media.ccc.de/public/conferences/{conference.Acronym}").Result;
|
||||||
|
File.WriteAllText(jsonpath, json);
|
||||||
|
}
|
||||||
|
else if (conference.Ongoing) {
|
||||||
|
json = httpc.GetStringAsync($"https://api.media.ccc.de/public/conferences/{conference.Acronym}").Result;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
json = File.ReadAllText(jsonpath);
|
||||||
}
|
}
|
||||||
|
|
||||||
//TODO: move this to the database as well
|
var parsed = Conference.FromJson(json);
|
||||||
public static void UpdateConference(ConferenceObject conference) {
|
lock (Lock) {
|
||||||
using var httpc = new HttpClient();
|
conference.Talks.Clear();
|
||||||
|
conference.LogoUri = parsed.LogoUrl.AbsoluteUri;
|
||||||
|
conference.Talks.AddRange(parsed.Events);
|
||||||
|
conference.Talks.ForEach(p => p.Guid = p.Guid.Trim());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
var jsonpath = Path.Combine(DataPath, conference.Acronym + "_index.json");
|
public static string UpdateCookie(HttpRequest request, HttpResponse response, string redirectUri) {
|
||||||
var json = "";
|
var cookie = "";
|
||||||
if (!File.Exists(jsonpath)) {
|
//if new bookmark is in uri
|
||||||
json = httpc.GetStringAsync($"https://api.media.ccc.de/public/conferences/{conference.Acronym}").Result;
|
if (request.Query.ContainsKey("bookmark") && Guid.TryParseExact(request.Query["bookmark"], "D", out _)) {
|
||||||
File.WriteAllText(jsonpath, json);
|
response.Cookies.Append("bookmark", request.Query["bookmark"], new CookieOptions { Expires = DateTimeOffset.MaxValue });
|
||||||
}
|
cookie = request.Query["bookmark"];
|
||||||
else if (conference.Ongoing) {
|
}
|
||||||
json = httpc.GetStringAsync($"https://api.media.ccc.de/public/conferences/{conference.Acronym}").Result;
|
//if no cookie exists or cookie is invalid
|
||||||
}
|
else if (!request.Cookies.ContainsKey("bookmark") || !Guid.TryParseExact(request.Cookies["bookmark"], "D", out _)) {
|
||||||
else {
|
var guid = Guid.NewGuid().ToString();
|
||||||
json = File.ReadAllText(jsonpath);
|
response.Cookies.Append("bookmark", guid, new CookieOptions { Expires = DateTimeOffset.MaxValue });
|
||||||
}
|
cookie = guid;
|
||||||
|
}
|
||||||
var parsed = Conference.FromJson(json);
|
else {
|
||||||
lock (Lock) {
|
cookie = request.Cookies["bookmark"];
|
||||||
conference.Talks.Clear();
|
|
||||||
conference.LogoUri = parsed.LogoUrl.AbsoluteUri;
|
|
||||||
conference.Talks.AddRange(parsed.Events);
|
|
||||||
conference.Talks.ForEach(p => p.Guid = p.Guid.Trim());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static string UpdateCookie(HttpRequest request, HttpResponse response, string redirectUri) {
|
if (request.Query.ContainsKey("bookmark"))
|
||||||
var cookie = "";
|
response.Redirect(redirectUri);
|
||||||
//if new bookmark is in uri
|
|
||||||
if (request.Query.ContainsKey("bookmark") && Guid.TryParseExact(request.Query["bookmark"], "D", out _)) {
|
|
||||||
response.Cookies.Append("bookmark", request.Query["bookmark"], new CookieOptions { Expires = DateTimeOffset.MaxValue });
|
|
||||||
cookie = request.Query["bookmark"];
|
|
||||||
}
|
|
||||||
//if no cookie exists or cookie is invalid
|
|
||||||
else if (!request.Cookies.ContainsKey("bookmark") || !Guid.TryParseExact(request.Cookies["bookmark"], "D", out _)) {
|
|
||||||
var guid = Guid.NewGuid().ToString();
|
|
||||||
response.Cookies.Append("bookmark", guid, new CookieOptions { Expires = DateTimeOffset.MaxValue });
|
|
||||||
cookie = guid;
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
cookie = request.Cookies["bookmark"];
|
|
||||||
}
|
|
||||||
|
|
||||||
if (request.Query.ContainsKey("bookmark")) {
|
return cookie;
|
||||||
response.Redirect(redirectUri);
|
}
|
||||||
}
|
|
||||||
|
|
||||||
return cookie;
|
public static Event GetEventByGuid(string guid) {
|
||||||
}
|
return Conferences.SelectMany(c => c.Talks.Where(e => e.Guid == guid)).FirstOrDefault();
|
||||||
|
}
|
||||||
|
|
||||||
public static Event GetEventByGuid(string guid) {
|
public static IEnumerable<Event> GetEventsByGuid(IEnumerable<string> guids) {
|
||||||
return Conferences.SelectMany(c => c.Talks.Where(talk => talk.Guid.ToString() == guid)).FirstOrDefault();
|
return Conferences.SelectMany(c => c.Talks.Where(e => guids.Contains(e.Guid)));
|
||||||
}
|
}
|
||||||
|
|
||||||
public static ConferenceObject GetConferenceByEventGuid(string guid) {
|
public static ConferenceObject GetConferenceByEventGuid(string guid) {
|
||||||
return Conferences.FirstOrDefault(c => c.Talks.Any(t => t.Guid.ToString() == guid));
|
return Conferences.FirstOrDefault(c => c.Talks.Any(t => t.Guid == guid));
|
||||||
}
|
}
|
||||||
|
|
||||||
public static IHostBuilder CreateHostBuilder(string[] args) =>
|
public static IHostBuilder CreateHostBuilder(string[] args) => Host.CreateDefaultBuilder(args).ConfigureWebHostDefaults(webBuilder => { webBuilder.UseStartup<Startup>(); });
|
||||||
Host.CreateDefaultBuilder(args).ConfigureWebHostDefaults(webBuilder => { webBuilder.UseStartup<Startup>(); });
|
|
||||||
|
|
||||||
public class ConferenceObject {
|
public class ConferenceObject {
|
||||||
public string Acronym;
|
public string Acronym;
|
||||||
public bool Ongoing;
|
public string LogoUri;
|
||||||
public string LogoUri;
|
public bool Ongoing;
|
||||||
public List<Event> Talks = new();
|
public List<Event> Talks = new();
|
||||||
|
|
||||||
public ConferenceObject(string acronym, bool ongoing = false) {
|
public ConferenceObject(string acronym, bool ongoing = false) {
|
||||||
Acronym = acronym;
|
Acronym = acronym;
|
||||||
Ongoing = ongoing;
|
Ongoing = ongoing;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
Loading…
Reference in a new issue