Compare commits
No commits in common. "v5" and "v3" have entirely different histories.
2
.gitignore
vendored
2
.gitignore
vendored
|
@ -2,5 +2,3 @@ bin/
|
|||
obj/
|
||||
/packages/
|
||||
/data/
|
||||
migration.sql
|
||||
.idea/
|
||||
|
|
0
.idea/.gitignore
vendored
Normal file
0
.idea/.gitignore
vendored
Normal file
2
.idea/.idea.c3stream/.idea/.gitignore
vendored
Normal file
2
.idea/.idea.c3stream/.idea/.gitignore
vendored
Normal file
|
@ -0,0 +1,2 @@
|
|||
# Default ignored files
|
||||
/workspace.xml
|
142
.idea/.idea.c3stream/.idea/contentModel.xml
Normal file
142
.idea/.idea.c3stream/.idea/contentModel.xml
Normal file
|
@ -0,0 +1,142 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="ContentModelStore">
|
||||
<e p="$USER_HOME$/Library/Caches/Rider2019.3/extResources" t="IncludeRecursive" />
|
||||
<e p="$USER_HOME$/Library/Caches/Rider2019.3/resharper-host/local/Transient/ReSharperHost/v193/SolutionCaches/_c3stream.-879192168.00" t="ExcludeRecursive" />
|
||||
<e p="$APPLICATION_CONFIG_DIR$/javascript/extLibs/http_kit.fontawesome.com_c1a632a160.js" t="IncludeRecursive" />
|
||||
<e p="$APPLICATION_CONFIG_DIR$/javascript/extLibs/http_pro.fontawesome.com_releases_v5.11.2_css_all.css" t="IncludeRecursive" />
|
||||
<e p="$PROJECT_DIR$" t="IncludeRecursive">
|
||||
<e p=".gitignore" t="Include" />
|
||||
<e p="appsettings.Development.json" t="Include" />
|
||||
<e p="appsettings.json" t="Include" />
|
||||
<e p="bin" t="ExcludeRecursive" />
|
||||
<e p="c3stream.cs" t="Include" />
|
||||
<e p="c3stream.csproj" t="IncludeRecursive" />
|
||||
<e p="c3stream.sln" t="IncludeFlat" />
|
||||
<e p="LICENSE" t="Include" />
|
||||
<e p="obj" t="ExcludeRecursive">
|
||||
<e p="Debug" t="Include">
|
||||
<e p="netcoreapp3.1" t="Include">
|
||||
<e p="c3stream.AssemblyInfo.cs" t="Include" />
|
||||
<e p="c3stream.RazorAssemblyInfo.cs" t="Include" />
|
||||
</e>
|
||||
</e>
|
||||
</e>
|
||||
<e p="packages" t="ExcludeRecursive" />
|
||||
<e p="Pages" t="Include">
|
||||
<e p="_ViewImports.cshtml" t="Include" />
|
||||
<e p="_ViewStart.cshtml" t="Include" />
|
||||
<e p="Conference.cshtml" t="Include" />
|
||||
<e p="Conference.cshtml.cs" t="Include" />
|
||||
<e p="Error.cshtml" t="Include" />
|
||||
<e p="Error.cshtml.cs" t="Include" />
|
||||
<e p="Index.cshtml" t="Include" />
|
||||
<e p="Index.cshtml.cs" t="Include" />
|
||||
<e p="Info.cshtml" t="Include" />
|
||||
<e p="Info.cshtml.cs" t="Include" />
|
||||
<e p="Privacy.cshtml" t="Include" />
|
||||
<e p="Privacy.cshtml.cs" t="Include" />
|
||||
<e p="Shared" t="Include">
|
||||
<e p="_Layout.cshtml" t="Include" />
|
||||
<e p="_ValidationScriptsPartial.cshtml" t="Include" />
|
||||
</e>
|
||||
</e>
|
||||
<e p="Properties" t="Include">
|
||||
<e p="launchSettings.json" t="Include" />
|
||||
</e>
|
||||
<e p="README.md" t="Include" />
|
||||
<e p="Startup.cs" t="Include" />
|
||||
<e p="Types.cs" t="Include" />
|
||||
<e p="wwwroot" t="Include">
|
||||
<e p="css" t="Include">
|
||||
<e p="fa.css" t="Include" />
|
||||
<e p="site.css" t="Include" />
|
||||
</e>
|
||||
<e p="favicon.ico" t="Include" />
|
||||
<e p="js" t="Include">
|
||||
<e p="site.js" t="Include" />
|
||||
</e>
|
||||
<e p="lib" t="Include">
|
||||
<e p="bootstrap" t="Include">
|
||||
<e p="dist" t="Include">
|
||||
<e p="css" t="Include">
|
||||
<e p="bootstrap-grid.css" t="Include" />
|
||||
<e p="bootstrap-grid.css.map" t="Include" />
|
||||
<e p="bootstrap-grid.min.css" t="Include" />
|
||||
<e p="bootstrap-grid.min.css.map" t="Include" />
|
||||
<e p="bootstrap-reboot.css" t="Include" />
|
||||
<e p="bootstrap-reboot.css.map" t="Include" />
|
||||
<e p="bootstrap-reboot.min.css" t="Include" />
|
||||
<e p="bootstrap-reboot.min.css.map" t="Include" />
|
||||
<e p="bootstrap.css" t="Include" />
|
||||
<e p="bootstrap.css.map" t="Include" />
|
||||
<e p="bootstrap.min.css" t="Include" />
|
||||
<e p="bootstrap.min.css.map" t="Include" />
|
||||
</e>
|
||||
<e p="js" t="Include">
|
||||
<e p="bootstrap.bundle.js" t="Include" />
|
||||
<e p="bootstrap.bundle.js.map" t="Include" />
|
||||
<e p="bootstrap.bundle.min.js" t="Include" />
|
||||
<e p="bootstrap.bundle.min.js.map" t="Include" />
|
||||
<e p="bootstrap.js" t="Include" />
|
||||
<e p="bootstrap.js.map" t="Include" />
|
||||
<e p="bootstrap.min.js" t="Include" />
|
||||
<e p="bootstrap.min.js.map" t="Include" />
|
||||
</e>
|
||||
</e>
|
||||
<e p="LICENSE" t="Include" />
|
||||
</e>
|
||||
<e p="jquery" t="Include">
|
||||
<e p="dist" t="Include">
|
||||
<e p="jquery.js" t="Include" />
|
||||
<e p="jquery.min.js" t="Include" />
|
||||
<e p="jquery.min.map" t="Include" />
|
||||
</e>
|
||||
<e p="LICENSE.txt" t="Include" />
|
||||
</e>
|
||||
<e p="jquery-validation" t="Include">
|
||||
<e p="dist" t="Include">
|
||||
<e p="additional-methods.js" t="Include" />
|
||||
<e p="additional-methods.min.js" t="Include" />
|
||||
<e p="jquery.validate.js" t="Include" />
|
||||
<e p="jquery.validate.min.js" t="Include" />
|
||||
</e>
|
||||
<e p="LICENSE.md" t="Include" />
|
||||
</e>
|
||||
<e p="jquery-validation-unobtrusive" t="Include">
|
||||
<e p="jquery.validate.unobtrusive.js" t="Include" />
|
||||
<e p="jquery.validate.unobtrusive.min.js" t="Include" />
|
||||
<e p="LICENSE.txt" t="Include" />
|
||||
</e>
|
||||
</e>
|
||||
<e p="webfonts" t="Include">
|
||||
<e p="fa-brands-400.eot" t="Include" />
|
||||
<e p="fa-brands-400.svg" t="Include" />
|
||||
<e p="fa-brands-400.ttf" t="Include" />
|
||||
<e p="fa-brands-400.woff" t="Include" />
|
||||
<e p="fa-brands-400.woff2" t="Include" />
|
||||
<e p="fa-duotone-900.eot" t="Include" />
|
||||
<e p="fa-duotone-900.svg" t="Include" />
|
||||
<e p="fa-duotone-900.ttf" t="Include" />
|
||||
<e p="fa-duotone-900.woff" t="Include" />
|
||||
<e p="fa-duotone-900.woff2" t="Include" />
|
||||
<e p="fa-light-300.eot" t="Include" />
|
||||
<e p="fa-light-300.svg" t="Include" />
|
||||
<e p="fa-light-300.ttf" t="Include" />
|
||||
<e p="fa-light-300.woff" t="Include" />
|
||||
<e p="fa-light-300.woff2" t="Include" />
|
||||
<e p="fa-regular-400.eot" t="Include" />
|
||||
<e p="fa-regular-400.svg" t="Include" />
|
||||
<e p="fa-regular-400.ttf" t="Include" />
|
||||
<e p="fa-regular-400.woff" t="Include" />
|
||||
<e p="fa-regular-400.woff2" t="Include" />
|
||||
<e p="fa-solid-900.eot" t="Include" />
|
||||
<e p="fa-solid-900.svg" t="Include" />
|
||||
<e p="fa-solid-900.ttf" t="Include" />
|
||||
<e p="fa-solid-900.woff" t="Include" />
|
||||
<e p="fa-solid-900.woff2" t="Include" />
|
||||
</e>
|
||||
</e>
|
||||
</e>
|
||||
</component>
|
||||
</project>
|
4
.idea/.idea.c3stream/.idea/encodings.xml
Normal file
4
.idea/.idea.c3stream/.idea/encodings.xml
Normal file
|
@ -0,0 +1,4 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="Encoding" addBOMForNewFiles="with BOM under Windows, with no BOM otherwise" />
|
||||
</project>
|
8
.idea/.idea.c3stream/.idea/indexLayout.xml
Normal file
8
.idea/.idea.c3stream/.idea/indexLayout.xml
Normal file
|
@ -0,0 +1,8 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="ContentModelUserStore">
|
||||
<attachedFolders />
|
||||
<explicitIncludes />
|
||||
<explicitExcludes />
|
||||
</component>
|
||||
</project>
|
6
.idea/.idea.c3stream/.idea/jsLibraryMappings.xml
Normal file
6
.idea/.idea.c3stream/.idea/jsLibraryMappings.xml
Normal file
|
@ -0,0 +1,6 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="JavaScriptLibraryMappings">
|
||||
<file url="PROJECT" libraries="{all, c1a632a160}" />
|
||||
</component>
|
||||
</project>
|
8
.idea/.idea.c3stream/.idea/modules.xml
Normal file
8
.idea/.idea.c3stream/.idea/modules.xml
Normal file
|
@ -0,0 +1,8 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="ProjectModuleManager">
|
||||
<modules>
|
||||
<module fileurl="file://$PROJECT_DIR$/.idea/.idea.c3stream/riderModule.iml" filepath="$PROJECT_DIR$/.idea/.idea.c3stream/riderModule.iml" />
|
||||
</modules>
|
||||
</component>
|
||||
</project>
|
6
.idea/.idea.c3stream/.idea/projectSettingsUpdater.xml
Normal file
6
.idea/.idea.c3stream/.idea/projectSettingsUpdater.xml
Normal file
|
@ -0,0 +1,6 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="RiderProjectSettingsUpdater">
|
||||
<option name="vcsConfiguration" value="1" />
|
||||
</component>
|
||||
</project>
|
6
.idea/.idea.c3stream/.idea/vcs.xml
Normal file
6
.idea/.idea.c3stream/.idea/vcs.xml
Normal file
|
@ -0,0 +1,6 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="VcsDirectoryMappings">
|
||||
<mapping directory="$PROJECT_DIR$" vcs="Git" />
|
||||
</component>
|
||||
</project>
|
9
.idea/.idea.c3stream/riderModule.iml
Normal file
9
.idea/.idea.c3stream/riderModule.iml
Normal file
|
@ -0,0 +1,9 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<module type="RIDER_MODULE" version="4">
|
||||
<component name="NewModuleRootManager">
|
||||
<content url="file://$MODULE_DIR$/../.." />
|
||||
<orderEntry type="sourceFolder" forTests="false" />
|
||||
<orderEntry type="library" name="c1a632a160" level="application" />
|
||||
<orderEntry type="library" name="all" level="application" />
|
||||
</component>
|
||||
</module>
|
|
@ -1,35 +0,0 @@
|
|||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using c3stream.DataModels.Tables;
|
||||
using LinqToDB;
|
||||
using LinqToDB.Configuration;
|
||||
using LinqToDB.Data;
|
||||
|
||||
namespace c3stream.DataModels;
|
||||
|
||||
public class Database {
|
||||
public class ConnectionStringSettings : IConnectionStringSettings {
|
||||
public string ConnectionString { get; set; }
|
||||
public string Name { get; set; }
|
||||
public string ProviderName { get; set; }
|
||||
public bool IsGlobal => false;
|
||||
}
|
||||
|
||||
public class Settings : ILinqToDBSettings {
|
||||
public IEnumerable<IDataProviderSettings> DataProviders => Enumerable.Empty<IDataProviderSettings>();
|
||||
|
||||
public string DefaultConfiguration => "SQLite";
|
||||
public string DefaultDataProvider => "SQLite";
|
||||
|
||||
public IEnumerable<IConnectionStringSettings> ConnectionStrings {
|
||||
get { yield return new ConnectionStringSettings { Name = "db", ProviderName = "SQLite", ConnectionString = @"Data Source=data/c3stream.sqlite;" }; }
|
||||
}
|
||||
}
|
||||
|
||||
public class DbConn : DataConnection {
|
||||
public DbConn() : base("db") { }
|
||||
|
||||
public ITable<States> States => GetTable<States>();
|
||||
public ITable<DbInfo> DbInfo => GetTable<DbInfo>();
|
||||
}
|
||||
}
|
|
@ -1,9 +0,0 @@
|
|||
using LinqToDB.Mapping;
|
||||
|
||||
namespace c3stream.DataModels.Tables;
|
||||
|
||||
[Table(Name = "DbInfo")]
|
||||
public class DbInfo {
|
||||
[Column(Name = "ID"), PrimaryKey, Identity, NotNull] public int Id { get; set; }
|
||||
[Column(Name = "DbVer"), NotNull] public int DbVer { get; set; }
|
||||
}
|
|
@ -1,10 +0,0 @@
|
|||
using LinqToDB.Mapping;
|
||||
|
||||
namespace c3stream.DataModels.Tables;
|
||||
|
||||
[Table(Name = "States")]
|
||||
public class States {
|
||||
[Column(Name = "TalkId"), PrimaryKey, NotNull] public string TalkId { get; set; }
|
||||
[Column(Name = "UserId"), PrimaryKey, NotNull] public string UserId { get; set; }
|
||||
[Column(Name = "State"), NotNull] public string State { get; set; }
|
||||
}
|
5
LICENSE
5
LICENSE
|
@ -1,4 +1,4 @@
|
|||
Be Gay, Do Crimes License
|
||||
MIT License
|
||||
|
||||
Copyright (c) 2020 Laura Hausmann
|
||||
|
||||
|
@ -9,9 +9,6 @@ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|||
copies of the Software, and to permit persons to whom the Software is
|
||||
furnished to do so, subject to the following conditions:
|
||||
|
||||
Be Gay
|
||||
Do Crimes
|
||||
|
||||
The above copyright notice and this permission notice shall be included in all
|
||||
copies or substantial portions of the Software.
|
||||
|
||||
|
|
|
@ -1,90 +0,0 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using c3stream.DataModels;
|
||||
using c3stream.DataModels.Tables;
|
||||
using LinqToDB;
|
||||
using LinqToDB.Data;
|
||||
|
||||
namespace c3stream;
|
||||
|
||||
public static class Migrations {
|
||||
private const int DbVer = 0;
|
||||
|
||||
private static readonly List<Migration> _migrations = new();
|
||||
|
||||
public static void RunMigrations() {
|
||||
using var db = new Database.DbConn();
|
||||
var ccolor = Console.ForegroundColor;
|
||||
|
||||
if (!db.DataProvider.GetSchemaProvider().GetSchema(db).Tables.Any()) {
|
||||
Console.ForegroundColor = ConsoleColor.DarkCyan;
|
||||
Console.Write("Running migration: ");
|
||||
Console.ForegroundColor = ConsoleColor.Yellow;
|
||||
Console.WriteLine("Initialize Database");
|
||||
|
||||
db.CreateTable<States>();
|
||||
db.CreateTable<DbInfo>();
|
||||
db.InsertWithIdentity(new DbInfo { DbVer = DbVer });
|
||||
}
|
||||
else if (db.DataProvider.GetSchemaProvider().GetSchema(db).Tables.All(t => t.TableName != "DbInfo")) {
|
||||
db.CreateTable<DbInfo>();
|
||||
db.InsertWithIdentity(new DbInfo { DbVer = 0 });
|
||||
}
|
||||
|
||||
Console.ForegroundColor = ConsoleColor.Yellow;
|
||||
Console.WriteLine($"Database version: {db.DbInfo.ToList().First().DbVer}");
|
||||
|
||||
var migrationsToRun = _migrations.FindAll(p => p.IntroducedWithDbVer > db.DbInfo.First().DbVer);
|
||||
if (migrationsToRun.Count == 0) {
|
||||
Console.ForegroundColor = ConsoleColor.Green;
|
||||
Console.WriteLine("No migrations to run.");
|
||||
}
|
||||
else {
|
||||
new Migration(0, "BEGIN TRANSACTION").Run(db);
|
||||
try {
|
||||
migrationsToRun.ForEach(p => p.Run(db));
|
||||
}
|
||||
catch {
|
||||
Console.ForegroundColor = ConsoleColor.DarkRed;
|
||||
Console.WriteLine($"Migrating to database version {DbVer} failed.");
|
||||
new Migration(0, "ROLLBACK TRANSACTION").Run(db);
|
||||
Console.ForegroundColor = ConsoleColor.DarkYellow;
|
||||
Console.WriteLine("Rolled back migrations.");
|
||||
Environment.Exit(1);
|
||||
}
|
||||
|
||||
new Migration(0, "COMMIT TRANSACTION").Run(db);
|
||||
|
||||
var newdb = new Database.DbConn();
|
||||
var dbinfo = newdb.DbInfo.First();
|
||||
dbinfo.DbVer = DbVer;
|
||||
newdb.Update(dbinfo);
|
||||
|
||||
Console.ForegroundColor = ConsoleColor.DarkGreen;
|
||||
Console.WriteLine($"Database version is now: {DbVer}");
|
||||
Console.ForegroundColor = ConsoleColor.Green;
|
||||
Console.WriteLine("Finished running migrations.");
|
||||
}
|
||||
|
||||
Console.ForegroundColor = ccolor;
|
||||
}
|
||||
|
||||
private class Migration {
|
||||
private readonly string _sql;
|
||||
public readonly int IntroducedWithDbVer;
|
||||
|
||||
public Migration(int introducedWithDbVer, string sql) {
|
||||
IntroducedWithDbVer = introducedWithDbVer;
|
||||
_sql = sql;
|
||||
}
|
||||
|
||||
public void Run(DataConnection db) {
|
||||
Console.ForegroundColor = ConsoleColor.DarkCyan;
|
||||
Console.Write("Running migration: ");
|
||||
Console.ForegroundColor = ConsoleColor.Yellow;
|
||||
Console.WriteLine(_sql);
|
||||
db.Execute(_sql);
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,114 +1,128 @@
|
|||
@page
|
||||
@using global::c3stream.DataModels
|
||||
@model ConferenceModel
|
||||
@using System.Net
|
||||
@using static ConferenceModel
|
||||
@{
|
||||
if (c3stream.Conferences.All(c => c.Acronym != Request.Query["c"])) {
|
||||
Response.Redirect("/");
|
||||
return;
|
||||
}
|
||||
int tagFormat;
|
||||
switch (Request.Query["c"]) {
|
||||
case "36c3":
|
||||
tagFormat = 2;
|
||||
break;
|
||||
case "35c3":
|
||||
case "34c3":
|
||||
tagFormat = 1;
|
||||
break;
|
||||
case "33c3":
|
||||
tagFormat = 0;
|
||||
break;
|
||||
default:
|
||||
Response.Redirect("/");
|
||||
return;
|
||||
}
|
||||
|
||||
var cookie = c3stream.UpdateCookie(Request, Response, $"/Conference?c={Request.Query["c"]}");
|
||||
ViewData["Title"] = Request.Query["c"];
|
||||
var conference = c3stream.Conferences.First(c => c.Acronym == Request.Query["c"]);
|
||||
if (conference.Ongoing) {
|
||||
c3stream.UpdateConference(conference);
|
||||
}
|
||||
await using var db = new Database.DbConn();
|
||||
var states = db.States.ToList();
|
||||
c3stream.UpdateCookie(Request, Response, $"/Conference?c={Request.Query["c"]}&");
|
||||
|
||||
ViewData["Title"] = Request.Query["c"];
|
||||
var wc = new WebClient();
|
||||
var jsonpath = System.IO.Path.Combine(c3stream.DataPath, Request.Query["c"] + ".json");
|
||||
var json = System.IO.File.Exists(jsonpath) ? System.IO.File.ReadAllText(jsonpath) : wc.DownloadString($"https://api.media.ccc.de/public/conferences/{Request.Query["c"]}");
|
||||
var conference = Conference.FromJson(json);
|
||||
wc.Dispose();
|
||||
}
|
||||
|
||||
<table class="table">
|
||||
<thead>
|
||||
<tr>
|
||||
<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" ? conference.Talks.OrderByDescending(p => p.ReleaseDate) : conference.Talks.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 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],
|
||||
7 => talk.Tags[3],
|
||||
_ => "<unknown tag format>"
|
||||
};
|
||||
<tr>
|
||||
<td>@Html.Raw(eventName)</td>
|
||||
<td class="text-nowrap">@(Request.Query["orderby"] == "published" ? talk.ReleaseDate?.Date.ToString("yyyy-MM-dd") : talk.Date?.Date.ToString("yyyy-MM-dd"))</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" 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>
|
||||
<thead>
|
||||
<tr>
|
||||
<th scope="col">Event</th>
|
||||
<th scope="col">
|
||||
@Html.Raw(Request.Query["orderby"] == "published" ? $"<a href=\"/Conference?c={Request.Query["c"]}&bookmark={Request.Query["bookmark"]}\">Published" : $"<a href=\"/Conference?c={Request.Query["c"]}&bookmark={Request.Query["bookmark"]}&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" ? conference.Events.OrderByDescending(p => p.ReleaseDate) : conference.Events.OrderBy(p => p.Date)) {
|
||||
TalkMetadata metadata;
|
||||
if (EventMetadata.Any(p => p.Guid == talk.Guid?.ToString())) {
|
||||
metadata = EventMetadata.First(p => p.Guid == talk.Guid?.ToString());
|
||||
}
|
||||
else {
|
||||
metadata = GetEvent(talk.Guid?.ToString());
|
||||
EventMetadata.Add(metadata);
|
||||
}
|
||||
var isWatched = EventMetadata.Any(p => p.Guid == talk.Guid?.ToString() && p.State.FirstOrDefault(q => q.Guid == Request.Cookies["bookmark"])?.State == "watched");
|
||||
var isMarked = EventMetadata.Any(p => p.Guid == talk.Guid?.ToString() && p.State.FirstOrDefault(q => q.Guid == Request.Cookies["bookmark"])?.State == "marked");
|
||||
var file = metadata.Talk.Recordings.FirstOrDefault(p => System.IO.File.Exists(System.IO.Path.Combine(c3stream.CachePath, conference.Acronym, p.Filename)));
|
||||
var eventName = tagFormat == 0 ? conference.Acronym : talk.Tags[0].Replace("-", "-<br/>");
|
||||
var category = tagFormat switch {
|
||||
0 => talk.Tags[0],
|
||||
1 => talk.Tags[2],
|
||||
2 => talk.Tags[3],
|
||||
_ => ""
|
||||
};
|
||||
<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: limegreen">@talk.Title</td>
|
||||
}
|
||||
else if (isMarked) {
|
||||
<td style="color: orangered">@talk.Title</td>
|
||||
}
|
||||
else {
|
||||
<td>@talk.Title</td>
|
||||
}
|
||||
<td>@talk.Persons.Aggregate((s, s1) => $"{s}, {s1}")</td>
|
||||
<td>@talk.OriginalLanguage</td>
|
||||
<td>
|
||||
<div class="btn-group w-100" role="group">
|
||||
<a href="@talk.FrontendLink.AbsoluteUri" target="_blank" type="button" class="btn btn-primary w-100" data-toggle="tooltip" data-placement="top" title="Play">
|
||||
<i class="fas fa-play-circle"></i>
|
||||
</a>
|
||||
@if (file != null) {
|
||||
<a href="@(c3stream.CacheUrl + $"{conference.Acronym}/{file.Filename}")" target="_blank" type="button" class="btn btn-primary w-100" data-toggle="tooltip" data-placement="top" title="Mirror">
|
||||
<i class="fas fa-cloud-download"></i>
|
||||
</a>
|
||||
}
|
||||
else {
|
||||
<a href="/" target="_blank" type="button" class="btn btn-primary disabled">
|
||||
<i class="fas fa-cloud-download"></i>
|
||||
</a>
|
||||
}
|
||||
<a href="/Info?guid=@talk.Guid?.ToString()&bookmark=@Request.Query["bookmark"]" target="_blank" type="button" class="btn btn-primary 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 w-100" data-toggle="tooltip" data-placement="top" title="Mark unwatched">
|
||||
<i class="fas fa-times"></i>
|
||||
</button>
|
||||
<button class="btn btn-primary disabled">
|
||||
<i class="fas fa-clock"></i>
|
||||
</button>
|
||||
}
|
||||
else if (isMarked) {
|
||||
<button onclick="SetState('@talk.Guid', 'watched')" class="btn btn-primary 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 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 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 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>
|
|
@ -1,37 +1,81 @@
|
|||
using System.Linq;
|
||||
using c3stream.DataModels;
|
||||
using c3stream.DataModels.Tables;
|
||||
using LinqToDB;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Net;
|
||||
using Microsoft.AspNetCore.Mvc.RazorPages;
|
||||
using Microsoft.Extensions.Logging;
|
||||
using Newtonsoft.Json;
|
||||
|
||||
namespace c3stream.Pages;
|
||||
namespace c3stream.Pages {
|
||||
public class ConferenceModel : PageModel {
|
||||
public static List<TalkMetadata> EventMetadata = new List<TalkMetadata>();
|
||||
private readonly ILogger<ConferenceModel> _logger;
|
||||
|
||||
public class ConferenceModel : PageModel {
|
||||
private readonly ILogger<ConferenceModel> _logger;
|
||||
public ConferenceModel(ILogger<ConferenceModel> logger) => _logger = logger;
|
||||
|
||||
public ConferenceModel(ILogger<ConferenceModel> logger) => _logger = logger;
|
||||
public void OnGet() {
|
||||
var guid = Request.Query["guid"];
|
||||
var state = Request.Query["state"];
|
||||
if (string.IsNullOrWhiteSpace(guid) || string.IsNullOrWhiteSpace(state) || !Request.Cookies.ContainsKey("bookmark"))
|
||||
return;
|
||||
|
||||
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);
|
||||
lock (c3stream.Lock) {
|
||||
ReadEventMetadata();
|
||||
var existing = EventMetadata.FirstOrDefault(p => p.Guid == guid)?.State.FirstOrDefault(p => p.Guid == Request.Cookies["bookmark"]);
|
||||
if (existing != null)
|
||||
existing.State = state;
|
||||
else
|
||||
EventMetadata.FirstOrDefault(p => p.Guid == guid)?.State.Add(new UserState(Request.Cookies["bookmark"], state));
|
||||
WriteEventMetadata();
|
||||
Response.Redirect("/");
|
||||
}
|
||||
else {
|
||||
existing.State = state;
|
||||
db.Update(existing);
|
||||
}
|
||||
else
|
||||
db.Insert(new States { State = state, TalkId = guid, UserId = userid });
|
||||
}
|
||||
|
||||
Response.Redirect("/");
|
||||
public static TalkMetadata GetEvent(string guid) {
|
||||
TalkMetadata metadata;
|
||||
lock (c3stream.Lock) {
|
||||
ReadEventMetadata();
|
||||
using (var wc = new WebClient()) {
|
||||
var json = wc.DownloadString($"https://api.media.ccc.de/public/events/{guid}");
|
||||
var talk = Talk.FromJson(json);
|
||||
metadata = new TalkMetadata(guid, talk);
|
||||
EventMetadata.Add(metadata);
|
||||
}
|
||||
|
||||
WriteEventMetadata();
|
||||
}
|
||||
|
||||
return metadata;
|
||||
}
|
||||
|
||||
public static void ReadEventMetadata() {
|
||||
lock (c3stream.Lock)
|
||||
EventMetadata = JsonConvert.DeserializeObject<List<TalkMetadata>>(System.IO.File.ReadAllText(c3stream.DbPath));
|
||||
}
|
||||
|
||||
public static void WriteEventMetadata() {
|
||||
System.IO.File.WriteAllText(c3stream.DbPath, JsonConvert.SerializeObject(EventMetadata));
|
||||
}
|
||||
|
||||
public class TalkMetadata {
|
||||
public readonly string Guid;
|
||||
public List<UserState> State;
|
||||
public Talk Talk;
|
||||
|
||||
public TalkMetadata(string guid, Talk talk) {
|
||||
Guid = guid;
|
||||
State = new List<UserState>();
|
||||
Talk = talk;
|
||||
}
|
||||
}
|
||||
|
||||
public class UserState {
|
||||
public string Guid;
|
||||
public string State;
|
||||
|
||||
public UserState(string guid, string state = "unwatched") {
|
||||
Guid = guid;
|
||||
State = state;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,25 +1,25 @@
|
|||
@page
|
||||
@model ErrorModel
|
||||
@{
|
||||
ViewData["Title"] = "Error";
|
||||
ViewData["Title"] = "Error";
|
||||
}
|
||||
|
||||
<h1 class="text-danger">Error.</h1>
|
||||
<h2 class="text-danger">An error occurred while processing your request.</h2>
|
||||
|
||||
@if (Model.ShowRequestId) {
|
||||
<p>
|
||||
<strong>Request ID:</strong> <code>@Model.RequestId</code>
|
||||
</p>
|
||||
<p>
|
||||
<strong>Request ID:</strong> <code>@Model.RequestId</code>
|
||||
</p>
|
||||
}
|
||||
|
||||
<h3>Development Mode</h3>
|
||||
<p>
|
||||
Swapping to the <strong>Development</strong> environment displays detailed information about the error that occurred.
|
||||
Swapping to the <strong>Development</strong> environment displays detailed information about the error that occurred.
|
||||
</p>
|
||||
<p>
|
||||
<strong>The Development environment shouldn't be enabled for deployed applications.</strong>
|
||||
It can result in displaying sensitive information from exceptions to end users.
|
||||
For local debugging, enable the <strong>Development</strong> environment by setting the <strong>ASPNETCORE_ENVIRONMENT</strong> environment variable to <strong>Development</strong>
|
||||
and restarting the app.
|
||||
<strong>The Development environment shouldn't be enabled for deployed applications.</strong>
|
||||
It can result in displaying sensitive information from exceptions to end users.
|
||||
For local debugging, enable the <strong>Development</strong> environment by setting the <strong>ASPNETCORE_ENVIRONMENT</strong> environment variable to <strong>Development</strong>
|
||||
and restarting the app.
|
||||
</p>
|
|
@ -3,19 +3,19 @@ using Microsoft.AspNetCore.Mvc;
|
|||
using Microsoft.AspNetCore.Mvc.RazorPages;
|
||||
using Microsoft.Extensions.Logging;
|
||||
|
||||
namespace c3stream.Pages;
|
||||
namespace c3stream.Pages {
|
||||
[ResponseCache(Duration = 0, Location = ResponseCacheLocation.None, NoStore = true)]
|
||||
public class ErrorModel : PageModel {
|
||||
private readonly ILogger<ErrorModel> _logger;
|
||||
|
||||
[ResponseCache(Duration = 0, Location = ResponseCacheLocation.None, NoStore = true)]
|
||||
public class ErrorModel : PageModel {
|
||||
private readonly ILogger<ErrorModel> _logger;
|
||||
public ErrorModel(ILogger<ErrorModel> logger) => _logger = logger;
|
||||
|
||||
public ErrorModel(ILogger<ErrorModel> logger) => _logger = logger;
|
||||
public string RequestId { get; set; }
|
||||
|
||||
public string RequestId { get; set; }
|
||||
public bool ShowRequestId => !string.IsNullOrEmpty(RequestId);
|
||||
|
||||
public bool ShowRequestId => !string.IsNullOrEmpty(RequestId);
|
||||
|
||||
public void OnGet() {
|
||||
RequestId = Activity.Current?.Id ?? HttpContext.TraceIdentifier;
|
||||
public void OnGet() {
|
||||
RequestId = Activity.Current?.Id ?? HttpContext.TraceIdentifier;
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,22 +1,18 @@
|
|||
@page
|
||||
@using global::c3stream.DataModels
|
||||
@model IndexModel
|
||||
@{
|
||||
ViewData["Title"] = "Home";
|
||||
var cookie = c3stream.UpdateCookie(Request, Response, "/");
|
||||
var marked = new Database.DbConn().States.Any(p => p.UserId == cookie && p.State == "marked");
|
||||
ViewData["Title"] = "Home";
|
||||
c3stream.UpdateCookie(Request, Response, $"/?");
|
||||
}
|
||||
|
||||
<div style="text-align: center">
|
||||
<h1>Welcome to c3stream!</h1>
|
||||
Your bookmark link:<br/>
|
||||
<code onclick="copyToClipboard(this)">https://@Request.Host.Value?bookmark=@cookie</code><br/><br/>
|
||||
<div class="btn-group">
|
||||
@if (marked) {
|
||||
<a role="button" class="btn btn-primary" href="/Watchlist">watchlist</a>
|
||||
}
|
||||
@foreach (var conf in c3stream.Conferences) {
|
||||
<a role="button" class="btn btn-primary" href="/Conference?c=@conf.Acronym">@conf.Acronym</a>
|
||||
}
|
||||
</div>
|
||||
<h1>Welcome to c3stream!</h1>
|
||||
Your bookmark link:<br/>
|
||||
<code onclick="copyToClipboard(this)">https://@Request.Host.Value?bookmark=@Request.Cookies["bookmark"]</code><br/><br/>
|
||||
<div class="btn-group">
|
||||
<a type="button" class="btn btn-primary" href="/Conference?c=36c3&bookmark=@Request.Cookies["bookmark"]">36c3</a>
|
||||
<a type="button" class="btn btn-primary" href="/Conference?c=35c3&bookmark=@Request.Cookies["bookmark"]">35c3</a>
|
||||
<a type="button" class="btn btn-primary" href="/Conference?c=34c3&bookmark=@Request.Cookies["bookmark"]">34c3</a>
|
||||
<a type="button" class="btn btn-primary" href="/Conference?c=33c3&bookmark=@Request.Cookies["bookmark"]">33c3</a>
|
||||
</div>
|
||||
</div>
|
|
@ -1,12 +1,12 @@
|
|||
using Microsoft.AspNetCore.Mvc.RazorPages;
|
||||
using Microsoft.Extensions.Logging;
|
||||
|
||||
namespace c3stream.Pages;
|
||||
namespace c3stream.Pages {
|
||||
public class IndexModel : PageModel {
|
||||
private readonly ILogger<IndexModel> _logger;
|
||||
|
||||
public class IndexModel : PageModel {
|
||||
private readonly ILogger<IndexModel> _logger;
|
||||
public IndexModel(ILogger<IndexModel> logger) => _logger = logger;
|
||||
|
||||
public IndexModel(ILogger<IndexModel> logger) => _logger = logger;
|
||||
|
||||
public void OnGet() { }
|
||||
public void OnGet() { }
|
||||
}
|
||||
}
|
|
@ -1,122 +1,112 @@
|
|||
@page
|
||||
@using global::c3stream.DataModels
|
||||
@using System.Net.Http
|
||||
@using System.Net
|
||||
@using System.IO
|
||||
@model InfoModel
|
||||
@{
|
||||
ViewData["Title"] = "Info";
|
||||
ViewData["Title"] = "Info";
|
||||
}
|
||||
@{
|
||||
if (string.IsNullOrWhiteSpace(Request.Query["guid"]) || c3stream.GetEventByGuid(Request.Query["guid"]) == null) {
|
||||
Response.Redirect("/");
|
||||
return;
|
||||
}
|
||||
if (string.IsNullOrWhiteSpace(Request.Query["guid"])) {
|
||||
Response.Redirect("/");
|
||||
return;
|
||||
}
|
||||
|
||||
var cookie = c3stream.UpdateCookie(Request, Response, $"/Info?guid={Request.Query["guid"]}");
|
||||
c3stream.UpdateCookie(Request, Response, $"/Info?guid={Request.Query["guid"]}&");
|
||||
|
||||
await using var db = new Database.DbConn();
|
||||
ConferenceModel.ReadEventMetadata();
|
||||
var talk = ConferenceModel.EventMetadata.FirstOrDefault(p => p.Guid == Request.Query["guid"]);
|
||||
if (talk == null) {
|
||||
Response.Redirect("/");
|
||||
return;
|
||||
}
|
||||
|
||||
var talk = c3stream.GetEventByGuid(Request.Query["guid"]);
|
||||
var state = db.States.FirstOrDefault(p => p.TalkId == Request.Query["guid"].ToString() && p.UserId == cookie)?.State;
|
||||
if (talk == null) {
|
||||
Response.Redirect("/");
|
||||
return;
|
||||
}
|
||||
if (state == null) {
|
||||
state = "unwatched";
|
||||
}
|
||||
var title = talk.Talk.Title;
|
||||
var speakers = talk.Talk.Persons.Aggregate((s, s1) => $"{s}, {s1}");
|
||||
var description = talk.Talk.Description;
|
||||
if (string.IsNullOrEmpty(description)) {
|
||||
description = "<missing description>";
|
||||
}
|
||||
|
||||
var title = talk.Title;
|
||||
var speakers = talk.Persons.Any() ? talk.Persons.Aggregate((s, s1) => $"{s}, {s1}") : "<no speakers>";
|
||||
var description = talk.Description;
|
||||
if (string.IsNullOrEmpty(description)) {
|
||||
description = "<missing description>";
|
||||
}
|
||||
var isWatched = talk.State.FirstOrDefault(q => q.Guid == Request.Cookies["bookmark"])?.State == "watched";
|
||||
var isMarked = talk.State.FirstOrDefault(q => q.Guid == Request.Cookies["bookmark"])?.State == "marked";
|
||||
var file = talk.Talk.Recordings.FirstOrDefault(p => System.IO.File.Exists(System.IO.Path.Combine(c3stream.CachePath, talk.Talk.ConferenceUrl.AbsoluteUri.Split("/").Last(), p.Filename)));
|
||||
var eventName = talk.Talk.Tags[0].Replace("-", "-<br/>");
|
||||
|
||||
var isWatched = state == "watched";
|
||||
var isMarked = state == "marked";
|
||||
var file = $"{talk.Slug}.mp4";
|
||||
var conference = c3stream.GetConferenceByEventGuid(talk.Guid);
|
||||
int tagFormat;
|
||||
switch (talk.Talk.ConferenceUrl.AbsoluteUri.Split("/").Last()) {
|
||||
case "36c3":
|
||||
tagFormat = 2;
|
||||
break;
|
||||
case "35c3":
|
||||
case "34c3":
|
||||
tagFormat = 1;
|
||||
break;
|
||||
case "33c3":
|
||||
tagFormat = 0;
|
||||
break;
|
||||
default:
|
||||
Response.Redirect("/");
|
||||
return;
|
||||
}
|
||||
|
||||
var eventName = talk.Tags.Count <= 1 ? conference.Acronym : talk.Tags[0];
|
||||
var logoPath = System.IO.Path.Combine(c3stream.LogoPath, conference.Acronym + ".png");
|
||||
|
||||
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],
|
||||
7 => talk.Tags[3],
|
||||
_ => "<unknown tag format>"
|
||||
};
|
||||
var category = tagFormat switch {
|
||||
0 => talk.Talk.Tags[0],
|
||||
1 => talk.Talk.Tags[2],
|
||||
2 => talk.Talk.Tags[3],
|
||||
_ => ""
|
||||
};
|
||||
}
|
||||
|
||||
@if (!System.IO.File.Exists(logoPath)) {
|
||||
using var httpClient = new HttpClient();
|
||||
await using var stream = httpClient.GetStreamAsync(conference.LogoUri).Result;
|
||||
await using var fileStream = new FileStream(logoPath, FileMode.CreateNew);
|
||||
await stream.CopyToAsync(fileStream);
|
||||
}
|
||||
|
||||
<img src="@(c3stream.LogoUrl + $"{conference.Acronym}.png")" alt="Conference logo" style="max-height: 110px; float: right;"/>
|
||||
|
||||
@if (isWatched) {
|
||||
<h3 style="color: #95cb7a">@title - <i>@speakers</i></h3>
|
||||
<h3 style="color: limegreen">@title - <i>@speakers</i></h3>
|
||||
}
|
||||
else if (isMarked) {
|
||||
<h3 style="color: #da7d4f">@title - <i>@speakers</i></h3>
|
||||
<h3 style="color: orangered">@title - <i>@speakers</i></h3>
|
||||
}
|
||||
else {
|
||||
<h3>@title - <i>@speakers</i></h3>
|
||||
<h3>@title - <i>@speakers</i></h3>
|
||||
}
|
||||
|
||||
<h5>@eventName - @category - @talk.Date?.Date.ToShortDateString()</h5>
|
||||
<div class="btn-group" role="group" style="margin-bottom: 10px">
|
||||
<a href="@talk.FrontendLink.AbsoluteUri" role="button" class="btn btn-primary btn-c3saction w-100" data-toggle="tooltip" data-placement="right" 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="right" 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>
|
||||
}
|
||||
@if (isWatched) {
|
||||
<button onclick="SetState('@talk.Guid', 'unwatched')" class="btn btn-primary btn-c3saction w-100" data-toggle="tooltip" data-placement="left" 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="left" 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="left" 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="left" 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="left" title="Add to watch later">
|
||||
<i class="fas fa-clock"></i>
|
||||
</button>
|
||||
}
|
||||
<a href="@talk.Talk.FrontendLink.AbsoluteUri" target="_blank" type="button" class="btn btn-primary w-100" data-toggle="tooltip" data-placement="right" title="Play">
|
||||
<i class="fas fa-play-circle"></i>
|
||||
</a>
|
||||
@if (file != null) {
|
||||
<a href="@(c3stream.CacheUrl + $"{talk.Talk.ConferenceUrl.AbsoluteUri.Split("/").Last()}/{file.Filename}")" target="_blank" type="button" class="btn btn-primary w-100" data-toggle="tooltip" data-placement="right" title="Mirror">
|
||||
<i class="fas fa-cloud-download"></i>
|
||||
</a>
|
||||
}
|
||||
else {
|
||||
<a href="/" target="_blank" type="button" class="btn btn-primary disabled">
|
||||
<i class="fas fa-cloud-download"></i>
|
||||
</a>
|
||||
}
|
||||
@if (isWatched) {
|
||||
<button onclick="SetState('@talk.Guid', 'unwatched')" class="btn btn-primary w-100" data-toggle="tooltip" data-placement="left" title="Mark unwatched">
|
||||
<i class="fas fa-times"></i>
|
||||
</button>
|
||||
<button class="btn btn-primary disabled">
|
||||
<i class="fas fa-clock"></i>
|
||||
</button>
|
||||
}
|
||||
else if (isMarked) {
|
||||
<button onclick="SetState('@talk.Guid', 'watched')" class="btn btn-primary w-100" data-toggle="tooltip" data-placement="left" title="Mark watched">
|
||||
<i class="fas fa-check"></i>
|
||||
</button>
|
||||
<button onclick="SetState('@talk.Guid', 'unwatched')" class="btn btn-primary w-100" data-toggle="tooltip" data-placement="left" title="Remove from watch later">
|
||||
<i class="fas fa-undo-alt"></i>
|
||||
</button>
|
||||
}
|
||||
else {
|
||||
<button onclick="SetState('@talk.Guid', 'watched')" class="btn btn-primary w-100" data-toggle="tooltip" data-placement="left" title="Mark watched">
|
||||
<i class="fas fa-check"></i>
|
||||
</button>
|
||||
<button onclick="SetState('@talk.Guid', 'marked')" class="btn btn-primary w-100" data-toggle="tooltip" data-placement="left" title="Add to watch later">
|
||||
<i class="fas fa-clock"></i>
|
||||
</button>
|
||||
}
|
||||
</div>
|
||||
|
||||
<p style="text-align: justify">
|
||||
@Html.Raw(description.Replace("\n", "<br/>").Replace("<p>", "").Replace("</p>", ""))
|
||||
@Html.Raw(description.Replace("\n", "<br/>").Replace("<p>", "").Replace("</p>", ""))
|
||||
</p>
|
||||
|
||||
Share this talk:<br/>
|
||||
|
|
|
@ -1,12 +1,12 @@
|
|||
using Microsoft.AspNetCore.Mvc.RazorPages;
|
||||
using Microsoft.Extensions.Logging;
|
||||
|
||||
namespace c3stream.Pages;
|
||||
namespace c3stream.Pages {
|
||||
public class InfoModel : PageModel {
|
||||
private readonly ILogger<InfoModel> _logger;
|
||||
|
||||
public class InfoModel : PageModel {
|
||||
private readonly ILogger<InfoModel> _logger;
|
||||
public InfoModel(ILogger<InfoModel> logger) => _logger = logger;
|
||||
|
||||
public InfoModel(ILogger<InfoModel> logger) => _logger = logger;
|
||||
|
||||
public void OnGet() { }
|
||||
public void OnGet() { }
|
||||
}
|
||||
}
|
|
@ -1,28 +1,11 @@
|
|||
@page
|
||||
@model PrivacyModel
|
||||
@{
|
||||
ViewData["Title"] = "Privacy";
|
||||
ViewData["Title"] = "Privacy";
|
||||
}
|
||||
<h3>Privacy Policy</h3>
|
||||
<h3>Privacy</h3>
|
||||
<p style="text-align: justify">
|
||||
Last update: 2022-12-15 <br/><br/>
|
||||
All data saved about you on this website is the watched status of talks you marked in association with your randomly generated bookmark UUID. <br/>
|
||||
No unnecessary logs are kept, no trackers used. <br/>
|
||||
</p>
|
||||
|
||||
<p>Web server access logs are enabled for statistical purposes only. The following data is collected and kept in anonymized form for a maximum of 28 days:</p>
|
||||
<ul>
|
||||
<li>Request date & time</li>
|
||||
<li>Anonymized IP address (first 48 bits of the IPv6 address, IPv4 addresses go through a NAT46 gateway and are not logged)</li>
|
||||
<li>Request type, URL and protocol</li>
|
||||
<li>HTTP response code</li>
|
||||
<li>Response body size</li>
|
||||
<li>Referer</li>
|
||||
<li>User agent</li>
|
||||
<li>Response time</li>
|
||||
</ul>
|
||||
|
||||
<p style="text-align: justify">
|
||||
Keep in mind that you are forwarded to media.ccc.de when you watch a talk (except when using our mirror). <br/>
|
||||
You should therefore reference their privacy policy as well. <br/>
|
||||
All data saved about you on this website is the watched status of talks you marked in association with your randomly generated bookmark UUID.
|
||||
No logs are kept, no trackers used. Keep in mind that you are forwarded to media.ccc.de if you choose to open a talk, and have checked their respective privacy policy as well. <br/>
|
||||
Have fun!
|
||||
</p>
|
|
@ -1,12 +1,12 @@
|
|||
using Microsoft.AspNetCore.Mvc.RazorPages;
|
||||
using Microsoft.Extensions.Logging;
|
||||
|
||||
namespace c3stream.Pages;
|
||||
namespace c3stream.Pages {
|
||||
public class PrivacyModel : PageModel {
|
||||
private readonly ILogger<PrivacyModel> _logger;
|
||||
|
||||
public class PrivacyModel : PageModel {
|
||||
private readonly ILogger<PrivacyModel> _logger;
|
||||
public PrivacyModel(ILogger<PrivacyModel> logger) => _logger = logger;
|
||||
|
||||
public PrivacyModel(ILogger<PrivacyModel> logger) => _logger = logger;
|
||||
|
||||
public void OnGet() { }
|
||||
public void OnGet() { }
|
||||
}
|
||||
}
|
|
@ -1,38 +1,36 @@
|
|||
<!DOCTYPE html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="utf-8"/>
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0"/>
|
||||
<title>@ViewData["Title"] - c3stream</title>
|
||||
<link rel="stylesheet" href="~/lib/bootstrap/dist/css/bootstrap.min.css"/>
|
||||
<link rel="stylesheet" href="~/css/fa.css" crossorigin="anonymous">
|
||||
<link rel="stylesheet" href="~/css/site.css"/>
|
||||
<meta charset="utf-8"/>
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0"/>
|
||||
<title>@ViewData["Title"] - c3stream</title>
|
||||
<link rel="stylesheet" href="~/lib/bootstrap/dist/css/bootstrap.min.css"/>
|
||||
<link rel="stylesheet" href="~/css/fa.css" crossorigin="anonymous">
|
||||
<link rel="stylesheet" href="~/css/site.css"/>
|
||||
</head>
|
||||
<body>
|
||||
<header>
|
||||
<nav class="navbar navbar-expand-sm navbar-toggleable-sm navbar-light bg-white border-bottom box-shadow mb-3">
|
||||
<div class="container-fluid" style="width: 90%">
|
||||
<a class="navbar-brand" asp-area="" asp-page="/Index">c3stream <small style="font-size: x-small">v5</small></a>
|
||||
<button class="navbar-toggler" role="button" data-toggle="collapse" data-target=".navbar-collapse" aria-controls="navbarSupportedContent"
|
||||
aria-expanded="false" aria-label="Toggle navigation">
|
||||
<span class="navbar-toggler-icon"></span>
|
||||
</button>
|
||||
</div>
|
||||
</nav>
|
||||
<nav class="navbar navbar-expand-sm navbar-toggleable-sm navbar-light bg-white border-bottom box-shadow mb-3">
|
||||
<div class="container-fluid" style="width: 90%">
|
||||
<a class="navbar-brand" asp-area="" asp-page="/Index">c3stream</a>
|
||||
<button class="navbar-toggler" type="button" data-toggle="collapse" data-target=".navbar-collapse" aria-controls="navbarSupportedContent"
|
||||
aria-expanded="false" aria-label="Toggle navigation">
|
||||
<span class="navbar-toggler-icon"></span>
|
||||
</button>
|
||||
</div>
|
||||
</nav>
|
||||
</header>
|
||||
<div class="container-fluid" style="width: 90%">
|
||||
<main role="main" class="pb-3">
|
||||
@RenderBody()
|
||||
</main>
|
||||
<main role="main" class="pb-3">
|
||||
@RenderBody()
|
||||
</main>
|
||||
</div>
|
||||
|
||||
<footer class="border-top footer">
|
||||
<div class="container-fluid" style="width: 90%; text-align: center">
|
||||
<a href="/Privacy">Privacy</a> -
|
||||
<a href="mailto:c3stream-contact@zotan.email">Contact</a> -
|
||||
<a href="https://git.zotan.services/zotan/c3stream/">Source Code</a> -
|
||||
c3stream is not affiliated with media.ccc.de in any way. Mirrored video files display their license in the video, no rights reserved.
|
||||
</div>
|
||||
<footer class="border-top footer text-muted">
|
||||
<div class="container-fluid" style="width: 90%; text-align: center">
|
||||
© 2020 - c3stream - <a href="/Privacy">Privacy</a> -
|
||||
c3stream is not associated with media.ccc.de in any way and is just providing an easy way to store watched talks.
|
||||
</div>
|
||||
</footer>
|
||||
|
||||
<script src="~/lib/jquery/dist/jquery.min.js"></script>
|
||||
|
|
|
@ -1,110 +0,0 @@
|
|||
@page
|
||||
@using global::c3stream.DataModels
|
||||
@model 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=\"/Watchlist\">Published" : "<a href=\"/Watchlist?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],
|
||||
7 => talk.Tags[3],
|
||||
_ => "<unknown tag format>"
|
||||
};
|
||||
<tr>
|
||||
<td>@Html.Raw(conference.Acronym)</td>
|
||||
<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>
|
|
@ -1,37 +0,0 @@
|
|||
using System.Linq;
|
||||
using c3stream.DataModels;
|
||||
using c3stream.DataModels.Tables;
|
||||
using LinqToDB;
|
||||
using Microsoft.AspNetCore.Mvc.RazorPages;
|
||||
using Microsoft.Extensions.Logging;
|
||||
|
||||
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("/");
|
||||
}
|
||||
}
|
|
@ -1,3 +1,3 @@
|
|||
@{
|
||||
Layout = "_Layout";
|
||||
Layout = "_Layout";
|
||||
}
|
|
@ -1,4 +1,12 @@
|
|||
{
|
||||
"iisSettings": {
|
||||
"windowsAuthentication": false,
|
||||
"anonymousAuthentication": true,
|
||||
"iisExpress": {
|
||||
"applicationUrl": "http://localhost:37898",
|
||||
"sslPort": 44314
|
||||
}
|
||||
},
|
||||
"profiles": {
|
||||
"c3stream": {
|
||||
"commandName": "Project",
|
||||
|
|
|
@ -1,2 +1 @@
|
|||
c3stream is a small proxy site meant for saving watched status & watch-later-lists for media.ccc.de talks. Test in
|
||||
production at https://c3stream.de
|
||||
c3stream is a small proxy site meant for saving watched status & watch-later-lists for media.ccc.de talks. Test in production at https://c3stream.de
|
40
Startup.cs
40
Startup.cs
|
@ -4,32 +4,32 @@ using Microsoft.Extensions.Configuration;
|
|||
using Microsoft.Extensions.DependencyInjection;
|
||||
using Microsoft.Extensions.Hosting;
|
||||
|
||||
namespace c3stream;
|
||||
namespace c3stream {
|
||||
public class Startup {
|
||||
public Startup(IConfiguration configuration) => Configuration = configuration;
|
||||
|
||||
public class Startup {
|
||||
public Startup(IConfiguration configuration) => Configuration = configuration;
|
||||
public IConfiguration Configuration { get; }
|
||||
|
||||
public IConfiguration Configuration { get; }
|
||||
// This method gets called by the runtime. Use this method to add services to the container.
|
||||
public void ConfigureServices(IServiceCollection services) {
|
||||
services.AddRazorPages();
|
||||
}
|
||||
|
||||
// This method gets called by the runtime. Use this method to add services to the container.
|
||||
public void ConfigureServices(IServiceCollection services) {
|
||||
services.AddRazorPages();
|
||||
}
|
||||
// This method gets called by the runtime. Use this method to configure the HTTP request pipeline.
|
||||
public void Configure(IApplicationBuilder app, IWebHostEnvironment env) {
|
||||
if (env.IsDevelopment())
|
||||
app.UseDeveloperExceptionPage();
|
||||
else
|
||||
app.UseExceptionHandler("/Error");
|
||||
// The default HSTS value is 30 days. You may want to change this for production scenarios, see https://aka.ms/aspnetcore-hsts.
|
||||
|
||||
// This method gets called by the runtime. Use this method to configure the HTTP request pipeline.
|
||||
public void Configure(IApplicationBuilder app, IWebHostEnvironment env) {
|
||||
if (env.IsDevelopment())
|
||||
app.UseDeveloperExceptionPage();
|
||||
else
|
||||
app.UseExceptionHandler("/Error");
|
||||
// The default HSTS value is 30 days. You may want to change this for production scenarios, see https://aka.ms/aspnetcore-hsts.
|
||||
app.UseStaticFiles();
|
||||
|
||||
app.UseStaticFiles();
|
||||
app.UseRouting();
|
||||
|
||||
app.UseRouting();
|
||||
app.UseAuthorization();
|
||||
|
||||
app.UseAuthorization();
|
||||
|
||||
app.UseEndpoints(endpoints => { endpoints.MapRazorPages(); });
|
||||
app.UseEndpoints(endpoints => { endpoints.MapRazorPages(); });
|
||||
}
|
||||
}
|
||||
}
|
4
Types.cs
4
Types.cs
|
@ -56,7 +56,7 @@ namespace c3stream {
|
|||
|
||||
public class Event {
|
||||
[JsonProperty("guid", NullValueHandling = NullValueHandling.Ignore)]
|
||||
public string Guid { get; set; }
|
||||
public Guid? Guid { get; set; }
|
||||
|
||||
[JsonProperty("title", NullValueHandling = NullValueHandling.Ignore)]
|
||||
public string Title { get; set; }
|
||||
|
@ -144,7 +144,7 @@ namespace c3stream {
|
|||
|
||||
public partial class Talk {
|
||||
[JsonProperty("guid", NullValueHandling = NullValueHandling.Ignore)]
|
||||
public string Guid { get; set; }
|
||||
public Guid? Guid { get; set; }
|
||||
|
||||
[JsonProperty("title", NullValueHandling = NullValueHandling.Ignore)]
|
||||
public string Title { get; set; }
|
||||
|
|
165
c3stream.cs
165
c3stream.cs
|
@ -1,147 +1,46 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.IO;
|
||||
using System.Linq;
|
||||
using System.Net.Http;
|
||||
using c3stream.DataModels;
|
||||
using LinqToDB.Data;
|
||||
using c3stream.Pages;
|
||||
using Microsoft.AspNetCore.Hosting;
|
||||
using Microsoft.AspNetCore.Http;
|
||||
using Microsoft.Extensions.Hosting;
|
||||
|
||||
namespace c3stream;
|
||||
namespace c3stream {
|
||||
public static class c3stream {
|
||||
public const string DataPath = "data";
|
||||
public const string DbFile = "_c3stream.json";
|
||||
public const string CachePath = "/mnt/storage/archive/Video/congress/";
|
||||
public const string CacheUrl = "https://mirror.c3stream.de/";
|
||||
public static object Lock = new object();
|
||||
public static string DbPath = Path.Combine(DataPath, DbFile);
|
||||
|
||||
public static class c3stream {
|
||||
public const string DataPath = "data";
|
||||
public const string DbFile = "c3stream.sqlite";
|
||||
public const string LogoPath = "/mnt/nvme-data/c3stream-logos/";
|
||||
public const string LogoUrl = "https://mirror.c3stream.de/logos/";
|
||||
public const string CachePath = "/mnt/zfs/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() {
|
||||
new ConferenceObject("37c3", true),
|
||||
new ConferenceObject("mrmcd23"),
|
||||
new ConferenceObject("camp2023"),
|
||||
new ConferenceObject("gpn21"),
|
||||
new ConferenceObject("trans-tech-tent"),
|
||||
new ConferenceObject("jev22"),
|
||||
new ConferenceObject("MCH2022"),
|
||||
new ConferenceObject("gpn20"),
|
||||
new ConferenceObject("rc3-2021"),
|
||||
new ConferenceObject("rc3"),
|
||||
new ConferenceObject("36c3"),
|
||||
new ConferenceObject("camp2019"),
|
||||
new ConferenceObject("gpn19"),
|
||||
new ConferenceObject("35c3"),
|
||||
new ConferenceObject("34c3"),
|
||||
new ConferenceObject("33c3"),
|
||||
new ConferenceObject("32c3"),
|
||||
new ConferenceObject("31c3"),
|
||||
new ConferenceObject("30c3")
|
||||
};
|
||||
|
||||
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);
|
||||
|
||||
DataConnection.DefaultSettings = new Database.Settings();
|
||||
Migrations.RunMigrations();
|
||||
|
||||
foreach (var conference in Conferences)
|
||||
UpdateConference(conference);
|
||||
|
||||
if (args.Length != 0) {
|
||||
if (args[0] == "logo")
|
||||
foreach (var conference in Conferences)
|
||||
Console.WriteLine($"wget {conference.LogoUri} -O {Path.Combine(LogoPath, conference.Acronym + ".png")}");
|
||||
else if (Conferences.All(p => p.Acronym != args[0]))
|
||||
Console.WriteLine("No matching conference found.");
|
||||
else
|
||||
foreach (var talk in Conferences.First(p => p.Acronym == args[0]).Talks)
|
||||
Console.WriteLine($"youtube-dl -f \"best[ext = mp4]\" {talk.FrontendLink} -o \"{Path.Combine(CachePath, args[0], talk.Slug)}.mp4\"");
|
||||
}
|
||||
else {
|
||||
public static void Main(string[] args) {
|
||||
if (!Directory.Exists(DataPath))
|
||||
Directory.CreateDirectory(DataPath);
|
||||
if (!File.Exists(DbPath))
|
||||
ConferenceModel.WriteEventMetadata();
|
||||
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);
|
||||
public static void UpdateCookie(HttpRequest resquest, HttpResponse response, string redirectUri) {
|
||||
//if new bookmark is in uri
|
||||
if (resquest.Query.ContainsKey("bookmark") && resquest.Cookies["bookmark"] != resquest.Query["bookmark"]) {
|
||||
response.Cookies.Append("bookmark", resquest.Query["bookmark"], new CookieOptions {Expires = DateTimeOffset.MaxValue});
|
||||
response.Redirect(redirectUri + "bookmark=" + resquest.Query["bookmark"]);
|
||||
}
|
||||
//if no cookie exists or cookie is invalid
|
||||
else if (!resquest.Cookies.ContainsKey("bookmark") || !Guid.TryParseExact(resquest.Cookies["bookmark"], "D", out _)) {
|
||||
var guid = Guid.NewGuid().ToString();
|
||||
response.Cookies.Append("bookmark", guid, new CookieOptions {Expires = DateTimeOffset.MaxValue});
|
||||
response.Redirect(redirectUri + "bookmark=" + guid);
|
||||
}
|
||||
//redir to cookie
|
||||
else if (!resquest.Query.ContainsKey("bookmark")) {
|
||||
response.Redirect(redirectUri + "bookmark=" + resquest.Cookies["bookmark"]);
|
||||
}
|
||||
}
|
||||
|
||||
var parsed = Conference.FromJson(json);
|
||||
lock (Lock) {
|
||||
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) {
|
||||
var cookie = "";
|
||||
//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"))
|
||||
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 IEnumerable<Event> GetEventsByGuid(IEnumerable<string> guids) {
|
||||
return Conferences.SelectMany(c => c.Talks.Where(e => guids.Contains(e.Guid)));
|
||||
}
|
||||
|
||||
public static ConferenceObject GetConferenceByEventGuid(string guid) {
|
||||
return Conferences.FirstOrDefault(c => c.Talks.Any(t => t.Guid == guid));
|
||||
}
|
||||
|
||||
public static IHostBuilder CreateHostBuilder(string[] args) => Host.CreateDefaultBuilder(args).ConfigureWebHostDefaults(webBuilder => { webBuilder.UseStartup<Startup>(); });
|
||||
|
||||
public class ConferenceObject {
|
||||
public string Acronym;
|
||||
public string LogoUri;
|
||||
public bool Ongoing;
|
||||
public List<Event> Talks = new();
|
||||
|
||||
public ConferenceObject(string acronym, bool ongoing = false) {
|
||||
Acronym = acronym;
|
||||
Ongoing = ongoing;
|
||||
}
|
||||
public static IHostBuilder CreateHostBuilder(string[] args) =>
|
||||
Host.CreateDefaultBuilder(args).ConfigureWebHostDefaults(webBuilder => { webBuilder.UseStartup<Startup>(); });
|
||||
}
|
||||
}
|
|
@ -1,43 +1,36 @@
|
|||
<Project Sdk="Microsoft.NET.Sdk.Web">
|
||||
|
||||
<PropertyGroup>
|
||||
<TargetFramework>net70</TargetFramework>
|
||||
<Configurations>Release;Debug</Configurations>
|
||||
<Platforms>x64</Platforms>
|
||||
<TargetFramework>netcoreapp3.1</TargetFramework>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="linq2db" Version="3.6.0" />
|
||||
<PackageReference Include="Newtonsoft.Json" Version="13.0.3" />
|
||||
<PackageReference Include="System.Data.SQLite.Core" Version="1.0.115" />
|
||||
<PackageReference Include="System.Data.SQLite.Core.osx.arm64" Version="1.0.117" />
|
||||
<PackageReference Include="Newtonsoft.Json" Version="12.0.3" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<_ContentIncludedByDefault Remove="data\database.json" />
|
||||
<_ContentIncludedByDefault Remove="data\_c3stream.json" />
|
||||
<_ContentIncludedByDefault Remove="data\33c3.json" />
|
||||
<_ContentIncludedByDefault Remove="data\34c3.json" />
|
||||
<_ContentIncludedByDefault Remove="data\35c3.json" />
|
||||
<_ContentIncludedByDefault Remove="data\database.json" />
|
||||
<_ContentIncludedByDefault Remove="data\_c3stream.json" />
|
||||
<_ContentIncludedByDefault Remove="data\33c3.json" />
|
||||
<_ContentIncludedByDefault Remove="data\34c3.json" />
|
||||
<_ContentIncludedByDefault Remove="data\35c3.json" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<Compile Remove="data\**" />
|
||||
<Compile Remove="data\**" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<EmbeddedResource Remove="data\**" />
|
||||
<EmbeddedResource Remove="data\**" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<None Remove="data\**" />
|
||||
<None Remove="data\**" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<Content Remove="data\**" />
|
||||
<Content Remove="data\**" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<Folder Include="DataModels" />
|
||||
</ItemGroup>
|
||||
|
||||
</Project>
|
||||
|
|
12
c3stream.sln
12
c3stream.sln
|
@ -4,13 +4,13 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "c3stream", "c3stream.csproj
|
|||
EndProject
|
||||
Global
|
||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||
Debug|x64 = Debug|x64
|
||||
Release|x64 = Release|x64
|
||||
Debug|Any CPU = Debug|Any CPU
|
||||
Release|Any CPU = Release|Any CPU
|
||||
EndGlobalSection
|
||||
GlobalSection(ProjectConfigurationPlatforms) = postSolution
|
||||
{BC6A24FE-B35F-4F0A-8E8E-221E61E41EEF}.Debug|x64.ActiveCfg = Debug|x64
|
||||
{BC6A24FE-B35F-4F0A-8E8E-221E61E41EEF}.Debug|x64.Build.0 = Debug|x64
|
||||
{BC6A24FE-B35F-4F0A-8E8E-221E61E41EEF}.Release|x64.ActiveCfg = Release|x64
|
||||
{BC6A24FE-B35F-4F0A-8E8E-221E61E41EEF}.Release|x64.Build.0 = Release|x64
|
||||
{BC6A24FE-B35F-4F0A-8E8E-221E61E41EEF}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{BC6A24FE-B35F-4F0A-8E8E-221E61E41EEF}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{BC6A24FE-B35F-4F0A-8E8E-221E61E41EEF}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{BC6A24FE-B35F-4F0A-8E8E-221E61E41EEF}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
EndGlobalSection
|
||||
EndGlobal
|
||||
|
|
Binary file not shown.
|
@ -1,12 +0,0 @@
|
|||
#!/bin/bash
|
||||
# Requires jq and sqlite3
|
||||
|
||||
cp database.init.sqlite data/c3stream.sqlite
|
||||
|
||||
rm migration.sql
|
||||
|
||||
for row in $(cat "data/c3stream.user.json" | jq -c '.[]'); do
|
||||
echo "INSERT INTO States (TalkId, UserId, State) VALUES ($(echo $row | jq '.TalkId'),$(echo $row | jq '.UserId'),$(echo $row | jq '.State'));" | tee -a migration.sql
|
||||
done
|
||||
|
||||
cat migration.sql | sqlite3 data/c3stream.sqlite
|
BIN
wwwroot/.DS_Store
vendored
BIN
wwwroot/.DS_Store
vendored
Binary file not shown.
File diff suppressed because one or more lines are too long
|
@ -2,112 +2,74 @@
|
|||
for details on configuring this project to bundle and minify static web assets. */
|
||||
|
||||
a.navbar-brand {
|
||||
color: #e3d7c0 !important;
|
||||
white-space: normal;
|
||||
text-align: center;
|
||||
word-break: break-all;
|
||||
white-space: normal;
|
||||
text-align: center;
|
||||
word-break: break-all;
|
||||
}
|
||||
|
||||
a:not(.btn):not(.navbar-brand) {
|
||||
color: #7ca9c8 !important;
|
||||
}
|
||||
|
||||
.navbar {
|
||||
color: #ebddc4 !important;
|
||||
background-color: #1b1b18 !important;
|
||||
border-color: #3c3d3c !important;
|
||||
}
|
||||
|
||||
/* Sticky footer styles
|
||||
-------------------------------------------------- */
|
||||
html {
|
||||
font-size: 14px;
|
||||
}
|
||||
|
||||
@media (min-width: 768px) {
|
||||
html {
|
||||
font-size: 16px;
|
||||
}
|
||||
}
|
||||
|
||||
.box-shadow {
|
||||
box-shadow: 0 .25rem .75rem rgba(0, 0, 0, .05);
|
||||
}
|
||||
|
||||
button.accept-policy {
|
||||
font-size: 1rem;
|
||||
line-height: inherit;
|
||||
}
|
||||
|
||||
/* Sticky footer styles
|
||||
-------------------------------------------------- */
|
||||
html {
|
||||
position: relative;
|
||||
min-height: 100%;
|
||||
}
|
||||
|
||||
body {
|
||||
/* Margin bottom by footer height */
|
||||
margin-bottom: 60px;
|
||||
color: #ebddc4 !important;
|
||||
background-color: #1b1b18 !important;
|
||||
}
|
||||
|
||||
.table {
|
||||
color: #ebddc4 !important;
|
||||
}
|
||||
|
||||
.footer {
|
||||
position: absolute;
|
||||
bottom: 0;
|
||||
width: 100%;
|
||||
white-space: nowrap;
|
||||
line-height: 60px; /* Vertically center the text there */
|
||||
}
|
||||
|
||||
.table td {
|
||||
vertical-align: middle;
|
||||
border-color: #3c3d3c !important;
|
||||
}
|
||||
|
||||
.table th {
|
||||
vertical-align: middle;
|
||||
border-color: #3c3d3c !important;
|
||||
}
|
||||
|
||||
.btn-primary.disabled {
|
||||
color: #afafa2;
|
||||
border-color: #3c6385;
|
||||
background-color: #254667 !important;
|
||||
opacity: 1;
|
||||
border-top-color: #254667 !important;
|
||||
border-bottom-color: #254667 !important;
|
||||
/* Provide sufficient contrast against white background */
|
||||
a {
|
||||
color: #0366d6;
|
||||
}
|
||||
|
||||
.btn-primary {
|
||||
color: #ffffff;
|
||||
border-color: #3c6385;
|
||||
background-color: #375a7a;
|
||||
color: #fff;
|
||||
background-color: #1b6ec2;
|
||||
border-color: #1861ac;
|
||||
}
|
||||
|
||||
.btn-c3saction {
|
||||
width: 42px !important;
|
||||
.nav-pills .nav-link.active, .nav-pills .show > .nav-link {
|
||||
color: #fff;
|
||||
background-color: #1b6ec2;
|
||||
border-color: #1861ac;
|
||||
}
|
||||
|
||||
.btn-primary:hover, .btn-primary:focus, .btn-primary:active, .btn-primary.active, .open > .dropdown-toggle.btn-primary {
|
||||
color: #ffffff !important;
|
||||
border-color: #3c6385 !important;
|
||||
background-color: #2c5f93 !important;
|
||||
/* Sticky footer styles
|
||||
-------------------------------------------------- */
|
||||
html {
|
||||
font-size: 14px;
|
||||
}
|
||||
@media (min-width: 768px) {
|
||||
html {
|
||||
font-size: 16px;
|
||||
}
|
||||
}
|
||||
|
||||
.border-top {
|
||||
border-top-color: #3c3d3c !important;
|
||||
border-top: 1px solid #e5e5e5;
|
||||
}
|
||||
.border-bottom {
|
||||
border-bottom: 1px solid #e5e5e5;
|
||||
}
|
||||
|
||||
code {
|
||||
color: #cB6d87 !important;
|
||||
.box-shadow {
|
||||
box-shadow: 0 .25rem .75rem rgba(0, 0, 0, .05);
|
||||
}
|
||||
|
||||
.fa-times {
|
||||
|
||||
button.accept-policy {
|
||||
font-size: 1rem;
|
||||
line-height: inherit;
|
||||
}
|
||||
|
||||
/* Sticky footer styles
|
||||
-------------------------------------------------- */
|
||||
html {
|
||||
position: relative;
|
||||
min-height: 100%;
|
||||
}
|
||||
|
||||
body {
|
||||
/* Margin bottom by footer height */
|
||||
margin-bottom: 60px;
|
||||
}
|
||||
.footer {
|
||||
position: absolute;
|
||||
bottom: 0;
|
||||
width: 100%;
|
||||
white-space: nowrap;
|
||||
line-height: 60px; /* Vertically center the text there */
|
||||
}
|
||||
|
||||
.table td {
|
||||
vertical-align: middle;
|
||||
}
|
|
@ -231,21 +231,21 @@ select {
|
|||
}
|
||||
|
||||
button,
|
||||
[role="button"],
|
||||
[type="button"],
|
||||
[type="reset"],
|
||||
[type="submit"] {
|
||||
-webkit-appearance: button;
|
||||
}
|
||||
|
||||
button:not(:disabled),
|
||||
[role="button"]:not(:disabled),
|
||||
[type="button"]:not(:disabled),
|
||||
[type="reset"]:not(:disabled),
|
||||
[type="submit"]:not(:disabled) {
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
button::-moz-focus-inner,
|
||||
[role="button"]::-moz-focus-inner,
|
||||
[type="button"]::-moz-focus-inner,
|
||||
[type="reset"]::-moz-focus-inner,
|
||||
[type="submit"]::-moz-focus-inner {
|
||||
padding: 0;
|
||||
|
|
8
wwwroot/lib/bootstrap/dist/css/bootstrap.css
vendored
8
wwwroot/lib/bootstrap/dist/css/bootstrap.css
vendored
|
@ -261,21 +261,21 @@ select {
|
|||
}
|
||||
|
||||
button,
|
||||
[role="button"],
|
||||
[type="button"],
|
||||
[type="reset"],
|
||||
[type="submit"] {
|
||||
-webkit-appearance: button;
|
||||
}
|
||||
|
||||
button:not(:disabled),
|
||||
[role="button"]:not(:disabled),
|
||||
[type="button"]:not(:disabled),
|
||||
[type="reset"]:not(:disabled),
|
||||
[type="submit"]:not(:disabled) {
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
button::-moz-focus-inner,
|
||||
[role="button"]::-moz-focus-inner,
|
||||
[type="button"]::-moz-focus-inner,
|
||||
[type="reset"]::-moz-focus-inner,
|
||||
[type="submit"]::-moz-focus-inner {
|
||||
padding: 0;
|
||||
|
@ -2993,7 +2993,7 @@ fieldset:disabled a.btn {
|
|||
|
||||
input[type="submit"].btn-block,
|
||||
input[type="reset"].btn-block,
|
||||
input[role="button"].btn-block {
|
||||
input[type="button"].btn-block {
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
|
|
BIN
wwwroot/webfonts/fa-brands-400.eot
Normal file
BIN
wwwroot/webfonts/fa-brands-400.eot
Normal file
Binary file not shown.
3496
wwwroot/webfonts/fa-brands-400.svg
Normal file
3496
wwwroot/webfonts/fa-brands-400.svg
Normal file
File diff suppressed because it is too large
Load diff
After Width: | Height: | Size: 684 KiB |
BIN
wwwroot/webfonts/fa-brands-400.ttf
Normal file
BIN
wwwroot/webfonts/fa-brands-400.ttf
Normal file
Binary file not shown.
BIN
wwwroot/webfonts/fa-brands-400.woff
Normal file
BIN
wwwroot/webfonts/fa-brands-400.woff
Normal file
Binary file not shown.
BIN
wwwroot/webfonts/fa-brands-400.woff2
Normal file
BIN
wwwroot/webfonts/fa-brands-400.woff2
Normal file
Binary file not shown.
BIN
wwwroot/webfonts/fa-duotone-900.eot
Normal file
BIN
wwwroot/webfonts/fa-duotone-900.eot
Normal file
Binary file not shown.
13908
wwwroot/webfonts/fa-duotone-900.svg
Normal file
13908
wwwroot/webfonts/fa-duotone-900.svg
Normal file
File diff suppressed because it is too large
Load diff
After Width: | Height: | Size: 2.3 MiB |
BIN
wwwroot/webfonts/fa-duotone-900.ttf
Normal file
BIN
wwwroot/webfonts/fa-duotone-900.ttf
Normal file
Binary file not shown.
BIN
wwwroot/webfonts/fa-duotone-900.woff
Normal file
BIN
wwwroot/webfonts/fa-duotone-900.woff
Normal file
Binary file not shown.
BIN
wwwroot/webfonts/fa-duotone-900.woff2
Normal file
BIN
wwwroot/webfonts/fa-duotone-900.woff2
Normal file
Binary file not shown.
BIN
wwwroot/webfonts/fa-light-300.eot
Normal file
BIN
wwwroot/webfonts/fa-light-300.eot
Normal file
Binary file not shown.
11174
wwwroot/webfonts/fa-light-300.svg
Normal file
11174
wwwroot/webfonts/fa-light-300.svg
Normal file
File diff suppressed because it is too large
Load diff
After Width: | Height: | Size: 2.1 MiB |
BIN
wwwroot/webfonts/fa-light-300.ttf
Normal file
BIN
wwwroot/webfonts/fa-light-300.ttf
Normal file
Binary file not shown.
BIN
wwwroot/webfonts/fa-light-300.woff
Normal file
BIN
wwwroot/webfonts/fa-light-300.woff
Normal file
Binary file not shown.
BIN
wwwroot/webfonts/fa-light-300.woff2
Normal file
BIN
wwwroot/webfonts/fa-light-300.woff2
Normal file
Binary file not shown.
BIN
wwwroot/webfonts/fa-regular-400.eot
Normal file
BIN
wwwroot/webfonts/fa-regular-400.eot
Normal file
Binary file not shown.
10229
wwwroot/webfonts/fa-regular-400.svg
Normal file
10229
wwwroot/webfonts/fa-regular-400.svg
Normal file
File diff suppressed because it is too large
Load diff
After Width: | Height: | Size: 1.9 MiB |
BIN
wwwroot/webfonts/fa-regular-400.ttf
Normal file
BIN
wwwroot/webfonts/fa-regular-400.ttf
Normal file
Binary file not shown.
BIN
wwwroot/webfonts/fa-regular-400.woff
Normal file
BIN
wwwroot/webfonts/fa-regular-400.woff
Normal file
Binary file not shown.
BIN
wwwroot/webfonts/fa-regular-400.woff2
Normal file
BIN
wwwroot/webfonts/fa-regular-400.woff2
Normal file
Binary file not shown.
BIN
wwwroot/webfonts/fa-solid-900.eot
Normal file
BIN
wwwroot/webfonts/fa-solid-900.eot
Normal file
Binary file not shown.
8736
wwwroot/webfonts/fa-solid-900.svg
Normal file
8736
wwwroot/webfonts/fa-solid-900.svg
Normal file
File diff suppressed because it is too large
Load diff
After Width: | Height: | Size: 1.5 MiB |
BIN
wwwroot/webfonts/fa-solid-900.ttf
Normal file
BIN
wwwroot/webfonts/fa-solid-900.ttf
Normal file
Binary file not shown.
BIN
wwwroot/webfonts/fa-solid-900.woff
Normal file
BIN
wwwroot/webfonts/fa-solid-900.woff
Normal file
Binary file not shown.
BIN
wwwroot/webfonts/fa-solid-900.woff2
Normal file
BIN
wwwroot/webfonts/fa-solid-900.woff2
Normal file
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Some files were not shown because too many files have changed in this diff Show more
Loading…
Reference in a new issue