2018-09-27 11:11:26 -05:00
"use strict" ;
2018-11-30 14:46:22 -06:00
// I M P O R T S
2018-09-27 11:11:26 -05:00
2018-11-30 14:46:22 -06:00
import got from "got" ;
import html from "choo/html" ;
2018-09-27 11:11:26 -05:00
2018-10-10 12:56:35 -05:00
// U T I L S
2018-09-27 11:11:26 -05:00
2018-11-30 14:46:22 -06:00
import fetchMetadata from "./helpers/fetch-metadata" ;
import { generateGitHubFeed } from "./helpers/github" ;
import messageSlack from "./helpers/slack" ;
2018-09-27 11:11:26 -05:00
// P R O G R A M
2018-11-30 14:46:22 -06:00
export default ( socket , action ) => {
2018-09-27 11:11:26 -05:00
if ( typeof socket !== "object" && typeof action !== "object" ) return ;
switch ( true ) {
2018-11-30 14:46:22 -06:00
case action . message === "fetch metadata" :
2018-09-27 11:11:26 -05:00
fetchMetadata ( action , socket ) ;
break ;
2018-11-30 14:46:22 -06:00
case action . message === "landed on homepage" :
2018-09-27 11:11:26 -05:00
generateGitHubFeed ( result => {
socket . send ( JSON . stringify ( {
2018-10-06 15:53:01 -05:00
html : result ,
message : "updated html" ,
selector : "#github-feed"
2018-09-27 11:11:26 -05:00
} ) ) ;
} ) ;
break ;
2018-11-30 14:46:22 -06:00
case action . message === "landed on playground" :
2018-09-27 11:11:26 -05:00
generateContent ( 1 , result => {
socket . send ( JSON . stringify ( {
2018-10-06 15:53:01 -05:00
html : result ,
message : "updated html" ,
selector : "#playground-loader"
2018-09-27 11:11:26 -05:00
} ) ) ;
} ) ;
break ;
2018-11-30 14:46:22 -06:00
case action . message === "request for playground, example 1" :
2018-09-27 11:11:26 -05:00
generateContent ( 1 , result => {
socket . send ( JSON . stringify ( {
2018-10-06 15:53:01 -05:00
html : result ,
message : "updated html" ,
selector : "#playground-loader"
2018-09-27 11:11:26 -05:00
} ) ) ;
} ) ;
break ;
2018-11-30 14:46:22 -06:00
case action . message === "request for playground, example 2" :
2018-09-27 11:11:26 -05:00
generateMemeCreator ( socket ) ;
break ;
2018-11-30 14:46:22 -06:00
case action . message === "request for playground, example 3" :
2018-09-27 11:11:26 -05:00
generateContent ( 3 , result => {
socket . send ( JSON . stringify ( {
2018-10-06 15:53:01 -05:00
html : result ,
message : "updated html" ,
selector : "#playground-loader"
2018-09-27 11:11:26 -05:00
} ) ) ;
} ) ;
break ;
2018-11-30 14:46:22 -06:00
case action . message === "subscribe" :
2018-09-27 11:11:26 -05:00
newsletterSubscribe ( action , socket ) ;
break ;
default :
break ;
}
} ;
// H E L P E R S
2018-09-27 13:00:34 -05:00
function generateContent ( exampleNumber , displayTrendingContent ) {
if ( exampleNumber === 1 ) {
return getTrendingContent ( ) . then ( response => {
if ( ! response || ! response . success || response . success !== true || ! response . data ) return "" ;
const rawContentCollection = [ ] ;
const renderedContentCollection = [ ] ;
const trendingContentData = response . data ;
for ( const data of trendingContentData ) {
rawContentCollection . push ( fetchMetadata ( { claim : data . url , method : "resolve" , example : exampleNumber } ) ) ;
}
Promise . all ( rawContentCollection ) . then ( collection => {
for ( const part of collection ) {
try {
renderedContentCollection . push ( `
2018-12-04 17:12:39 -06:00
< section class = "playground-content__trend" >
< figure
class = "media__thumb"
data - action = "choose claim"
data - claim - id = "${part.name}"
style = "background-image: url(${makeImageSourceSecure(part.value.stream.metadata.thumbnail)})" >
< / f i g u r e >
< div class = "media__title" >
2018-09-27 13:00:34 -05:00
$ { part . value . stream . metadata . title }
2018-12-04 17:12:39 -06:00
< / d i v >
< div class = "media__subtitle" >
$ { part . channel _name }
< / d i v >
< / s e c t i o n >
2018-09-27 13:00:34 -05:00
` );
} catch ( err ) {
return ; // TODO: Return nice error message
}
}
renderedContentCollection . push ( `
< script >
2018-10-03 15:27:13 -05:00
document . getElementById ( "playground-example-description" ) . innerHTML = document . querySelector ( "[data-action='playground, example 1']" ) . dataset . description
2018-09-27 13:00:34 -05:00
< / s c r i p t >
` );
displayTrendingContent ( renderedContentCollection . join ( "" ) ) ;
} ) ;
} ) ;
}
if ( exampleNumber === 3 ) {
const approvedUrls = [
"LBRY#3db81c073f82fd1bb670c65f526faea3b8546720" ,
"correlation-can-imply-causation#173412f5b1b7aa63a752e8832406aafd9f1ecb4e" ,
2018-09-27 15:46:59 -05:00
"thanos-is-the-protagonist-how-infinity#2a7f5db2678177435b1dee6c9e38e035ead450b6" ,
2018-09-27 13:00:34 -05:00
"epic-arcade-mode-duos-nickatnyte-molt#d81bac6d49b1f92e58c37a5f633a27a45b43405e" ,
"political-correctness-a-force-for-good-a#b4668c0bd096317b44c40738c099b6618095e75f" ,
"10-secrets-hidden-inside-famous-logos#007789cc45cbb4255cf02ba77cbf84ca8e3d7561" ,
"ever-wonder-how-bitcoin-and-other#1ac47b8b3def40a25850dc726a09ce23d09e7009" ,
"bankrupt-pan-am#784b3c215a6f06b663fc1aa292bcb19f29c489bb" ,
"minecraft-in-real-life-iron-man#758dd6497cdfc401ae1f25984738d024d47b50af" ,
"ethan-shows-kyle-warframe-skyvault#8a7401b88d5ed0376d98f16808194d4dcb05b284"
] ;
const rawContentCollection = [ ] ;
const renderedContentCollection = [ ] ;
2018-09-28 14:09:45 -05:00
for ( const url of approvedUrls )
2018-09-27 13:00:34 -05:00
rawContentCollection . push ( fetchMetadata ( { claim : url , method : "resolve" , example : exampleNumber } ) ) ;
2018-09-27 15:46:59 -05:00
return Promise . all ( rawContentCollection ) . then ( collection => {
2018-09-27 13:00:34 -05:00
for ( const part of collection ) {
2018-09-27 15:46:59 -05:00
if (
part &&
part . value &&
part . value . stream . metadata . thumbnail &&
part . channel _name
) {
2018-09-27 13:00:34 -05:00
renderedContentCollection . push ( `
2018-12-04 17:12:39 -06:00
< section class = "playground-content__trend" >
< figure
class = "media__thumb"
data - action = "choose claim"
data - claim - id = "${part.claim_id}"
data - name = $ { part . name }
style = "background-image: url(${makeImageSourceSecure(part.value.stream.metadata.thumbnail)})" >
< / f i g u r e >
< div class = "media__title" >
2018-09-27 13:00:34 -05:00
$ { part . value . stream . metadata . title }
2018-12-04 17:12:39 -06:00
< / d i v >
< div class = "media__subtitle" >
$ { part . channel _name }
< / d i v >
< / s e c t i o n >
2018-09-27 13:00:34 -05:00
` );
}
}
renderedContentCollection . push ( `
< script >
2018-10-03 15:27:13 -05:00
document . getElementById ( "playground-example-description" ) . innerHTML = document . querySelector ( "[data-action='playground, example 3']" ) . dataset . description
2018-09-27 13:00:34 -05:00
< / s c r i p t >
` );
displayTrendingContent ( renderedContentCollection . join ( "" ) ) ;
} ) ;
}
}
2018-09-27 11:11:26 -05:00
function generateMemeCreator ( socket ) {
const images = [
{
alt : "Carl Sagan" ,
2018-12-12 12:47:53 -06:00
// src: "https://spee.ch/4f6b953e605a602434246743fd246d3e1fd4f5fd/carlsagan2.jpg"
src : "/assets/media/images/carlsagan2.jpg"
2018-09-27 11:11:26 -05:00
} ,
{
alt : "Doge" ,
2018-12-12 12:47:53 -06:00
// src: "https://spee.ch/2f90f2d91441a4d33d3d4eb82bdfc4c56ec742c7/doge-meme.jpg"
src : "/assets/media/images/doge-meme.jpg"
2018-09-27 11:11:26 -05:00
} ,
{
alt : "LBRY Logo With Green Background" ,
2018-12-12 12:47:53 -06:00
// src: "https://spee.ch/40ac6818bbac87a208722bf4467653341d460908/lbry-green.png"
src : "/assets/media/images/lbry-green.png"
2018-09-27 11:11:26 -05:00
}
] ;
const memePlaceholderData = {
bottomLine : {
placeholder : "Top line" ,
value : "that I made"
} ,
description : {
placeholder : "Description" ,
value : "Check out this image I published to LBRY via lbry.tech"
} ,
topLine : {
placeholder : "Top line" ,
value : "This is an example meme"
} ,
title : {
placeholder : "Title" ,
value : "Dank Meme Supreme da Cheese"
}
} ;
const renderedImages = [ ] ;
2018-09-28 14:09:45 -05:00
for ( const image of images )
2018-12-04 17:12:39 -06:00
renderedImages . push ( ` <img alt=" ${ image . alt } " class="playground-content__meme__canvas__thumbnail" src=" ${ image . src } "/> ` ) ;
2018-09-27 11:11:26 -05:00
const memeCreator = html `
2018-12-04 17:12:39 -06:00
< div class = "playground-content__meme__canvas" >
2018-09-27 11:11:26 -05:00
< img alt = "Base image for LBRY meme creator" id = "base-image" style = "height: 0; position: absolute; visibility: hidden;" / >
2018-09-28 16:59:38 -05:00
< canvas id = "meme-canvas" height = "600" width = "800" > Unfortunately , it looks like canvas is < strong > not supported < / s t r o n g > i n y o u r b r o w s e r < / c a n v a s >
2018-09-27 11:11:26 -05:00
$ { renderedImages }
< / d i v >
2018-12-04 17:12:39 -06:00
< form class = "playground-content__meme__editor" >
2018-09-27 11:11:26 -05:00
< h2 > Image Text < / h 2 >
< fieldset >
< label for = "meme-top-line" > Top line < / l a b e l >
< input id = "meme-top-line" name = "meme-top-line" placeholder = "${memePlaceholderData.topLine.placeholder}" spellcheck = "false" type = "text" value = "${memePlaceholderData.topLine.value}" required / >
< / f i e l d s e t >
< fieldset >
< label for = "meme-bottom-line" > Bottom line < / l a b e l >
< input id = "meme-bottom-line" name = "meme-bottom-line" placeholder = "${memePlaceholderData.bottomLine.placeholder}" spellcheck = "false" type = "text" value = "${memePlaceholderData.bottomLine.value}" required / >
< / f i e l d s e t >
< h2 class = "__metadata" > Metadata < / h 2 >
< fieldset >
< label for = "meme-title" > Title < / l a b e l >
< input id = "meme-title" name = "meme-title" placeholder = "${memePlaceholderData.title.placeholder}" spellcheck = "false" type = "text" value = "${memePlaceholderData.title.value}" required / >
< / f i e l d s e t >
< fieldset >
< label for = "meme-description" > Description < / l a b e l >
< textarea id = "meme-description" name = "meme-description" placeholder = "${memePlaceholderData.description.placeholder}" spellcheck = "false" type = "text" required > $ { memePlaceholderData . description . value } < / t e x t a r e a >
< / f i e l d s e t >
< fieldset >
< label for = "meme-language" > Language < / l a b e l >
< select id = "meme-language" name = "meme-language" >
< option value = "ar" > Arabic < / o p t i o n >
< option value = "zh" > Chinese ( Mandarin ) < / o p t i o n >
< option value = "en" > English < / o p t i o n >
< option value = "fr" > French < / o p t i o n >
< option value = "de" > German < / o p t i o n >
< option value = "it" > Italian < / o p t i o n >
< option value = "jp" > Japanese < / o p t i o n >
< option value = "ru" > Russian < / o p t i o n >
< option value = "es" > Spanish < / o p t i o n >
< option value = "" > Not specified < / o p t i o n >
< / s e l e c t >
< / f i e l d s e t >
< fieldset >
< label for = "meme-license" > License < / l a b e l >
< select id = "meme-license" name = "meme-license" required >
< option value = "Public Domain" > Public Domain < / o p t i o n >
< option value = "Creative Commons Attribution 4.0 International" > Creative Commons Attribution 4.0 International < / o p t i o n >
< option value = "Creative Commons Attribution-ShareAlike 4.0 International" > Creative Commons Attribution - ShareAlike 4.0 International < / o p t i o n >
< option value = "Creative Commons Attribution-NoDerivatives 4.0 International" > Creative Commons Attribution - NoDerivatives 4.0 International < / o p t i o n >
< option value = "Creative Commons Attribution-NonCommercial 4.0 International" > Creative Commons Attribution - NonCommercial 4.0 International < / o p t i o n >
< option value = "Creative Commons Attribution-NonCommercial-ShareAlike 4.0 International" > Creative Commons Attribution - NonCommercial - ShareAlike 4.0 International < / o p t i o n >
< option value = "Creative Commons Attribution-NonCommercial-NoDerivatives 4.0 International" > Creative Commons Attribution - NonCommercial - NoDerivatives 4.0 International < / o p t i o n >
< option value = "None" > None < / o p t i o n >
< / s e l e c t >
< / f i e l d s e t >
< fieldset >
< label > < input id = "meme-nsfw-flag" name = "nsfw" type = "checkbox" / > NSFW < / l a b e l >
< / f i e l d s e t >
< fieldset >
2018-10-09 13:28:58 -05:00
< button data - action = "upload image" class = "button" type = "button" > Submit < / b u t t o n >
2018-09-27 11:11:26 -05:00
< / f i e l d s e t >
< / f o r m >
` ;
return socket . send ( JSON . stringify ( {
2018-10-06 15:53:01 -05:00
example : 2 ,
html : memeCreator ,
message : "updated html" ,
selector : "#playground-loader"
2018-09-27 11:11:26 -05:00
} ) ) ;
}
2018-11-21 15:57:43 -06:00
async function getTrendingContent ( ) {
try {
const response = await got ( "https://api.lbry.io/file/list_trending" ) ;
return JSON . parse ( response . body ) ; // eslint-disable-line padding-line-between-statements
} catch ( error ) {
return error ;
}
2018-09-27 11:11:26 -05:00
}
2018-10-09 17:11:22 -05:00
function makeImageSourceSecure ( url ) {
const originalUrl = new URL ( url ) ;
if ( originalUrl . protocol !== "https" )
return ` https:// ${ originalUrl . host } ${ originalUrl . pathname } ` ;
return originalUrl . href ;
}
2018-11-21 15:57:43 -06:00
async function newsletterSubscribe ( data , socket ) {
2018-09-27 11:11:26 -05:00
const email = data . email ;
if ( ! validateEmail ( email ) ) return socket . send ( JSON . stringify ( {
2018-10-10 14:04:16 -05:00
class : "error" ,
html : "Your email address is invalid" ,
2018-10-06 15:53:01 -05:00
message : "updated html" ,
selector : "#emailMessage"
2018-09-27 11:11:26 -05:00
} ) ) ;
2018-11-21 15:57:43 -06:00
try {
await got . post ( ` https://api.lbry.io/list/subscribe?email= ${ encodeURIComponent ( email ) } &tag=developer ` ) ;
2018-09-27 11:11:26 -05:00
2018-11-21 15:57:43 -06:00
return socket . send ( JSON . stringify ( {
html : "Thank you! Please confirm subscription in your inbox." ,
message : "updated html" ,
selector : "#emailMessage"
} ) ) ;
} catch ( error ) {
const response = JSON . parse ( error . body ) ;
2018-09-27 11:11:26 -05:00
2018-11-21 15:57:43 -06:00
if ( ! response . success ) {
2018-11-30 14:46:22 -06:00
messageSlack (
2018-10-10 14:04:16 -05:00
"\n" +
2018-11-21 15:57:43 -06:00
"> *NEWSLETTER ERROR:* ```" + response . error + "```" + "\n" +
2018-10-10 14:04:16 -05:00
` > _Cause: ${ email } interacted with the form_ \n `
) ;
2018-11-21 15:57:43 -06:00
return socket . send ( JSON . stringify ( {
2018-10-10 14:04:16 -05:00
class : "error" ,
2018-11-21 15:57:43 -06:00
html : response . error ,
2018-10-10 14:04:16 -05:00
message : "updated html" ,
selector : "#emailMessage"
2018-11-21 15:57:43 -06:00
} ) ) ;
2018-10-10 14:04:16 -05:00
}
2018-11-30 14:46:22 -06:00
messageSlack (
2018-11-21 15:57:43 -06:00
"\n" +
"> *NEWSLETTER ERROR:* ```¯\\_(ツ)_/¯ This should be an unreachable error```" + "\n" +
` > _Cause: ${ email } interacted with the form_ \n `
) ;
return socket . send ( JSON . stringify ( {
class : "error" ,
html : "Something is terribly wrong" ,
2018-10-10 14:04:16 -05:00
message : "updated html" ,
selector : "#emailMessage"
} ) ) ;
2018-11-21 15:57:43 -06:00
}
2018-09-27 11:11:26 -05:00
}
function validateEmail ( email ) {
const emailRegex = /^(([^<>()[\].,;:\s@"]+(\.[^<>()[\].,;:\s@"]+)*)|(".+"))@(([^<>()[\].,;:\s@"]+\.)+[^<>()[\\.,;:\s@"]{2,})$/i ;
2018-11-21 15:57:43 -06:00
return emailRegex . test ( String ( email ) ) ; // eslint-disable-line padding-line-between-statements
2018-09-27 11:11:26 -05:00
}