2018-08-28 18:57:18 -05:00
"use strict" ;
2018-10-01 15:47:10 -05:00
2018-11-30 14:46:22 -06:00
// I M P O R T S
2018-10-01 15:47:10 -05:00
2018-08-29 11:58:55 -05:00
import asyncHtml from "choo-async/html" ;
2018-08-28 18:57:18 -05:00
import dedent from "dedent" ;
2019-01-08 11:25:34 -06:00
import got from "got" ;
2019-07-10 12:34:19 -05:00
import Octokit from "@octokit/rest" ;
2018-10-01 15:47:10 -05:00
2018-10-10 12:56:35 -05:00
// U T I L S
2018-08-28 18:57:18 -05:00
2019-04-12 16:17:39 -04:00
import headerBlockchain from "~component/api/header-blockchain" ;
import headerSdk from "~component/api/header-sdk" ;
import redirects from "~data/redirects.json" ;
2018-11-30 14:46:22 -06:00
2019-04-16 14:52:51 -05:00
const cache = new Map ( ) ;
2019-07-10 12:34:19 -05:00
const filePathBlockchain = "/contrib/devtools/generated/api_v1.json" ;
2019-12-31 19:20:37 -05:00
const filePathSdk = "/docs/api.json" ;
2019-07-10 12:34:19 -05:00
const rawGitHubBase = "https://raw.githubusercontent.com/lbryio/" ;
const octokit = new Octokit ( {
auth : ` token ${ process . env . GITHUB _OAUTH _TOKEN } `
} ) ;
2018-10-01 15:47:10 -05:00
// E X P O R T
2018-08-28 18:57:18 -05:00
2018-11-30 14:46:22 -06:00
export default async ( state ) => {
2019-07-10 16:48:45 -05:00
const { tag } = state ;
2019-07-10 12:34:19 -05:00
const { wildcard } = state . params ;
const repository = wildcard === "sdk" ?
"lbry-sdk" :
"lbrycrd" ;
2019-07-10 16:48:45 -05:00
state . lbry = {
title : tag ? tag + " API Documentation" : "API Documentation" ,
description : "See API documentation, signatures, and sample calls for the LBRY APIs."
} ;
2019-07-10 12:34:19 -05:00
const tags = await getTags ( repository ) ;
2019-07-11 17:46:38 -05:00
const currentTag = tag && tag . length ? tag : tags [ 0 ] ;
2019-07-10 12:34:19 -05:00
2018-11-30 14:46:22 -06:00
try {
2019-07-11 17:09:36 -05:00
const apiResponse = await parseApiFile ( { repo : repository , tag : currentTag } ) ;
2018-11-30 14:46:22 -06:00
return asyncHtml `
< div class = "__slate" >
2019-01-30 16:15:10 -06:00
< aside class = "api-toc" >
2019-07-10 12:34:19 -05:00
< select class = "api-toc__select" onchange = "changeDocumentationVersion(value);" >
2019-07-10 16:48:45 -05:00
$ { renderVersionSelector ( wildcard , tags , tag ) }
2019-07-10 12:34:19 -05:00
< / s e l e c t >
2019-07-10 16:48:45 -05:00
2019-01-30 16:15:10 -06:00
< div class = "api-toc__search" >
< input class = "api-toc__search-field" id = "input-search" placeholder = "Search" type = "search" / >
< div class = "api-toc__search-clear" id = "clear-search" title = "Clear search query" > & times ; < / d i v >
< ul class = "api-toc__search-results" > < / u l >
2018-11-30 14:46:22 -06:00
< / d i v >
2019-04-12 15:08:35 -04:00
< ul class = "api-toc__commands" id = "toc" role = "navigation" >
2019-07-10 12:34:19 -05:00
$ { wildcard === "sdk" ? createSdkSidebar ( apiResponse ) : createApiSidebar ( apiResponse ) }
2019-04-12 15:08:35 -04:00
< / u l >
2018-11-30 14:46:22 -06:00
< / a s i d e >
2019-07-10 16:48:45 -05:00
2019-01-30 16:15:10 -06:00
< section class = "api-content" >
< div class = "api-documentation" id = "toc-content" >
2019-02-19 17:42:52 -06:00
< div > < / d i v >
2019-07-10 16:48:45 -05:00
2019-01-30 16:15:10 -06:00
< nav class = "api-content__items" >
2019-07-10 12:34:19 -05:00
$ { renderCodeLanguageToggles ( wildcard ) }
2019-01-30 16:15:10 -06:00
< / n a v >
2019-07-11 17:09:36 -05:00
$ { createApiHeader ( wildcard , currentTag ) }
2019-07-10 12:34:19 -05:00
$ { wildcard === "sdk" ? createSdkContent ( apiResponse ) : createApiContent ( apiResponse ) }
2018-11-30 14:46:22 -06:00
< / d i v >
< / s e c t i o n >
2019-07-10 16:48:45 -05:00
< script src = "/assets/scripts/plugins/jets.js" > < / s c r i p t >
< script src = "/assets/scripts/api.js" > < / s c r i p t >
2019-01-30 16:15:10 -06:00
2019-07-10 16:48:45 -05:00
< script >
initializeApiFunctionality ( ) ;
if ( window . location . pathname === "/api/blockchain" )
document . getElementById ( "toggle-cli" ) . click ( ) ;
else
document . getElementById ( "toggle-curl" ) . click ( ) ;
< / s c r i p t >
< / d i v >
2018-11-30 14:46:22 -06:00
` ;
2019-07-11 17:09:36 -05:00
} catch ( error ) {
2018-10-06 15:53:01 -05:00
const redirectUrl = redirects [ state . href ] ;
2018-08-28 18:57:18 -05:00
2018-10-06 15:53:01 -05:00
return asyncHtml `
2018-10-06 15:59:46 -05:00
< article class = "page" itemtype = "http://schema.org/BlogPosting" >
< header class = "page__header" >
< div class = "page__header-wrap" >
< div class = "inner-wrap" >
< h1 class = "page__header__title" itemprop = "name headline" > 404 < / h 1 >
< / d i v >
2018-10-01 15:47:10 -05:00
< / d i v >
2018-10-06 15:59:46 -05:00
< / h e a d e r >
2018-08-28 18:57:18 -05:00
2018-10-06 15:59:46 -05:00
< section class = "page__content page__markup" itemprop = "articleBody" >
< div class = "inner-wrap" >
< p > Redirecting you to < strong > $ { redirectUrl } < / s t r o n g > < / p >
< / d i v >
< / s e c t i o n >
< / a r t i c l e >
< script >
setTimeout ( ( ) => {
window . location . href = "${redirectUrl}" ;
} , 2000 ) ;
< / s c r i p t >
` ;
2018-11-30 14:46:22 -06:00
}
} ;
2018-09-30 13:34:29 -04:00
2018-08-28 18:57:18 -05:00
2018-10-01 15:47:10 -05:00
// H E L P E R S
2018-10-01 00:40:24 -04:00
2018-08-28 18:57:18 -05:00
function createApiContent ( apiDetails ) {
const apiContent = [ ] ;
2019-05-09 17:37:52 -05:00
apiDetails . forEach ( apiDetail => {
2018-08-28 18:57:18 -05:00
let apiDetailsReturns = "" ;
2018-10-06 15:53:01 -05:00
2018-11-30 14:46:22 -06:00
if ( apiDetail . returns )
apiDetailsReturns = JSON . parse ( JSON . stringify ( apiDetail . returns ) ) ;
2018-08-28 18:57:18 -05:00
apiContent . push ( `
2019-01-30 16:15:10 -06:00
< div class = "api-content__body" >
2018-08-28 18:57:18 -05:00
< h2 id = "${apiDetail.name}" > $ { apiDetail . name } < / h 2 >
< p > $ { apiDetail . description } < / p >
2019-01-30 16:15:10 -06:00
$ { apiDetail . arguments . length ? ` <h3>Arguments</h3><ul class="api-content__body-arguments"> ${ renderArguments ( apiDetail . arguments ) . join ( "" ) } </ul> ` : "" }
2019-05-09 17:37:52 -05:00
$ { apiDetail . returns ? ` <h3>Returns</h3><pre><code> ${ dedent ( apiDetailsReturns ) } </code></pre> ` : "" }
2018-08-28 18:57:18 -05:00
< / d i v >
2019-01-30 16:15:10 -06:00
< div class = "api-content__example" >
$ { apiDetail . examples && apiDetail . examples . length ? renderExamples ( apiDetail . examples ) . join ( "" ) : ` <pre><code>// example(s) for ${ apiDetail . name } to come later</code></pre> ` }
2018-08-28 18:57:18 -05:00
< / d i v >
` );
2019-05-09 17:37:52 -05:00
} ) ;
2018-08-28 18:57:18 -05:00
return apiContent ;
}
2019-07-10 16:48:45 -05:00
function createApiHeader ( slug , apiVersion ) {
2018-10-01 15:47:10 -05:00
switch ( slug ) {
case "blockchain" :
2019-07-10 16:48:45 -05:00
return headerBlockchain ( apiVersion ) ;
2018-10-01 15:47:10 -05:00
case "sdk" :
2019-07-10 16:48:45 -05:00
return headerSdk ( apiVersion ) ;
2018-10-01 15:47:10 -05:00
default :
break ;
}
}
2018-08-28 18:57:18 -05:00
function createApiSidebar ( apiDetails ) {
const apiSidebar = [ ] ;
2019-05-09 17:37:52 -05:00
apiDetails . forEach ( apiDetail => {
2018-08-28 18:57:18 -05:00
apiSidebar . push ( `
2019-04-12 15:08:35 -04:00
< li class = "api-toc__command" >
2018-08-28 18:57:18 -05:00
< a href = "#${apiDetail.name}" title = "Go to ${apiDetail.name} section" >
$ { apiDetail . name }
< / a >
< / l i >
` );
2019-05-09 17:37:52 -05:00
} ) ;
2018-08-28 18:57:18 -05:00
return apiSidebar ;
}
2019-04-12 15:08:35 -04:00
function createSdkContent ( apiDetails ) {
const apiContent = [ ] ;
const sectionTitles = Object . keys ( apiDetails ) ;
2019-05-09 17:37:52 -05:00
sectionTitles . forEach ( title => {
2019-04-12 15:08:35 -04:00
const commands = apiDetails [ title ] . commands ;
const description = apiDetails [ title ] . doc ;
apiContent . push (
commands . length ?
commands . map ( command => createSdkContentSections ( title , description , command ) ) . join ( "" ) :
""
) ;
2019-05-09 17:37:52 -05:00
} ) ;
2019-04-12 15:08:35 -04:00
return apiContent ;
}
function createSdkContentSections ( sectionTitle , sectionDescription , sectionDetails ) {
return `
< div class = "api-content__body" >
< h2 id = "${sectionDetails.name}" > $ { sectionDetails . name } < / h 2 >
< p > $ { sectionDetails . description } < / p >
< h3 > Arguments < / h 3 >
< ul class = "api-content__body-arguments" >
$ { renderArguments ( sectionDetails . arguments ) . join ( "" ) }
< / u l >
< h3 > Returns < / h 3 >
< pre > < code > $ { renderReturns ( sectionDetails . returns ) } < / c o d e > < / p r e >
< / d i v >
< div class = "api-content__example" >
$ { renderExamples ( sectionDetails . examples ) . join ( "" ) }
< / d i v >
` ;
}
function createSdkSidebar ( apiDetails ) {
const sectionTitles = Object . keys ( apiDetails ) ;
const apiSidebar = [ ] ;
2019-05-09 17:37:52 -05:00
sectionTitles . forEach ( title => {
2019-04-12 15:08:35 -04:00
const commands = apiDetails [ title ] . commands ;
apiSidebar . push ( `
< ul class = "api-toc__section" >
< li class = "api-toc__title" > $ { title } < / l i >
$ { ( commands . map ( command => ` <li class="api-toc__command"><a href="# ${ command . name } " title="Go to ${ command . name } section"> ${ command . name } </a></li> ` ) ) . join ( "" ) }
< / u l >
` );
2019-05-09 17:37:52 -05:00
} ) ;
2019-04-12 15:08:35 -04:00
return apiSidebar ;
}
2019-07-10 12:34:19 -05:00
async function getTags ( repositoryName ) {
const { data } = await octokit . repos . listTags ( {
owner : "lbryio" ,
repo : repositoryName
} ) ;
2019-10-24 16:39:26 -04:00
const tags = [ "master" ] ;
2019-07-10 12:34:19 -05:00
// NOTE:
// The versioning in our repos do not make sense so extra
// exclusion code is needed to make this work.
//
// Documentation is only available after specific versions.
2018-08-28 18:57:18 -05:00
2018-10-01 15:47:10 -05:00
switch ( true ) {
2019-07-10 12:34:19 -05:00
case repositoryName === "lbry-sdk" :
data . forEach ( tag => {
2019-12-31 19:20:37 -05:00
if ( tag . name >= "v0.52.0" ) tags . push ( tag . name ) ;
2019-07-10 12:34:19 -05:00
} ) ;
2018-10-01 15:47:10 -05:00
break ;
2018-08-28 18:57:18 -05:00
2019-07-10 12:34:19 -05:00
case repositoryName === "lbrycrd" :
data . forEach ( tag => {
if (
tag . name >= "v0.17.1.0" &&
tag . name !== "v0.3.16" &&
tag . name !== "v0.3.15" &&
tag . name !== "v0.3-osx" &&
tag . name !== "v0.2-alpha"
) tags . push ( tag . name ) ;
} ) ;
2018-10-01 15:47:10 -05:00
break ;
default :
break ;
}
2018-08-28 18:57:18 -05:00
2019-07-10 12:34:19 -05:00
return tags ;
}
async function parseApiFile ( { repo , tag } ) {
let apiFileLink = ` ${ rawGitHubBase } ${ repo } / ${ tag } ` ;
switch ( true ) {
case ( repo === "lbrycrd" ) :
apiFileLink = ` ${ apiFileLink } ${ filePathBlockchain } ` ;
break ;
case ( repo === "lbry-sdk" ) :
apiFileLink = ` ${ apiFileLink } ${ filePathSdk } ` ;
break ;
default :
return Promise . reject ( new Error ( "Failed to fetch API docs" ) ) ;
}
2019-01-08 11:25:34 -06:00
2019-04-16 14:52:51 -05:00
const response = await got ( apiFileLink , { cache , json : true } ) ;
2019-01-08 11:25:34 -06:00
try {
return response . body ;
} catch ( error ) {
return "Issue loading API documentation" ;
}
2018-08-28 18:57:18 -05:00
}
function renderArguments ( args ) {
const argumentContent = [ ] ;
2019-04-12 15:08:35 -04:00
if ( ! args || args . length === 0 )
return argumentContent ;
2019-05-09 17:37:52 -05:00
args . forEach ( arg => {
2018-08-28 18:57:18 -05:00
argumentContent . push ( `
2019-01-30 16:15:10 -06:00
< li class = "api-content__body-argument" >
2018-08-28 18:57:18 -05:00
< div class = "left" >
< strong > $ { arg . name } < /strong><br/ >
2019-07-11 17:09:36 -05:00
$ { arg . is _required === true ? "" : "<span>optional</span>" } < span > $ { arg . type } < / s p a n >
2018-08-28 18:57:18 -05:00
< / d i v >
< div class = "right" > $ { typeof arg . description === "string" ? arg . description . replace ( /</g , "<" ) . replace ( />/g , ">" ) : "" } < / d i v >
< / l i >
` );
2019-05-09 17:37:52 -05:00
} ) ;
2018-08-28 18:57:18 -05:00
return argumentContent ;
}
2019-01-30 16:15:10 -06:00
function renderExamples ( args ) {
const exampleContent = [ ] ;
2019-04-12 15:08:35 -04:00
if ( ! args || args . length === 0 ) {
exampleContent . push ( "<pre><code>// example(s) to come later</code></pre>" ) ;
return exampleContent ;
}
2019-05-09 17:37:52 -05:00
args . forEach ( arg => {
2019-01-30 16:15:10 -06:00
exampleContent . push ( `
2019-05-09 17:37:52 -05:00
$ { arg . title ? ` <h3> ${ arg . title } </h3><br/> ` : "" }
$ { arg . cli ? ` <pre data-api-example-type="cli"><code> ${ arg . cli } </code></pre> ` : "" }
$ { arg . curl ? ` <pre data-api-example-type="curl"><code> ${ arg . curl } </code></pre> ` : "" }
$ { arg . lbrynet ? ` <pre data-api-example-type="lbrynet"><code> ${ arg . lbrynet } </code></pre> ` : "" }
$ { arg . python ? ` <pre data-api-example-type="python"><code> ${ arg . python } </code></pre> ` : "" }
$ { arg . output ? `
< h3 > Output < /h3><br/ >
< pre > < code > $ { arg . output } < / c o d e > < / p r e >
< hr / >
` : ""}
2019-01-30 16:15:10 -06:00
` );
2019-05-09 17:37:52 -05:00
} ) ;
2019-01-30 16:15:10 -06:00
return exampleContent ;
}
2019-04-12 15:08:35 -04:00
function renderReturns ( args ) {
let returnContent = [ ] ;
if ( ! args || args . length === 0 )
return returnContent ;
returnContent = dedent ( JSON . parse ( JSON . stringify ( args ) ) ) ;
return returnContent ;
}
2019-05-09 17:37:52 -05:00
2019-07-10 16:48:45 -05:00
function renderVersionSelector ( pageSlug , versions , desiredTag ) {
2019-07-10 12:34:19 -05:00
const options = [
"<option disabled>Select a version</option>"
] ;
2019-07-10 16:48:45 -05:00
let optionIndex = 0 ;
2019-07-10 12:34:19 -05:00
versions . forEach ( version => {
2019-07-10 16:48:45 -05:00
optionIndex ++ ;
let selectedOption = false ;
if ( desiredTag && desiredTag === version )
selectedOption = true ;
else if ( optionIndex === 1 )
selectedOption = true ;
options . push (
` <option value=" ${ pageSlug } - ${ version } " ${ selectedOption ? " selected" : "" } > ${ version } </option> `
) ;
2019-07-10 12:34:19 -05:00
} ) ;
return options ;
}
function renderCodeLanguageToggles ( pageSlug ) {
const onSdkPage = pageSlug === "sdk" ;
2019-05-09 17:37:52 -05:00
return [
2019-05-14 14:40:35 -05:00
"<button class='api-content__item menu' id='toggle-menu'>menu</button>" ,
2019-05-09 17:37:52 -05:00
! onSdkPage ? "<button class='api-content__item' id='toggle-cli' type='button'>cli</button>" : "" ,
"<button class='api-content__item' id='toggle-curl' type='button'>curl</button>" ,
onSdkPage ? "<button class='api-content__item' id='toggle-lbrynet' type='button'>lbrynet</button>" : "" ,
onSdkPage ? "<button class='api-content__item' id='toggle-python' type='button'>python</button>" : ""
] ;
}