diff --git a/index.js b/index.js index 2539a036..0a9fca28 100644 --- a/index.js +++ b/index.js @@ -1,2 +1,4198 @@ -module.exports=function(e){function n(r){if(t[r])return t[r].exports;var a=t[r]={i:r,l:!1,exports:{}};return e[r].call(a.exports,a,a.exports,n),a.l=!0,a.exports}var t={};return n.m=e,n.c=t,n.d=function(e,t,r){n.o(e,t)||Object.defineProperty(e,t,{configurable:!1,enumerable:!0,get:r})},n.n=function(e){var t=e&&e.__esModule?function(){return e.default}:function(){return e};return n.d(t,"a",t),t},n.o=function(e,n){return Object.prototype.hasOwnProperty.call(e,n)},n.p="/",n(n.s=26)}([function(e,n){e.exports=require("winston")},function(e,n,t){"use strict";var r=function(){function e(e,n){var t=[],r=!0,a=!1,i=void 0;try{for(var o,l=e[Symbol.iterator]();!(r=(o=l.next()).done)&&(t.push(o.value),!n||t.length!==n);r=!0);}catch(e){a=!0,i=e}finally{try{!r&&l.return&&l.return()}finally{if(a)throw i}}return t}return function(n,t){if(Array.isArray(n))return n;if(Symbol.iterator in Object(n))return e(n,t);throw new TypeError("Invalid attempt to destructure non-iterable instance")}}(),a=t(0);e.exports={handleErrorResponse:function(n,t,i,o){a.error("Error on "+n,e.exports.useObjectPropertiesIfNoKeys(i));var l=e.exports.returnErrorMessageAndStatus(i),s=r(l,2),u=s[0],c=s[1];o.status(u).json(e.exports.createErrorResponsePayload(u,c))},returnErrorMessageAndStatus:function(e){var n=void 0,t=void 0;return"ECONNREFUSED"===e.code?(n=503,t="Connection refused. The daemon may not be running."):(n=400,t=e.message?e.message:e),[n,t]},useObjectPropertiesIfNoKeys:function(e){if(0===Object.keys(e).length){var n={};return Object.getOwnPropertyNames(e).forEach(function(t){n[t]=e[t]}),n}return e},createErrorResponsePayload:function(e,n){return{status:e,success:!1,message:n}}}},function(e,n,t){"use strict";var r=t(45),a=t(46),i=t(47),o=t(48),l=t(49),s=t(50),u=t(13),c=t(0),d=t(10),f=d.database,h=d.username,m=d.password,p=new u(f,h,m,{host:"localhost",dialect:"mysql",dialectOptions:{decimalNumbers:!0},logging:!1,pool:{max:5,min:0,idle:1e4,acquire:1e4}});p.authenticate().then(function(){c.info("Sequelize has established mysql connection successfully.")}).catch(function(e){c.error("Sequelize was unable to connect to the database:",e)});var g={};g.Certificate=p.import("Certificate",r),g.Channel=p.import("Channel",a),g.Claim=p.import("Claim",i),g.File=p.import("File",o),g.Request=p.import("Request",l),g.User=p.import("User",s),c.info("associating db models..."),Object.keys(g).forEach(function(e){g[e].associate&&(c.info("Associating model:",e),g[e].associate(g))}),g.sequelize=p,g.Sequelize=u,g.upsert=function(e,n,t,r){return e.findOne({where:t}).then(function(t){return t?(c.debug("updating record in db."+r),t.update(n)):(c.debug("creating record in db."+r),e.create(n))}).catch(function(e){throw c.error(r+".upsert error",e),e})},e.exports=g},function(e,n,t){"use strict";function r(){var e=this;this.analytics={googleId:"default"},this.assetDefaults={description:"An asset published on Spee.ch",thumbnail:"https://spee.ch/assets/img/video_thumb_default.png",title:"Spee.ch"},this.auth={sessionKey:"default"},this.customComponents={components:{},containers:{},pages:{}},this.details={description:"Open-source, decentralized image and video sharing.",host:"default",port:3e3,title:"Spee.ch",twitter:"@spee_ch"},this.publishing={additionalClaimAddresses:[],disabled:!1,disabledMessage:"Please check back soon.",primaryClaimAddress:"default",thumbnailChannel:"default",thumbnailChannelId:"default",uploadDirectory:"/home/lbry/Uploads"},this.routes={},this.update=function(n){if(!n)return console.log("No site config received.");var t=n.analytics,r=n.assetDefaults,a=n.auth,i=n.customComponents,o=n.details,l=n.publishing,s=n.routes;console.log("Configuring site details..."),e.analytics=t,e.assetDefaults=r,e.auth=a,e.details=o,e.publishing=l,e.customComponents=i,e.routes=s}}e.exports=new r},function(e,n,t){"use strict";function r(e,n,t){return{eventCategory:"client requests",eventAction:"serve request",eventLabel:t,ipOverride:n,userAgentOverride:e["user-agent"]}}function a(e,n,t,r,a){return{userTimingCategory:e,userTimingVariableName:n,userTimingTime:a-r,userTimingLabel:t}}function i(e,n){var t=e.replace(/\./g,"-");s(c,t,{strictCidFormat:!1,https:!0}).event(n,function(e){e&&l.error("Google Analytics Event Error >>",e)})}function o(e,n){s(c,e,{strictCidFormat:!1,https:!0}).timing(n,function(e){e&&l.error("Google Analytics Event Error >>",e),l.debug("Timing event successfully sent to google analytics")})}var l=t(0),s=t(55),u=t(3),c=u.analytics.googleId,d=u.details.title;e.exports={sendGAServeEvent:function(e,n,t){i(n,r(e,n,t))},sendGATimingEvent:function(e,n,t,r,i){var l=a(e,n,t,r,i);o(d,l)},chooseGaLbrynetPublishLabel:function(e){var n=e.channel_name,t=e.channel_id;return n||t?"PUBLISH_IN_CHANNEL_CLAIM":"PUBLISH_ANONYMOUS_CLAIM"}}},function(e,n,t){"use strict";var r=t(53),a=t(0),i=t(54),o=i.api,l=o.apiHost,s=o.apiPort,u="http://"+l+":"+s,c=t(4),d=c.chooseGaLbrynetPublishLabel,f=c.sendGATimingEvent,h=function(e,n,t){var r=e.data;if(a.debug("lbry api data:",r),r.result)return r.result.error?(a.debug("Lbrynet api error:",r.result.error),void t(new Error(r.result.error))):void n(r.result);t(JSON.stringify(r))};e.exports={publishClaim:function(e){a.debug('lbryApi >> Publishing claim to "'+e.name+'"');var n=Date.now();return new Promise(function(t,a){r.post(u,{method:"publish",params:e}).then(function(r){f("lbrynet","publish",d(e),n,Date.now()),h(r,t,a)}).catch(function(e){a(e)})})},getClaim:function(e){a.debug('lbryApi >> Getting Claim for "'+e+'"');var n=Date.now();return new Promise(function(t,a){r.post(u,{method:"get",params:{uri:e,timeout:20}}).then(function(e){f("lbrynet","getClaim","GET",n,Date.now()),h(e,t,a)}).catch(function(e){a(e)})})},getClaimList:function(e){a.debug('lbryApi >> Getting claim_list for "'+e+'"');var n=Date.now();return new Promise(function(t,a){r.post(u,{method:"claim_list",params:{name:e}}).then(function(e){f("lbrynet","getClaimList","CLAIM_LIST",n,Date.now()),h(e,t,a)}).catch(function(e){a(e)})})},resolveUri:function(e){a.debug('lbryApi >> Resolving URI for "'+e+'"');var n=Date.now();return new Promise(function(t,a){r.post(u,{method:"resolve",params:{uri:e}}).then(function(r){var i=r.data;f("lbrynet","resolveUri","RESOLVE",n,Date.now()),i.result[e].error?a(i.result[e].error):t(i.result[e])}).catch(function(e){a(e)})})},getDownloadDirectory:function(){a.debug("lbryApi >> Retrieving the download directory path from lbry daemon...");var e=Date.now();return new Promise(function(n,t){r.post(u,{method:"settings_get"}).then(function(t){var r=t.data;if(f("lbrynet","getDownloadDirectory","SETTINGS_GET",e,Date.now()),!r.result)return new Error("Successfully connected to lbry daemon, but unable to retrieve the download directory.");n(r.result.download_directory)}).catch(function(e){a.error("Lbrynet Error:",e),n("/home/lbry/Downloads/")})})},createChannel:function(e){a.debug("lbryApi >> Creating channel for "+e+"...");var n=Date.now();return new Promise(function(t,a){r.post(u,{method:"channel_new",params:{channel_name:e,amount:.1}}).then(function(e){f("lbrynet","createChannel","CHANNEL_NEW",n,Date.now()),h(e,t,a)}).catch(function(e){a(e)})})}}},function(e,n,t){"use strict";var r=function(){function e(e,n){var t=[],r=!0,a=!1,i=void 0;try{for(var o,l=e[Symbol.iterator]();!(r=(o=l.next()).done)&&(t.push(o.value),!n||t.length!==n);r=!0);}catch(e){a=!0,i=e}finally{try{!r&&l.return&&l.return()}finally{if(a)throw i}}return t}return function(n,t){if(Array.isArray(n))return n;if(Symbol.iterator in Object(n))return e(n,t);throw new TypeError("Invalid attempt to destructure non-iterable instance")}}(),a=t(2),i=t(0),o=t(66),l=o.returnPaginatedChannelClaims;e.exports={getClaimId:function(n,t,r,a){return n?e.exports.getClaimIdByChannel(n,t,r):e.exports.getClaimIdByClaim(r,a)},getClaimIdByClaim:function(e,n){return i.debug("getClaimIdByClaim("+e+", "+n+")"),new Promise(function(t,r){a.Claim.getLongClaimId(e,n).then(function(e){e||t("NO_CLAIM"),t(e)}).catch(function(e){r(e)})})},getClaimIdByChannel:function(e,n,t){return i.debug("getClaimIdByChannel("+e+", "+n+", "+t+")"),new Promise(function(i,o){a.Certificate.getLongChannelId(e,n).then(function(e){return e?Promise.all([e,a.Claim.getClaimIdByLongChannelId(e,t)]):[null,null]}).then(function(e){var n=r(e,2),t=n[0],a=n[1];return t?a?void i(a):i("NO_CLAIM"):i("NO_CHANNEL")}).catch(function(e){o(e)})})},getChannelData:function(e,n,t){return new Promise(function(t,i){a.Certificate.getLongChannelId(e,n).then(function(n){return n?Promise.all([n,a.Certificate.getShortChannelIdFromLongChannelId(n,e)]):[null,null,null]}).then(function(n){var a=r(n,2),i=a[0],o=a[1];if(!i)return t("NO_CHANNEL");t({channelName:e,longChannelClaimId:i,shortChannelClaimId:o})}).catch(function(e){i(e)})})},getChannelClaims:function(e,n,t){return new Promise(function(i,o){a.Certificate.getLongChannelId(e,n).then(function(e){return e?Promise.all([e,a.Claim.getAllChannelClaims(e)]):[null,null,null]}).then(function(n){var a=r(n,2),o=a[0],s=a[1];if(!o)return i("NO_CHANNEL");var u=l(e,o,s,t);i(u)}).catch(function(e){o(e)})})},getLocalFileRecord:function(e,n){return a.File.findOne({where:{claimId:e,name:n}}).then(function(e){return e?e.dataValues:"NO_FILE"})}}},function(e,n,t){"use strict";var r=t(43),a=t(44),i=t(52),o=t(56),l=o.serializeSpeechUser,s=o.deserializeSpeechUser;r.deserializeUser(s),r.serializeUser(l),r.use("local-login",a),r.use("local-signup",i),e.exports=r},function(e,n,t){"use strict";function r(e,n,t){return n in e?Object.defineProperty(e,n,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[n]=t,e}var a=function(){function e(e,n){var t=[],r=!0,a=!1,i=void 0;try{for(var o,l=e[Symbol.iterator]();!(r=(o=l.next()).done)&&(t.push(o.value),!n||t.length!==n);r=!0);}catch(e){a=!0,i=e}finally{try{!r&&l.return&&l.return()}finally{if(a)throw i}}return t}return function(n,t){if(Array.isArray(n))return n;if(Symbol.iterator in Object(n))return e(n,t);throw new TypeError("Invalid attempt to destructure non-iterable instance")}}(),i=t(0),o=t(2),l=t(5),s=t(9),u=t(3),c=u.publishing,d=c.primaryClaimAddress,f=c.additionalClaimAddresses,h=t(13),m=h.Op;e.exports={publish:function(e,n,t){return new Promise(function(r,u){var c=void 0,d=void 0,f=void 0;return l.publishClaim(e).then(function(t){return i.info("Successfully published "+e.name+" "+n,t),c=t,e.channel_name?(i.debug("this claim was published in channel: "+e.channel_name),o.Channel.findOne({where:{channelName:e.channel_name}})):(i.debug("this claim was not published in a channel"),null)}).then(function(e){d=null,f=null,e&&(d=e.channelClaimId,f=e.channelName),i.debug("certificateId: "+d)}).then(function(){var r={name:e.name,claimId:c.claim_id,title:e.metadata.title,description:e.metadata.description,address:e.claim_address,outpoint:c.txid+":"+c.nout,height:0,fileName:n,filePath:e.file_path,fileType:t,nsfw:e.metadata.nsfw},a={name:e.name,claimId:c.claim_id,title:e.metadata.title,description:e.metadata.description,address:e.claim_address,thumbnail:e.metadata.thumbnail,outpoint:c.txid+":"+c.nout,height:0,contentType:t,nsfw:e.metadata.nsfw,amount:e.bid,certificateId:d,channelName:f},i={name:e.name,claimId:c.claim_id};return Promise.all([o.upsert(o.File,r,i,"File"),o.upsert(o.Claim,a,i,"Claim")])}).then(function(e){var n=a(e,2),t=n[0],r=n[1];return i.debug("File and Claim records successfully created"),Promise.all([t.setClaim(r),r.setFile(t)])}).then(function(){i.debug("File and Claim records successfully associated"),r(c)}).catch(function(n){i.error("PUBLISH ERROR",n),s.deleteTemporaryFile(e.file_path),u(n)})})},claimNameIsAvailable:function(e){var n=f||[];return n.push(d),o.Claim.findAll({attributes:["address"],where:{name:e,address:r({},m.or,n)}}).then(function(n){if(n.length>=1)throw new Error("That claim is already in use");return e}).catch(function(e){throw e})},checkChannelAvailability:function(e){return o.Channel.findAll({where:{channelName:e}}).then(function(n){if(n.length>=1)throw new Error("That channel has already been claimed");return e}).catch(function(e){throw e})}}},function(e,n,t){"use strict";var r=t(0),a=t(64),i=t(3),o=i.details,l=i.publishing;e.exports={parsePublishApiRequestBody:function(e){var n=e.name,t=e.nsfw,r=e.license,a=e.title,i=e.description,o=e.thumbnail;if(!n)throw new Error("no name field found in request");if(/[^A-Za-z0-9,-]/.exec(n))throw new Error('The claim name you provided is not allowed. Only the following characters are allowed: A-Z, a-z, 0-9, and "-"');return t="true"===t,r=r||null,a=a||null,i=i||null,o=o||null,{name:n,nsfw:t,license:r,title:a,description:i,thumbnail:o}},parsePublishApiRequestFiles:function(n){var t=n.file,r=n.thumbnail;if(!t)throw new Error("no file with key of [file] found in request");if(!t.path)throw new Error("no file path found");if(!t.type)throw new Error("no file type found");if(!t.size)throw new Error("no file type found");if(/'/.test(t.name))throw new Error("apostrophes are not allowed in the file name");return e.exports.validateFileTypeAndSize(t),{fileName:t.name,filePath:t.path,fileType:t.type,thumbnailFileName:r?r.name:null,thumbnailFilePath:r?r.path:null,thumbnailFileType:r?r.type:null}},validateFileTypeAndSize:function(e){switch(e.type){case"image/jpeg":case"image/jpg":case"image/png":if(e.size>1e7)throw r.debug("publish > file validation > .jpeg/.jpg/.png was too big"),new Error("Sorry, images are limited to 10 megabytes.");break;case"image/gif":if(e.size>5e7)throw r.debug("publish > file validation > .gif was too big"),new Error("Sorry, .gifs are limited to 50 megabytes.");break;case"video/mp4":if(e.size>5e7)throw r.debug("publish > file validation > .mp4 was too big"),new Error("Sorry, videos are limited to 50 megabytes.");break;default:throw r.debug("publish > file validation > unrecognized file type"),new Error("The "+e.type+" content type is not supported. Only, .jpeg, .png, .gif, and .mp4 files are currently supported.")}return e},createBasicPublishParams:function(e,n,t,a,i,s,u){r.debug("Creating Publish Parameters"),null!==t&&""!==t.trim()||(t=n),null!==a&&""!==a.trim()||(a=""),null!==i&&""!==i.trim()||(i=" ");var c={name:n,file_path:e,bid:.01,metadata:{description:a,title:t,author:o.title,language:"en",license:i,nsfw:s},claim_address:l.primaryClaimAddress};return u&&(c.metadata.thumbnail=u),c},createThumbnailPublishParams:function(e,n,t,a){if(e)return r.debug("Creating Thumbnail Publish Parameters"),{name:n+"-thumb",file_path:e,bid:.01,metadata:{title:n+" thumbnail",description:"a thumbnail for "+n,author:o.title,language:"en",license:t,nsfw:a},claim_address:l.primaryClaimAddress,channel_name:l.thumbnailChannel,channel_id:l.thumbnailChannelId}},deleteTemporaryFile:function(e){a.unlink(e,function(n){if(n)throw r.error("error deleting temporary file "+e),n;r.debug("successfully deleted "+e)})},addGetResultsToFileData:function(e,n){return e.fileName=n.file_name,e.filePath=n.download_path,e},createFileData:function(e){var n=e.name,t=e.claimId,r=e.outpoint,a=e.height,i=e.address,o=e.nsfw;return{name:n,claimId:t,outpoint:r,height:a,address:i,fileName:"",filePath:"",fileType:e.contentType,nsfw:o}}}},function(e,n,t){"use strict";function r(){var e=this;this.database="default",this.username="default",this.password="default",this.update=function(n){if(!n)return a.warn("No MySQL config received.");a.info("configuring mysql...");var t=n.database,r=n.username,i=n.password;e.database=t,e.username=r,e.password=i}}var a=t(0);e.exports=new r},function(e,n){e.exports=require("passport-local")},function(e,n,t){"use strict";e.exports={returnShortId:function(e,n){var t=void 0,r=n.substring(0,1),a=0;if((t=e.findIndex(function(e){return e.claimId===n}))<0)throw new Error("claim id not found in claims list");for(var i=e.slice(0,t);i.length>0;)a+=1,r=n.substring(0,a),i=i.filter(function(e){return e.claimId&&e.claimId.substring(0,a)===r});return r}}},function(e,n){e.exports=require("sequelize")},function(e,n,t){"use strict";function r(e){return e&&e.__esModule?e:{default:e}}var a=t(15),i=r(a),o=t(16),l=t(17),s=t(18),u=t(19),c=t(20),d=t(21),f=r(d),h=t(22),m=r(h);e.exports=function(e,n){var t={},r=(0,l.createStore)(c.Reducers),a=(0,o.renderToString)(i.default.createElement(s.Provider,{store:r},i.default.createElement(u.StaticRouter,{location:e.url,context:t},i.default.createElement(c.GAListener,null,i.default.createElement(c.App,null))))),d=m.default.renderStatic();if(t.url)return n.redirect(301,t.url);var h=r.getState();n.send((0,f.default)(d,a,h))}},function(e,n){e.exports=require("react")},function(e,n){e.exports=require("react-dom/server")},function(e,n){e.exports=require("redux")},function(e,n){e.exports=require("react-redux")},function(e,n){e.exports=require("react-router-dom")},function(e,n){e.exports=require("spee.ch-components")},function(e,n,t){"use strict";e.exports=function(e,n,t){return'\n \n \n \n \n \n \n \x3c!--helmet--\x3e\n '+e.title.toString()+"\n "+e.meta.toString()+"\n "+e.link.toString()+'\n \x3c!--style sheets--\x3e\n \n \n \n \x3c!--google font--\x3e\n \n \n \n
\n
'+n+"
\n
\n \n \n \n \n '; +}; + +/***/ }), +/* 22 */ +/***/ (function(module, exports) { + +module.exports = require("react-helmet"); + +/***/ }), +/* 23 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var logger = __webpack_require__(0); + +var _require = __webpack_require__(6), + getClaimId = _require.getClaimId, + getLocalFileRecord = _require.getLocalFileRecord; + +var _require2 = __webpack_require__(1), + handleErrorResponse = _require2.handleErrorResponse; + +var SERVE = 'SERVE'; +var SHOW = 'SHOW'; +var NO_FILE = 'NO_FILE'; +var NO_CHANNEL = 'NO_CHANNEL'; +var NO_CLAIM = 'NO_CLAIM'; + +function clientAcceptsHtml(_ref) { + var accept = _ref.accept; + + return accept && accept.match(/text\/html/); +}; + +function requestIsFromBrowser(headers) { + return headers['user-agent'] && headers['user-agent'].match(/Mozilla/); +}; + +function clientWantsAsset(_ref2) { + var accept = _ref2.accept, + range = _ref2.range; + + var imageIsWanted = accept && accept.match(/image\/.*/) && !accept.match(/text\/html/) && !accept.match(/text\/\*/); + var videoIsWanted = accept && range; + return imageIsWanted || videoIsWanted; +}; + +function isValidClaimId(claimId) { + return claimId.length === 40 && !/[^A-Za-z0-9]/g.test(claimId); +}; + +function isValidShortId(claimId) { + return claimId.length === 1; // it should really evaluate the short url itself +}; + +function isValidShortIdOrClaimId(input) { + return isValidClaimId(input) || isValidShortId(input); +}; + +function serveAssetToClient(claimId, name, res) { + return getLocalFileRecord(claimId, name).then(function (fileRecord) { + // check that a local record was found + if (fileRecord === NO_FILE) { + return res.status(307).redirect('/api/claim/get/' + name + '/' + claimId); + } + // serve the file + var filePath = fileRecord.filePath, + fileType = fileRecord.fileType; + + logger.verbose('serving file: ' + filePath); + var sendFileOptions = { + headers: { + 'X-Content-Type-Options': 'nosniff', + 'Content-Type': fileType || 'image/jpeg' + } + }; + res.status(200).sendFile(filePath, sendFileOptions); + }).catch(function (error) { + throw error; + }); +}; + +module.exports = { + getClaimIdAndServeAsset: function getClaimIdAndServeAsset(channelName, channelClaimId, claimName, claimId, originalUrl, ip, res) { + // get the claim Id and then serve the asset + getClaimId(channelName, channelClaimId, claimName, claimId).then(function (fullClaimId) { + if (fullClaimId === NO_CLAIM) { + return res.status(404).json({ success: false, message: 'no claim id could be found' }); + } else if (fullClaimId === NO_CHANNEL) { + return res.status(404).json({ success: false, message: 'no channel id could be found' }); + } + serveAssetToClient(fullClaimId, claimName, res); + // postToStats(responseType, originalUrl, ip, claimName, fullClaimId, 'success'); + }).catch(function (error) { + handleErrorResponse(originalUrl, ip, error, res); + // postToStats(responseType, originalUrl, ip, claimName, fullClaimId, 'fail'); + }); + }, + determineResponseType: function determineResponseType(hasFileExtension, headers) { + var responseType = void 0; + if (hasFileExtension) { + responseType = SERVE; // assume a serve request if file extension is present + if (clientAcceptsHtml(headers)) { + // if the request comes from a browser, change it to a show request + responseType = SHOW; + } + } else { + responseType = SHOW; + if (clientWantsAsset(headers) && requestIsFromBrowser(headers)) { + // this is in case someone embeds a show url + logger.debug('Show request came from browser but wants an image/video. Changing response to serve...'); + responseType = SERVE; + } + } + return responseType; + }, + flipClaimNameAndIdForBackwardsCompatibility: function flipClaimNameAndIdForBackwardsCompatibility(identifier, name) { + // this is a patch for backwards compatability with '/name/claim_id' url format + if (isValidShortIdOrClaimId(name) && !isValidShortIdOrClaimId(identifier)) { + var tempName = name; + name = identifier; + identifier = tempName; + } + return [identifier, name]; + }, + logRequestData: function logRequestData(responseType, claimName, channelName, claimId) { + logger.debug('responseType ===', responseType); + logger.debug('claim name === ', claimName); + logger.debug('channel name ===', channelName); + logger.debug('claim id ===', claimId); + } +}; + +/***/ }), +/* 24 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"]) _i["return"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError("Invalid attempt to destructure non-iterable instance"); } }; }(); + +var logger = __webpack_require__(0); + +module.exports = { + REGEXP_INVALID_CLAIM: /[^A-Za-z0-9-]/g, + REGEXP_INVALID_CHANNEL: /[^A-Za-z0-9-@]/g, + REGEXP_ADDRESS: /^b(?=[^0OIl]{32,33})[0-9A-Za-z]{32,33}$/, + CHANNEL_CHAR: '@', + parseIdentifier: function parseIdentifier(identifier) { + logger.debug('parsing identifier:', identifier); + var componentsRegex = new RegExp('([^:$#/]*)' + // value (stops at the first separator or end) + '([:$#]?)([^/]*)' // modifier separator, modifier (stops at the first path separator or end) + ); + + var _componentsRegex$exec = componentsRegex.exec(identifier).map(function (match) { + return match || null; + }), + _componentsRegex$exec2 = _slicedToArray(_componentsRegex$exec, 4), + proto = _componentsRegex$exec2[0], + value = _componentsRegex$exec2[1], + modifierSeperator = _componentsRegex$exec2[2], + modifier = _componentsRegex$exec2[3]; + + logger.debug(proto + ', ' + value + ', ' + modifierSeperator + ', ' + modifier); + + // Validate and process name + if (!value) { + throw new Error('Check your url. No channel name provided before "' + modifierSeperator + '"'); + } + var isChannel = value.startsWith(module.exports.CHANNEL_CHAR); + var channelName = isChannel ? value : null; + var claimId = void 0; + if (isChannel) { + if (!channelName) { + throw new Error('No channel name after @.'); + } + var nameBadChars = channelName.match(module.exports.REGEXP_INVALID_CHANNEL); + if (nameBadChars) { + throw new Error('Invalid characters in channel name: ' + nameBadChars.join(', ') + '.'); + } + } else { + claimId = value; + } + + // Validate and process modifier + var channelClaimId = void 0; + if (modifierSeperator) { + if (!modifier) { + throw new Error('No modifier provided after separator "' + modifierSeperator + '"'); + } + + if (modifierSeperator === ':') { + channelClaimId = modifier; + } else { + throw new Error('The "' + modifierSeperator + '" modifier is not currently supported'); + } + } + return { + isChannel: isChannel, + channelName: channelName, + channelClaimId: channelClaimId, + claimId: claimId + }; + }, + parseClaim: function parseClaim(claim) { + logger.debug('parsing name:', claim); + var componentsRegex = new RegExp('([^:$#/.]*)' + // name (stops at the first modifier) + '([:$#.]?)([^/]*)' // modifier separator, modifier (stops at the first path separator or end) + ); + + var _componentsRegex$exec3 = componentsRegex.exec(claim).map(function (match) { + return match || null; + }), + _componentsRegex$exec4 = _slicedToArray(_componentsRegex$exec3, 4), + proto = _componentsRegex$exec4[0], + claimName = _componentsRegex$exec4[1], + modifierSeperator = _componentsRegex$exec4[2], + modifier = _componentsRegex$exec4[3]; + + logger.debug(proto + ', ' + claimName + ', ' + modifierSeperator + ', ' + modifier); + + // Validate and process name + if (!claimName) { + throw new Error('No claim name provided before .'); + } + var nameBadChars = claimName.match(module.exports.REGEXP_INVALID_CLAIM); + if (nameBadChars) { + throw new Error('Invalid characters in claim name: ' + nameBadChars.join(', ') + '.'); + } + // Validate and process modifier + if (modifierSeperator) { + if (!modifier) { + throw new Error('No file extension provided after separator ' + modifierSeperator + '.'); + } + if (modifierSeperator !== '.') { + throw new Error('The ' + modifierSeperator + ' modifier is not supported in the claim name'); + } + } + // return results + return { + claimName: claimName + }; + }, + parseModifier: function parseModifier(claim) { + logger.debug('parsing modifier:', claim); + var componentsRegex = new RegExp('([^:$#/.]*)' + // name (stops at the first modifier) + '([:$#.]?)([^/]*)' // modifier separator, modifier (stops at the first path separator or end) + ); + + var _componentsRegex$exec5 = componentsRegex.exec(claim).map(function (match) { + return match || null; + }), + _componentsRegex$exec6 = _slicedToArray(_componentsRegex$exec5, 4), + proto = _componentsRegex$exec6[0], + claimName = _componentsRegex$exec6[1], + modifierSeperator = _componentsRegex$exec6[2], + modifier = _componentsRegex$exec6[3]; + + logger.debug(proto + ', ' + claimName + ', ' + modifierSeperator + ', ' + modifier); + // Validate and process modifier + var hasFileExtension = false; + if (modifierSeperator) { + hasFileExtension = true; + } + return { + hasFileExtension: hasFileExtension + }; + } +}; + +/***/ }), +/* 25 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var _react = __webpack_require__(15); + +var _react2 = _interopRequireDefault(_react); + +var _server = __webpack_require__(16); + +var _redux = __webpack_require__(17); + +var _reactRedux = __webpack_require__(18); + +var _reactRouterDom = __webpack_require__(19); + +var _renderFullPage = __webpack_require__(21); + +var _renderFullPage2 = _interopRequireDefault(_renderFullPage); + +var _reduxSaga = __webpack_require__(87); + +var _reduxSaga2 = _interopRequireDefault(_reduxSaga); + +var _effects = __webpack_require__(88); + +var _spee = __webpack_require__(20); + +var _reactHelmet = __webpack_require__(22); + +var _reactHelmet2 = _interopRequireDefault(_reactHelmet); + +function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } + +// configure the reducers by passing initial state configs +var siteConfig = __webpack_require__(3); +var customizedReducers = (0, _spee.Reducers)(siteConfig); + +var returnSagaWithParams = function returnSagaWithParams(saga, params) { + return (/*#__PURE__*/regeneratorRuntime.mark(function _callee() { + return regeneratorRuntime.wrap(function _callee$(_context) { + while (1) { + switch (_context.prev = _context.next) { + case 0: + _context.next = 2; + return (0, _effects.call)(saga, params); + + case 2: + case 'end': + return _context.stop(); + } + } + }, _callee, this); + }) + ); +}; + +module.exports = function (req, res) { + var context = {}; + + // create and apply middleware + var sagaMiddleware = (0, _reduxSaga2.default)(); + var middleware = (0, _redux.applyMiddleware)(sagaMiddleware); + + // create a new Redux store instance + var store = (0, _redux.createStore)(customizedReducers, middleware); + + // create saga + var action = _spee.Actions.onHandleShowPageUri(req.params); + var saga = returnSagaWithParams(_spee.Sagas.handleShowPageUri, action); + + // run the saga middleware + sagaMiddleware.run(saga).done.then(function () { + // render component to a string + var html = (0, _server.renderToString)(_react2.default.createElement( + _reactRedux.Provider, + { store: store }, + _react2.default.createElement( + _reactRouterDom.StaticRouter, + { location: req.url, context: context }, + _react2.default.createElement( + _spee.GAListener, + null, + _react2.default.createElement(_spee.App, null) + ) + ) + )); + + // get head tags from helmet + var helmet = _reactHelmet2.default.renderStatic(); + + // check for a redirect + if (context.url) { + return res.redirect(301, context.url); + } + + // get the initial state from our Redux store + var preloadedState = store.getState(); + + // send the rendered page back to the client + res.send((0, _renderFullPage2.default)(helmet, html, preloadedState)); + }); + + console.log('hello from spee.ch handleShowRender.jsx'); +}; + +/***/ }), +/* 26 */ +/***/ (function(module, exports, __webpack_require__) { + +__webpack_require__(27); +__webpack_require__(28); +module.exports = __webpack_require__(29); + + +/***/ }), +/* 27 */ +/***/ (function(module, exports) { + +module.exports = require("babel-polyfill"); + +/***/ }), +/* 28 */ +/***/ (function(module, exports) { + +module.exports = require("whatwg-fetch"); + +/***/ }), +/* 29 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var Server = __webpack_require__(30); + +var _exports = { + Server: Server +}; + +module.exports = _exports; + +/***/ }), +/* 30 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +// app dependencies +var express = __webpack_require__(31); +var bodyParser = __webpack_require__(32); +var expressHandlebars = __webpack_require__(33); +var Handlebars = __webpack_require__(34); +var helmet = __webpack_require__(35); +var cookieSession = __webpack_require__(36); +var http = __webpack_require__(37); +var logger = __webpack_require__(0); +var requestLogger = __webpack_require__(38); +var Path = __webpack_require__(39); +var loggerConfig = __webpack_require__(40); +var mysqlConfig = __webpack_require__(10); +var siteConfig = __webpack_require__(3); +var slackConfig = __webpack_require__(41); + +function Server() { + var _this = this; + + this.configureLogger = function (userConfig) { + loggerConfig.update(userConfig); + }; + this.configureMysql = function (userConfig) { + mysqlConfig.update(userConfig); + }; + this.configureSiteDetails = function (userConfig) { + siteConfig.update(userConfig); + }; + this.configureSlack = function (userConfig) { + slackConfig.update(userConfig); + }; + this.configureClientBundle = function () { + logger.debug('configure the client here by passing in the bundle and configuring it, or better yet: taking in the components to use dynamically from here.'); + }; + this.configureModels = function () { + logger.debug('here is where you could add/overwrite the default models'); + }; + this.configureRoutes = function () { + logger.debug('here is where you could add/overwrite the default routes'); + }; + this.createApp = function () { + // create an Express application + var app = express(); + + // trust the proxy to get ip address for us + app.enable('trust proxy'); + + /* add middleware */ + // set HTTP headers to protect against well-known web vulnerabilties + app.use(helmet()); + // 'express.static' to serve static files from public directory + if (siteConfig.routes.publicFolder) { + // take in a different public folder, so it can serve it's own bundle if needed + var publicFolder = Path.resolve(process.cwd(), siteConfig.routes.publicFolder); + app.use('/static', express.static(publicFolder)); + logger.info('serving static files from custom path:', publicFolder); + } else { + var publicPath = Path.resolve(__dirname, 'public'); + app.use('/static', express.static(publicPath)); + logger.info('serving static files from default path:', publicPath); + }; + // 'body parser' for parsing application/json + app.use(bodyParser.json()); + // 'body parser' for parsing application/x-www-form-urlencoded + app.use(bodyParser.urlencoded({ extended: true })); + + // add custom middleware (note: build out to accept dynamically use what is in server/middleware/ + app.use(requestLogger); + + // configure passport + var speechPassport = __webpack_require__(7); + // initialize passport + var sessionKey = siteConfig.auth.sessionKey; + app.use(cookieSession({ + name: 'session', + keys: [sessionKey], + maxAge: 24 * 60 * 60 * 1000 // i.e. 24 hours + })); + app.use(speechPassport.initialize()); + app.use(speechPassport.session()); + + // configure handlebars & register it with express app + var hbs = expressHandlebars.create({ + defaultLayout: 'embed', + handlebars: Handlebars + }); + app.engine('handlebars', hbs.engine); + app.set('view engine', 'handlebars'); + + // set the routes on the app + __webpack_require__(57)(app); + __webpack_require__(62)(app); + __webpack_require__(81)(app); + __webpack_require__(85)(app); + __webpack_require__(90)(app); + + _this.app = app; + }; + this.initialize = function () { + _this.createApp(); + _this.server = http.Server(_this.app); + }; + this.start = function () { + var db = __webpack_require__(2); + var PORT = siteConfig.details.port; + // sync sequelize + db.sequelize.sync() + // start the server + .then(function () { + _this.server.listen(PORT, function () { + logger.info('Server is listening on PORT ' + PORT); + }); + }).catch(function (error) { + logger.error('Startup Error:', error); + }); + }; +}; + +module.exports = Server; + +/***/ }), +/* 31 */ +/***/ (function(module, exports) { + +module.exports = require("express"); + +/***/ }), +/* 32 */ +/***/ (function(module, exports) { + +module.exports = require("body-parser"); + +/***/ }), +/* 33 */ +/***/ (function(module, exports) { + +module.exports = require("express-handlebars"); + +/***/ }), +/* 34 */ +/***/ (function(module, exports) { + +module.exports = require("handlebars"); + +/***/ }), +/* 35 */ +/***/ (function(module, exports) { + +module.exports = require("helmet"); + +/***/ }), +/* 36 */ +/***/ (function(module, exports) { + +module.exports = require("cookie-session"); + +/***/ }), +/* 37 */ +/***/ (function(module, exports) { + +module.exports = require("http"); + +/***/ }), +/* 38 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var logger = __webpack_require__(0); + +var requestLogger = function requestLogger(req, res, next) { + // custom logging middleware to log all incoming http requests + logger.verbose('Request on ' + req.originalUrl + ' from ' + req.ip); + next(); +}; + +module.exports = requestLogger; + +/***/ }), +/* 39 */ +/***/ (function(module, exports) { + +module.exports = require("path"); + +/***/ }), +/* 40 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var logger = __webpack_require__(0); + +function LoggerConfig() { + var _this = this; + + this.logLevel = 'debug'; + this.update = function (config) { + if (!config) { + return logger.warn('No logger config received.'); + } + logger.info('configuring winston logger...'); + // update values with local config params + var logLevel = config.logLevel; + + _this.logLevel = logLevel; + // configure the winston logger + logger.configure({ + transports: [new logger.transports.Console({ + level: _this.logLevel, + timestamp: false, + colorize: true, + prettyPrint: true, + handleExceptions: true, + humanReadableUnhandledException: true + })] + }); + // test all the log levels + logger.info('testing winston log levels...'); + logger.error('Level 0'); + logger.warn('Level 1'); + logger.info('Level 2'); + logger.verbose('Level 3'); + logger.debug('Level 4'); + logger.silly('Level 5'); + }; +}; + +module.exports = new LoggerConfig(); + +/***/ }), +/* 41 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var winstonSlackWebHook = __webpack_require__(42).SlackWebHook; +var winston = __webpack_require__(0); + +function SlackConfig() { + var _this = this; + + this.slackWebHook = 'default'; + this.slackErrorChannel = 'default'; + this.slackInfoChannel = 'default'; + this.update = function (config) { + if (!config) { + return winston.warn('No slack config received'); + } + // update variables + winston.info('configuring slack logger...'); + var slackWebHook = config.slackWebHook, + slackErrorChannel = config.slackErrorChannel, + slackInfoChannel = config.slackInfoChannel; + + _this.slackWebHook = slackWebHook; + _this.slackErrorChannel = slackErrorChannel; + _this.slackInfoChannel = slackInfoChannel; + // update slack webhook settings + if (_this.slackWebHook) { + // add a transport for errors to slack + if (_this.slackErrorChannel) { + winston.add(winstonSlackWebHook, { + name: 'slack-errors-transport', + level: 'warn', + webhookUrl: _this.slackWebHook, + channel: _this.slackErrorChannel, + username: 'spee.ch', + iconEmoji: ':face_with_head_bandage:' + }); + }; + if (slackInfoChannel) { + winston.add(winstonSlackWebHook, { + name: 'slack-info-transport', + level: 'info', + webhookUrl: _this.slackWebHook, + channel: _this.slackInfoChannel, + username: 'spee.ch', + iconEmoji: ':nerd_face:' + }); + }; + // send test messages + winston.info('testing slack logger...'); + winston.error('Slack "error" logging is online.'); + winston.info('Slack "info" logging is online.'); + } else { + winston.warn('Slack logging is not enabled because no slackWebHook config var provided.'); + } + }; +}; + +module.exports = new SlackConfig(); + +/***/ }), +/* 42 */ +/***/ (function(module, exports) { + +module.exports = require("winston-slack-webhook"); + +/***/ }), +/* 43 */ +/***/ (function(module, exports) { + +module.exports = require("passport"); + +/***/ }), +/* 44 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var PassportLocalStrategy = __webpack_require__(11).Strategy; +var logger = __webpack_require__(0); +var db = __webpack_require__(2); + +var returnUserAndChannelInfo = function returnUserAndChannelInfo(userInstance) { + return new Promise(function (resolve, reject) { + var userInfo = {}; + userInfo['id'] = userInstance.id; + userInfo['userName'] = userInstance.userName; + userInstance.getChannel().then(function (_ref) { + var channelName = _ref.channelName, + channelClaimId = _ref.channelClaimId; + + userInfo['channelName'] = channelName; + userInfo['channelClaimId'] = channelClaimId; + return db.Certificate.getShortChannelIdFromLongChannelId(channelClaimId, channelName); + }).then(function (shortChannelId) { + userInfo['shortChannelId'] = shortChannelId; + resolve(userInfo); + }).catch(function (error) { + reject(error); + }); + }); +}; + +module.exports = new PassportLocalStrategy({ + usernameField: 'username', + passwordField: 'password' +}, function (username, password, done) { + return db.User.findOne({ + where: { userName: username } + }).then(function (user) { + if (!user) { + logger.debug('no user found'); + return done(null, false, { message: 'Incorrect username or password' }); + } + return user.comparePassword(password).then(function (isMatch) { + if (!isMatch) { + logger.debug('incorrect password'); + return done(null, false, { message: 'Incorrect username or password' }); + } + logger.debug('Password was a match, returning User'); + return returnUserAndChannelInfo(user).then(function (userInfo) { + return done(null, userInfo); + }).catch(function (error) { + return error; + }); + }).catch(function (error) { + return error; + }); + }).catch(function (error) { + return done(error); + }); +}); + +/***/ }), +/* 45 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var logger = __webpack_require__(0); + +var _require = __webpack_require__(12), + returnShortId = _require.returnShortId; + +module.exports = function (sequelize, _ref) { + var STRING = _ref.STRING, + BOOLEAN = _ref.BOOLEAN, + INTEGER = _ref.INTEGER, + TEXT = _ref.TEXT, + DECIMAL = _ref.DECIMAL; + + var Certificate = sequelize.define('Certificate', { + address: { + type: STRING, + default: null + }, + amount: { + type: DECIMAL(19, 8), + default: null + }, + claimId: { + type: STRING, + default: null + }, + claimSequence: { + type: INTEGER, + default: null + }, + decodedClaim: { + type: BOOLEAN, + default: null + }, + depth: { + type: INTEGER, + default: null + }, + effectiveAmount: { + type: DECIMAL(19, 8), + default: null + }, + hasSignature: { + type: BOOLEAN, + default: null + }, + height: { + type: INTEGER, + default: null + }, + hex: { + type: TEXT('long'), + default: null + }, + name: { + type: STRING, + default: null + }, + nout: { + type: INTEGER, + default: null + }, + txid: { + type: STRING, + default: null + }, + validAtHeight: { + type: INTEGER, + default: null + }, + outpoint: { + type: STRING, + default: null + }, + valueVersion: { + type: STRING, + default: null + }, + claimType: { + type: STRING, + default: null + }, + certificateVersion: { + type: STRING, + default: null + }, + keyType: { + type: STRING, + default: null + }, + publicKey: { + type: TEXT('long'), + default: null + } + }, { + freezeTableName: true + }); + + Certificate.associate = function (db) { + Certificate.belongsTo(db.Channel, { + foreignKey: { + allowNull: true + } + }); + }; + + Certificate.getShortChannelIdFromLongChannelId = function (longChannelId, channelName) { + var _this = this; + + logger.debug('getShortChannelIdFromLongChannelId ' + channelName + ':' + longChannelId); + return new Promise(function (resolve, reject) { + _this.findAll({ + where: { name: channelName }, + order: [['height', 'ASC']] + }).then(function (result) { + switch (result.length) { + case 0: + throw new Error('No channel(s) found with that channel name'); + default: + return resolve(returnShortId(result, longChannelId)); + } + }).catch(function (error) { + reject(error); + }); + }); + }; + + Certificate.getLongChannelIdFromShortChannelId = function (channelName, channelClaimId) { + var _this2 = this; + + logger.debug('getLongChannelIdFromShortChannelId(' + channelName + ', ' + channelClaimId + ')'); + return new Promise(function (resolve, reject) { + _this2.findAll({ + where: { + name: channelName, + claimId: { + $like: channelClaimId + '%' + } + }, + order: [['height', 'ASC']] + }).then(function (result) { + switch (result.length) { + case 0: + return resolve(null); + default: + // note results must be sorted + return resolve(result[0].claimId); + } + }).catch(function (error) { + reject(error); + }); + }); + }; + + Certificate.getLongChannelIdFromChannelName = function (channelName) { + var _this3 = this; + + logger.debug('getLongChannelIdFromChannelName(' + channelName + ')'); + return new Promise(function (resolve, reject) { + _this3.findAll({ + where: { name: channelName }, + order: [['effectiveAmount', 'DESC'], ['height', 'ASC']] + }).then(function (result) { + switch (result.length) { + case 0: + return resolve(null); + default: + return resolve(result[0].claimId); + } + }).catch(function (error) { + reject(error); + }); + }); + }; + + Certificate.validateLongChannelId = function (name, claimId) { + var _this4 = this; + + logger.debug('validateLongChannelId(' + name + ', ' + claimId + ')'); + return new Promise(function (resolve, reject) { + _this4.findOne({ + where: { name: name, claimId: claimId } + }).then(function (result) { + if (!result) { + return resolve(null); + }; + resolve(claimId); + }).catch(function (error) { + reject(error); + }); + }); + }; + + Certificate.getLongChannelId = function (channelName, channelClaimId) { + logger.debug('getLongChannelId(' + channelName + ', ' + channelClaimId + ')'); + if (channelClaimId && channelClaimId.length === 40) { + // if a full channel id is provided + return this.validateLongChannelId(channelName, channelClaimId); + } else if (channelClaimId && channelClaimId.length < 40) { + // if a short channel id is provided + return this.getLongChannelIdFromShortChannelId(channelName, channelClaimId); + } else { + return this.getLongChannelIdFromChannelName(channelName); // if no channel id provided + } + }; + + return Certificate; +}; + +/***/ }), +/* 46 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +module.exports = function (sequelize, _ref) { + var STRING = _ref.STRING; + + var Channel = sequelize.define('Channel', { + channelName: { + type: STRING, + allowNull: false + }, + channelClaimId: { + type: STRING, + allowNull: false + } + }, { + freezeTableName: true + }); + + Channel.associate = function (db) { + Channel.belongsTo(db.User); + Channel.hasOne(db.Certificate); + }; + + return Channel; +}; + +/***/ }), +/* 47 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var logger = __webpack_require__(0); + +var _require = __webpack_require__(12), + returnShortId = _require.returnShortId; + +var _require2 = __webpack_require__(3), + defaultThumbnail = _require2.assetDefaults.thumbnail, + host = _require2.details.host; + +function determineFileExtensionFromContentType(contentType) { + switch (contentType) { + case 'image/jpeg': + case 'image/jpg': + return 'jpeg'; + case 'image/png': + return 'png'; + case 'image/gif': + return 'gif'; + case 'video/mp4': + return 'mp4'; + default: + logger.debug('setting unknown file type as file extension jpeg'); + return 'jpeg'; + } +}; + +function determineThumbnail(storedThumbnail, defaultThumbnail) { + if (storedThumbnail === '') { + return defaultThumbnail; + } + return storedThumbnail; +}; + +function prepareClaimData(claim) { + // logger.debug('preparing claim data based on resolved data:', claim); + claim['thumbnail'] = determineThumbnail(claim.thumbnail, defaultThumbnail); + claim['fileExt'] = determineFileExtensionFromContentType(claim.contentType); + claim['host'] = host; + return claim; +}; + +module.exports = function (sequelize, _ref) { + var STRING = _ref.STRING, + BOOLEAN = _ref.BOOLEAN, + INTEGER = _ref.INTEGER, + TEXT = _ref.TEXT, + DECIMAL = _ref.DECIMAL; + + var Claim = sequelize.define('Claim', { + address: { + type: STRING, + default: null + }, + amount: { + type: DECIMAL(19, 8), + default: null + }, + claimId: { + type: STRING, + default: null + }, + claimSequence: { + type: INTEGER, + default: null + }, + decodedClaim: { + type: BOOLEAN, + default: null + }, + depth: { + type: INTEGER, + default: null + }, + effectiveAmount: { + type: DECIMAL(19, 8), + default: null + }, + hasSignature: { + type: BOOLEAN, + default: null + }, + height: { + type: INTEGER, + default: null + }, + hex: { + type: TEXT('long'), + default: null + }, + name: { + type: STRING, + default: null + }, + nout: { + type: INTEGER, + default: null + }, + txid: { + type: STRING, + default: null + }, + validAtHeight: { + type: INTEGER, + default: null + }, + outpoint: { + type: STRING, + default: null + }, + claimType: { + type: STRING, + default: null + }, + certificateId: { + type: STRING, + default: null + }, + author: { + type: STRING, + default: null + }, + description: { + type: TEXT('long'), + default: null + }, + language: { + type: STRING, + default: null + }, + license: { + type: STRING, + default: null + }, + licenseUrl: { + type: STRING, + default: null + }, + nsfw: { + type: BOOLEAN, + default: null + }, + preview: { + type: STRING, + default: null + }, + thumbnail: { + type: STRING, + default: null + }, + title: { + type: STRING, + default: null + }, + metadataVersion: { + type: STRING, + default: null + }, + contentType: { + type: STRING, + default: null + }, + source: { + type: STRING, + default: null + }, + sourceType: { + type: STRING, + default: null + }, + sourceVersion: { + type: STRING, + default: null + }, + streamVersion: { + type: STRING, + default: null + }, + valueVersion: { + type: STRING, + default: null + }, + channelName: { + type: STRING, + allowNull: true, + default: null + } + }, { + freezeTableName: true + }); + + Claim.associate = function (db) { + Claim.belongsTo(db.File, { + foreignKey: { + allowNull: true + } + }); + }; + + Claim.getShortClaimIdFromLongClaimId = function (claimId, claimName) { + var _this = this; + + logger.debug('Claim.getShortClaimIdFromLongClaimId for ' + claimName + '#' + claimId); + return new Promise(function (resolve, reject) { + _this.findAll({ + where: { name: claimName }, + order: [['height', 'ASC']] + }).then(function (result) { + switch (result.length) { + case 0: + throw new Error('No claim(s) found with that claim name'); + default: + resolve(returnShortId(result, claimId)); + } + }).catch(function (error) { + reject(error); + }); + }); + }; + + Claim.getAllChannelClaims = function (channelClaimId) { + var _this2 = this; + + logger.debug('Claim.getAllChannelClaims for ' + channelClaimId); + return new Promise(function (resolve, reject) { + _this2.findAll({ + where: { certificateId: channelClaimId }, + order: [['height', 'ASC']], + raw: true // returns an array of only data, not an array of instances + }).then(function (channelClaimsArray) { + // logger.debug('channelclaimsarray length:', channelClaimsArray.length); + switch (channelClaimsArray.length) { + case 0: + return resolve(null); + default: + channelClaimsArray.forEach(function (claim) { + claim['fileExt'] = determineFileExtensionFromContentType(claim.contentType); + claim['thumbnail'] = determineThumbnail(claim.thumbnail, defaultThumbnail); + return claim; + }); + return resolve(channelClaimsArray); + } + }).catch(function (error) { + reject(error); + }); + }); + }; + + Claim.getClaimIdByLongChannelId = function (channelClaimId, claimName) { + var _this3 = this; + + logger.debug('finding claim id for claim ' + claimName + ' from channel ' + channelClaimId); + return new Promise(function (resolve, reject) { + _this3.findAll({ + where: { name: claimName, certificateId: channelClaimId }, + order: [['id', 'ASC']] + }).then(function (result) { + switch (result.length) { + case 0: + return resolve(null); + case 1: + return resolve(result[0].claimId); + default: + logger.error(result.length + ' records found for "' + claimName + '" in channel "' + channelClaimId + '"'); + return resolve(result[0].claimId); + } + }).catch(function (error) { + reject(error); + }); + }); + }; + + Claim.getLongClaimIdFromShortClaimId = function (name, shortId) { + var _this4 = this; + + return new Promise(function (resolve, reject) { + _this4.findAll({ + where: { + name: name, + claimId: { + $like: shortId + '%' + } }, + order: [['height', 'ASC']] + }).then(function (result) { + switch (result.length) { + case 0: + return resolve(null); + default: + // note results must be sorted + return resolve(result[0].claimId); + } + }).catch(function (error) { + reject(error); + }); + }); + }; + + Claim.getTopFreeClaimIdByClaimName = function (name) { + var _this5 = this; + + return new Promise(function (resolve, reject) { + _this5.findAll({ + where: { name: name }, + order: [['effectiveAmount', 'DESC'], ['height', 'ASC']] // note: maybe height and effective amount need to switch? + }).then(function (result) { + logger.debug('length of result', result.length); + switch (result.length) { + case 0: + return resolve(null); + default: + return resolve(result[0].dataValues.claimId); + } + }).catch(function (error) { + reject(error); + }); + }); + }; + + Claim.validateLongClaimId = function (name, claimId) { + var _this6 = this; + + return new Promise(function (resolve, reject) { + _this6.findOne({ + where: { name: name, claimId: claimId } + }).then(function (result) { + if (!result) { + return resolve(null); + }; + resolve(claimId); + }).catch(function (error) { + reject(error); + }); + }); + }; + + Claim.getLongClaimId = function (claimName, claimId) { + logger.debug('getLongClaimId(' + claimName + ', ' + claimId + ')'); + if (claimId && claimId.length === 40) { + // if a full claim id is provided + return this.validateLongClaimId(claimName, claimId); + } else if (claimId && claimId.length < 40) { + return this.getLongClaimIdFromShortClaimId(claimName, claimId); // if a short claim id is provided + } else { + return this.getTopFreeClaimIdByClaimName(claimName); // if no claim id is provided + } + }; + + Claim.resolveClaim = function (name, claimId) { + var _this7 = this; + + logger.debug('Claim.resolveClaim: ' + name + ' ' + claimId); + return new Promise(function (resolve, reject) { + _this7.findAll({ + where: { name: name, claimId: claimId } + }).then(function (claimArray) { + switch (claimArray.length) { + case 0: + return resolve(null); + case 1: + return resolve(prepareClaimData(claimArray[0].dataValues)); + default: + logger.error('more than one record matches ' + name + '#' + claimId + ' in db.Claim'); + return resolve(prepareClaimData(claimArray[0].dataValues)); + } + }).catch(function (error) { + reject(error); + }); + }); + }; + + return Claim; +}; + +/***/ }), +/* 48 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +module.exports = function (sequelize, _ref) { + var STRING = _ref.STRING, + BOOLEAN = _ref.BOOLEAN, + INTEGER = _ref.INTEGER; + + var File = sequelize.define('File', { + name: { + type: STRING, + allowNull: false + }, + claimId: { + type: STRING, + allowNull: false + }, + address: { + type: STRING, + allowNull: false + }, + outpoint: { + type: STRING, + allowNull: false + }, + height: { + type: INTEGER, + allowNull: false, + default: 0 + }, + fileName: { + type: STRING, + allowNull: false + }, + filePath: { + type: STRING, + allowNull: false + }, + fileType: { + type: STRING + }, + nsfw: { + type: BOOLEAN, + allowNull: false, + defaultValue: false + }, + trendingEligible: { + type: BOOLEAN, + allowNull: false, + defaultValue: true + } + }, { + freezeTableName: true + }); + + File.associate = function (db) { + File.hasMany(db.Request); + File.hasOne(db.Claim); + }; + + File.getRecentClaims = function () { + return this.findAll({ + where: { nsfw: false, trendingEligible: true }, + order: [['createdAt', 'DESC']], + limit: 25 + }); + }; + + return File; +}; + +/***/ }), +/* 49 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +module.exports = function (sequelize, _ref) { + var STRING = _ref.STRING, + BOOLEAN = _ref.BOOLEAN, + TEXT = _ref.TEXT; + + var Request = sequelize.define('Request', { + action: { + type: STRING, + allowNull: false + }, + url: { + type: STRING, + allowNull: false + }, + ipAddress: { + type: STRING, + allowNull: true + }, + result: { + type: TEXT('long'), + allowNull: true, + default: null + } + }, { + freezeTableName: true + }); + + Request.associate = function (db) { + Request.belongsTo(db.File, { + foreignKey: { + allowNull: true + } + }); + }; + + return Request; +}; + +/***/ }), +/* 50 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var bcrypt = __webpack_require__(51); +var logger = __webpack_require__(0); + +module.exports = function (sequelize, _ref) { + var STRING = _ref.STRING; + + var User = sequelize.define('User', { + userName: { + type: STRING, + allowNull: false + }, + password: { + type: STRING, + allowNull: false + } + }, { + freezeTableName: true + }); + + User.associate = function (db) { + User.hasOne(db.Channel); + }; + + User.prototype.comparePassword = function (password) { + return bcrypt.compare(password, this.password); + }; + + User.prototype.changePassword = function (newPassword) { + var _this = this; + + return new Promise(function (resolve, reject) { + // generate a salt string to use for hashing + bcrypt.genSalt(function (saltError, salt) { + if (saltError) { + logger.error('salt error', saltError); + reject(saltError); + return; + } + // generate a hashed version of the user's password + bcrypt.hash(newPassword, salt, function (hashError, hash) { + // if there is an error with the hash generation return the error + if (hashError) { + logger.error('hash error', hashError); + reject(hashError); + return; + } + // replace the current password with the new hash + _this.update({ password: hash }).then(function () { + resolve(); + }).catch(function (error) { + reject(error); + }); + }); + }); + }); + }; + + // pre-save hook method to hash the user's password before the user's info is saved to the db. + User.hook('beforeCreate', function (user, options) { + logger.debug('User.beforeCreate hook...'); + return new Promise(function (resolve, reject) { + // generate a salt string to use for hashing + bcrypt.genSalt(function (saltError, salt) { + if (saltError) { + logger.error('salt error', saltError); + reject(saltError); + return; + } + // generate a hashed version of the user's password + bcrypt.hash(user.password, salt, function (hashError, hash) { + // if there is an error with the hash generation return the error + if (hashError) { + logger.error('hash error', hashError); + reject(hashError); + return; + } + // replace the password string with the hash password value + user.password = hash; + resolve(); + }); + }); + }); + }); + + return User; +}; + +/***/ }), +/* 51 */ +/***/ (function(module, exports) { + +module.exports = require("bcrypt"); + +/***/ }), +/* 52 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"]) _i["return"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError("Invalid attempt to destructure non-iterable instance"); } }; }(); + +var PassportLocalStrategy = __webpack_require__(11).Strategy; +var lbryApi = __webpack_require__(5); +var logger = __webpack_require__(0); +var db = __webpack_require__(2); + +module.exports = new PassportLocalStrategy({ + usernameField: 'username', + passwordField: 'password' +}, function (username, password, done) { + logger.verbose('new channel signup request. user: ' + username + ' pass: ' + password + ' .'); + var userInfo = {}; + // server-side validaton of inputs (username, password) + + // create the channel and retrieve the metadata + return lbryApi.createChannel('@' + username).then(function (tx) { + // create user record + var userData = { + userName: username, + password: password + }; + logger.verbose('userData >', userData); + // create user record + var channelData = { + channelName: '@' + username, + channelClaimId: tx.claim_id + }; + logger.verbose('channelData >', channelData); + // create certificate record + var certificateData = { + claimId: tx.claim_id, + name: '@' + username + // address, + }; + logger.verbose('certificateData >', certificateData); + // save user and certificate to db + return Promise.all([db.User.create(userData), db.Channel.create(channelData), db.Certificate.create(certificateData)]); + }).then(function (_ref) { + var _ref2 = _slicedToArray(_ref, 3), + newUser = _ref2[0], + newChannel = _ref2[1], + newCertificate = _ref2[2]; + + logger.verbose('user and certificate successfully created'); + // store the relevant newUser info to be passed back for req.User + userInfo['id'] = newUser.id; + userInfo['userName'] = newUser.userName; + userInfo['channelName'] = newChannel.channelName; + userInfo['channelClaimId'] = newChannel.channelClaimId; + // associate the instances + return Promise.all([newCertificate.setChannel(newChannel), newChannel.setUser(newUser)]); + }).then(function () { + logger.verbose('user and certificate successfully associated'); + return db.Certificate.getShortChannelIdFromLongChannelId(userInfo.channelClaimId, userInfo.channelName); + }).then(function (shortChannelId) { + userInfo['shortChannelId'] = shortChannelId; + return done(null, userInfo); + }).catch(function (error) { + logger.error('signup error', error); + return done(error); + }); +}); + +/***/ }), +/* 53 */ +/***/ (function(module, exports) { + +module.exports = require("axios"); + +/***/ }), +/* 54 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var lbryConfig = { + api: { + apiHost: 'localhost', + apiPort: '5279' + } +}; + +module.exports = lbryConfig; + +/***/ }), +/* 55 */ +/***/ (function(module, exports) { + +module.exports = require("universal-analytics"); + +/***/ }), +/* 56 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +module.exports = { + serializeSpeechUser: function serializeSpeechUser(user, done) { + // returns user data to be serialized into session + console.log('serializing user'); + done(null, user); + }, + deserializeSpeechUser: function deserializeSpeechUser(user, done) { + // deserializes session and populates additional info to req.user + console.log('deserializing user'); + done(null, user); + } +}; + +/***/ }), +/* 57 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var speechPassport = __webpack_require__(7); +var handleSignupRequest = __webpack_require__(58); +var handleLoginRequest = __webpack_require__(59); +var handleLogoutRequest = __webpack_require__(60); +var handleUserRequest = __webpack_require__(61); + +module.exports = function (app) { + app.post('/signup', speechPassport.authenticate('local-signup'), handleSignupRequest); + app.post('/login', handleLoginRequest); + app.get('/logout', handleLogoutRequest); + app.get('/user', handleUserRequest); +}; + +/***/ }), +/* 58 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var signup = function signup(req, res) { + res.status(200).json({ + success: true, + channelName: req.user.channelName, + channelClaimId: req.user.channelClaimId, + shortChannelId: req.user.shortChannelId + }); +}; + +module.exports = signup; + +/***/ }), +/* 59 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var speechPassport = __webpack_require__(7); + +var login = function login(req, res, next) { + speechPassport.authenticate('local-login', function (err, user, info) { + if (err) { + return next(err); + } + if (!user) { + return res.status(400).json({ + success: false, + message: info.message + }); + } + req.logIn(user, function (err) { + if (err) { + return next(err); + } + return res.status(200).json({ + success: true, + channelName: req.user.channelName, + channelClaimId: req.user.channelClaimId, + shortChannelId: req.user.shortChannelId + }); + }); + })(req, res, next); +}; + +module.exports = login; + +/***/ }), +/* 60 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var logout = function logout(req, res) { + req.logout(); + res.status(200).json({ success: true, message: 'you successfully logged out' }); +}; + +module.exports = logout; + +/***/ }), +/* 61 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var user = function user(req, res) { + if (req.user) { + res.status(200).json({ success: true, data: req.user }); + } else { + res.status(401).json({ success: false, message: 'user is not logged in' }); + } +}; + +module.exports = user; + +/***/ }), +/* 62 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var channelAvailability = __webpack_require__(63); +var channelClaims = __webpack_require__(65); +var channelData = __webpack_require__(67); +var channelShortId = __webpack_require__(68); +var claimAvailability = __webpack_require__(69); +var claimData = __webpack_require__(70); +var claimGet = __webpack_require__(71); +var claimLongId = __webpack_require__(72); +var claimPublish = __webpack_require__(73); +var claimResolve = __webpack_require__(75); +var claimShortId = __webpack_require__(76); +var claimList = __webpack_require__(77); +var fileAvailability = __webpack_require__(78); + +var multipartMiddleware = __webpack_require__(79); + +module.exports = function (app) { + // channel routes + app.get('/api/channel/availability/:name', channelAvailability); + app.get('/api/channel/short-id/:longId/:name', channelShortId); + app.get('/api/channel/data/:channelName/:channelClaimId', channelData); + app.get('/api/channel/claims/:channelName/:channelClaimId/:page', channelClaims); + // claim routes + app.get('/api/claim/list/:name', claimList); + app.get('/api/claim/get/:name/:claimId', claimGet); + app.get('/api/claim/availability/:name', claimAvailability); + app.get('/api/claim/resolve/:name/:claimId', claimResolve); + app.post('/api/claim/publish', multipartMiddleware, claimPublish); + app.get('/api/claim/short-id/:longId/:name', claimShortId); + app.post('/api/claim/long-id', claimLongId); + app.get('/api/claim/data/:claimName/:claimId', claimData); + // file routes + app.get('/api/file/availability/:name/:claimId', fileAvailability); +}; + +/***/ }), +/* 63 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var _require = __webpack_require__(8), + checkChannelAvailability = _require.checkChannelAvailability; + +var _require2 = __webpack_require__(4), + sendGATimingEvent = _require2.sendGATimingEvent; + +var _require3 = __webpack_require__(1), + handleErrorResponse = _require3.handleErrorResponse; + +/* + + route to check whether site has published to a channel + +*/ + +var channelAvailability = function channelAvailability(_ref, res) { + var ip = _ref.ip, + originalUrl = _ref.originalUrl, + name = _ref.params.name; + + var gaStartTime = Date.now(); + checkChannelAvailability(name).then(function (availableName) { + res.status(200).json(availableName); + sendGATimingEvent('end-to-end', 'claim name availability', name, gaStartTime, Date.now()); + }).catch(function (error) { + handleErrorResponse(originalUrl, ip, error, res); + }); +}; + +module.exports = channelAvailability; + +/***/ }), +/* 64 */ +/***/ (function(module, exports) { + +module.exports = require("fs"); + +/***/ }), +/* 65 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var _require = __webpack_require__(6), + getChannelClaims = _require.getChannelClaims; + +var _require2 = __webpack_require__(1), + handleErrorResponse = _require2.handleErrorResponse; + +var NO_CHANNEL = 'NO_CHANNEL'; + +/* + + route to get all claims for channel + +*/ + +var channelClaims = function channelClaims(_ref, res) { + var ip = _ref.ip, + originalUrl = _ref.originalUrl, + body = _ref.body, + params = _ref.params; + + var channelName = params.channelName; + var channelClaimId = params.channelClaimId; + if (channelClaimId === 'none') channelClaimId = null; + var page = params.page; + getChannelClaims(channelName, channelClaimId, page).then(function (data) { + if (data === NO_CHANNEL) { + return res.status(404).json({ success: false, message: 'No matching channel was found' }); + } + res.status(200).json({ success: true, data: data }); + }).catch(function (error) { + handleErrorResponse(originalUrl, ip, error, res); + }); +}; + +module.exports = channelClaims; + +/***/ }), +/* 66 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var CLAIMS_PER_PAGE = 12; + +module.exports = { + returnPaginatedChannelClaims: function returnPaginatedChannelClaims(channelName, longChannelClaimId, claims, page) { + var totalPages = module.exports.determineTotalPages(claims); + var paginationPage = module.exports.getPageFromQuery(page); + var viewData = { + channelName: channelName, + longChannelClaimId: longChannelClaimId, + claims: module.exports.extractPageFromClaims(claims, paginationPage), + previousPage: module.exports.determinePreviousPage(paginationPage), + currentPage: paginationPage, + nextPage: module.exports.determineNextPage(totalPages, paginationPage), + totalPages: totalPages, + totalResults: module.exports.determineTotalClaims(claims) + }; + return viewData; + }, + getPageFromQuery: function getPageFromQuery(page) { + if (page) { + return parseInt(page); + } + return 1; + }, + extractPageFromClaims: function extractPageFromClaims(claims, pageNumber) { + if (!claims) { + return []; // if no claims, return this default + } + // logger.debug('claims is array?', Array.isArray(claims)); + // logger.debug(`pageNumber ${pageNumber} is number?`, Number.isInteger(pageNumber)); + var claimStartIndex = (pageNumber - 1) * CLAIMS_PER_PAGE; + var claimEndIndex = claimStartIndex + CLAIMS_PER_PAGE; + var pageOfClaims = claims.slice(claimStartIndex, claimEndIndex); + return pageOfClaims; + }, + determineTotalPages: function determineTotalPages(claims) { + if (!claims) { + return 0; + } else { + var totalClaims = claims.length; + if (totalClaims < CLAIMS_PER_PAGE) { + return 1; + } + var fullPages = Math.floor(totalClaims / CLAIMS_PER_PAGE); + var remainder = totalClaims % CLAIMS_PER_PAGE; + if (remainder === 0) { + return fullPages; + } + return fullPages + 1; + } + }, + determinePreviousPage: function determinePreviousPage(currentPage) { + if (currentPage === 1) { + return null; + } + return currentPage - 1; + }, + determineNextPage: function determineNextPage(totalPages, currentPage) { + if (currentPage === totalPages) { + return null; + } + return currentPage + 1; + }, + determineTotalClaims: function determineTotalClaims(claims) { + if (!claims) { + return 0; + } + return claims.length; + } +}; + +/***/ }), +/* 67 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var _require = __webpack_require__(6), + getChannelData = _require.getChannelData; + +var _require2 = __webpack_require__(1), + handleErrorResponse = _require2.handleErrorResponse; + +var NO_CHANNEL = 'NO_CHANNEL'; + +/* + + route to get data for a channel + +*/ + +var channelData = function channelData(_ref, res) { + var ip = _ref.ip, + originalUrl = _ref.originalUrl, + body = _ref.body, + params = _ref.params; + + var channelName = params.channelName; + var channelClaimId = params.channelClaimId; + if (channelClaimId === 'none') channelClaimId = null; + getChannelData(channelName, channelClaimId, 0).then(function (data) { + if (data === NO_CHANNEL) { + return res.status(404).json({ success: false, message: 'No matching channel was found' }); + } + res.status(200).json({ success: true, data: data }); + }).catch(function (error) { + handleErrorResponse(originalUrl, ip, error, res); + }); +}; + +module.exports = channelData; + +/***/ }), +/* 68 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var _require = __webpack_require__(1), + handleErrorResponse = _require.handleErrorResponse; + +var db = __webpack_require__(2); + +/* + +route to get a short channel id from long channel Id + +*/ + +var channelShortIdRoute = function channelShortIdRoute(_ref, res) { + var ip = _ref.ip, + originalUrl = _ref.originalUrl, + params = _ref.params; + + db.Certificate.getShortChannelIdFromLongChannelId(params.longId, params.name).then(function (shortId) { + res.status(200).json(shortId); + }).catch(function (error) { + handleErrorResponse(originalUrl, ip, error, res); + }); +}; + +module.exports = channelShortIdRoute; + +/***/ }), +/* 69 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var _require = __webpack_require__(8), + claimNameIsAvailable = _require.claimNameIsAvailable; + +var _require2 = __webpack_require__(4), + sendGATimingEvent = _require2.sendGATimingEvent; + +var _require3 = __webpack_require__(1), + handleErrorResponse = _require3.handleErrorResponse; + +/* + + route to check whether this site published to a claim + +*/ + +var claimAvailability = function claimAvailability(_ref, res) { + var ip = _ref.ip, + originalUrl = _ref.originalUrl, + name = _ref.params.name; + + var gaStartTime = Date.now(); + claimNameIsAvailable(name).then(function (result) { + res.status(200).json(result); + sendGATimingEvent('end-to-end', 'claim name availability', name, gaStartTime, Date.now()); + }).catch(function (error) { + handleErrorResponse(originalUrl, ip, error, res); + }); +}; + +module.exports = claimAvailability; + +/***/ }), +/* 70 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var _require = __webpack_require__(1), + handleErrorResponse = _require.handleErrorResponse; + +var db = __webpack_require__(2); + +/* + + route to return data for a claim + +*/ + +var claimData = function claimData(_ref, res) { + var ip = _ref.ip, + originalUrl = _ref.originalUrl, + body = _ref.body, + params = _ref.params; + + var claimName = params.claimName; + var claimId = params.claimId; + if (claimId === 'none') claimId = null; + db.Claim.resolveClaim(claimName, claimId).then(function (claimInfo) { + if (!claimInfo) { + return res.status(404).json({ success: false, message: 'No claim could be found' }); + } + res.status(200).json({ success: true, data: claimInfo }); + }).catch(function (error) { + handleErrorResponse(originalUrl, ip, error, res); + }); +}; + +module.exports = claimData; + +/***/ }), +/* 71 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"]) _i["return"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError("Invalid attempt to destructure non-iterable instance"); } }; }(); + +var _require = __webpack_require__(5), + getClaim = _require.getClaim; + +var _require2 = __webpack_require__(9), + addGetResultsToFileData = _require2.addGetResultsToFileData, + createFileData = _require2.createFileData; + +var _require3 = __webpack_require__(1), + handleErrorResponse = _require3.handleErrorResponse; + +var db = __webpack_require__(2); + +/* + + route to get a claim + +*/ + +var claimGet = function claimGet(_ref, res) { + var ip = _ref.ip, + originalUrl = _ref.originalUrl, + params = _ref.params; + + var name = params.name; + var claimId = params.claimId; + // resolve the claim + db.Claim.resolveClaim(name, claimId).then(function (resolveResult) { + // make sure a claim actually exists at that uri + if (!resolveResult) { + throw new Error('No matching uri found in Claim table'); + } + var fileData = createFileData(resolveResult); + // get the claim + return Promise.all([fileData, getClaim(name + '#' + claimId)]); + }).then(function (_ref2) { + var _ref3 = _slicedToArray(_ref2, 2), + fileData = _ref3[0], + getResult = _ref3[1]; + + fileData = addGetResultsToFileData(fileData, getResult); + return Promise.all([db.upsert(db.File, fileData, { name: name, claimId: claimId }, 'File'), getResult]); + }).then(function (_ref4) { + var _ref5 = _slicedToArray(_ref4, 2), + fileRecord = _ref5[0], + _ref5$ = _ref5[1], + message = _ref5$.message, + completed = _ref5$.completed; + + res.status(200).json({ success: true, message: message, completed: completed }); + }).catch(function (error) { + handleErrorResponse(originalUrl, ip, error, res); + }); +}; + +module.exports = claimGet; + +/***/ }), +/* 72 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var _require = __webpack_require__(6), + getClaimId = _require.getClaimId; + +var _require2 = __webpack_require__(1), + handleErrorResponse = _require2.handleErrorResponse; + +var NO_CHANNEL = 'NO_CHANNEL'; +var NO_CLAIM = 'NO_CLAIM'; + +/* + + route to get a long claim id + +*/ + +var claimLongId = function claimLongId(_ref, res) { + var ip = _ref.ip, + originalUrl = _ref.originalUrl, + body = _ref.body, + params = _ref.params; + + var channelName = body.channelName; + var channelClaimId = body.channelClaimId; + var claimName = body.claimName; + var claimId = body.claimId; + getClaimId(channelName, channelClaimId, claimName, claimId).then(function (result) { + if (result === NO_CHANNEL) { + return res.status(404).json({ success: false, message: 'No matching channel could be found' }); + } + if (result === NO_CLAIM) { + return res.status(404).json({ success: false, message: 'No matching claim id could be found' }); + } + res.status(200).json({ success: true, data: result }); + }).catch(function (error) { + handleErrorResponse(originalUrl, ip, error, res); + }); +}; + +module.exports = claimLongId; + +/***/ }), +/* 73 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"]) _i["return"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError("Invalid attempt to destructure non-iterable instance"); } }; }(); + +var _require = __webpack_require__(9), + createBasicPublishParams = _require.createBasicPublishParams, + createThumbnailPublishParams = _require.createThumbnailPublishParams, + parsePublishApiRequestBody = _require.parsePublishApiRequestBody, + parsePublishApiRequestFiles = _require.parsePublishApiRequestFiles; + +var _require2 = __webpack_require__(8), + claimNameIsAvailable = _require2.claimNameIsAvailable, + publish = _require2.publish; + +var _require3 = __webpack_require__(74), + authenticateUser = _require3.authenticateUser; + +var _require4 = __webpack_require__(4), + sendGATimingEvent = _require4.sendGATimingEvent; + +var _require5 = __webpack_require__(1), + handleErrorResponse = _require5.handleErrorResponse; + +var _require6 = __webpack_require__(3), + host = _require6.details.host; + +/* + + route to publish a claim through the daemon + +*/ + +var claimPublish = function claimPublish(_ref, res) { + var body = _ref.body, + files = _ref.files, + headers = _ref.headers, + ip = _ref.ip, + originalUrl = _ref.originalUrl, + user = _ref.user; + + // define variables + var channelName = void 0, + channelId = void 0, + channelPassword = void 0, + description = void 0, + fileName = void 0, + filePath = void 0, + fileType = void 0, + gaStartTime = void 0, + license = void 0, + name = void 0, + nsfw = void 0, + thumbnail = void 0, + thumbnailFileName = void 0, + thumbnailFilePath = void 0, + thumbnailFileType = void 0, + title = void 0; + // record the start time of the request + gaStartTime = Date.now(); + // validate the body and files of the request + try { + var _parsePublishApiReque = parsePublishApiRequestBody(body); + // validateApiPublishRequest(body, files); + + + name = _parsePublishApiReque.name; + nsfw = _parsePublishApiReque.nsfw; + license = _parsePublishApiReque.license; + title = _parsePublishApiReque.title; + description = _parsePublishApiReque.description; + thumbnail = _parsePublishApiReque.thumbnail; + + var _parsePublishApiReque2 = parsePublishApiRequestFiles(files); + + fileName = _parsePublishApiReque2.fileName; + filePath = _parsePublishApiReque2.filePath; + fileType = _parsePublishApiReque2.fileType; + thumbnailFileName = _parsePublishApiReque2.thumbnailFileName; + thumbnailFilePath = _parsePublishApiReque2.thumbnailFilePath; + thumbnailFileType = _parsePublishApiReque2.thumbnailFileType; + channelName = body.channelName; + channelId = body.channelId; + channelPassword = body.channelPassword; + } catch (error) { + return res.status(400).json({ success: false, message: error.message }); + } + // check channel authorization + Promise.all([authenticateUser(channelName, channelId, channelPassword, user), claimNameIsAvailable(name), createBasicPublishParams(filePath, name, title, description, license, nsfw, thumbnail), createThumbnailPublishParams(thumbnailFilePath, name, license, nsfw)]).then(function (_ref2) { + var _ref3 = _slicedToArray(_ref2, 4), + _ref3$ = _ref3[0], + channelName = _ref3$.channelName, + channelClaimId = _ref3$.channelClaimId, + validatedClaimName = _ref3[1], + publishParams = _ref3[2], + thumbnailPublishParams = _ref3[3]; + + // add channel details to the publish params + if (channelName && channelClaimId) { + publishParams['channel_name'] = channelName; + publishParams['channel_id'] = channelClaimId; + } + // publish the thumbnail + if (thumbnailPublishParams) { + publish(thumbnailPublishParams, thumbnailFileName, thumbnailFileType); + } + // publish the asset + return publish(publishParams, fileName, fileType); + }).then(function (result) { + res.status(200).json({ + success: true, + message: 'publish completed successfully', + data: { + name: name, + claimId: result.claim_id, + url: host + '/' + result.claim_id + '/' + name, + lbryTx: result + } + }); + // record the publish end time and send to google analytics + sendGATimingEvent('end-to-end', 'publish', fileType, gaStartTime, Date.now()); + }).catch(function (error) { + handleErrorResponse(originalUrl, ip, error, res); + }); +}; + +module.exports = claimPublish; + +/***/ }), +/* 74 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var db = __webpack_require__(2); +var logger = __webpack_require__(0); + +module.exports = { + authenticateUser: function authenticateUser(channelName, channelId, channelPassword, user) { + // case: no channelName or channel Id are provided (anonymous), regardless of whether user token is provided + if (!channelName && !channelId) { + return { + channelName: null, + channelClaimId: null + }; + } + // case: channelName or channel Id are provided with user token + if (user) { + if (channelName && channelName !== user.channelName) { + throw new Error('the provided channel name does not match user credentials'); + } + if (channelId && channelId !== user.channelClaimId) { + throw new Error('the provided channel id does not match user credentials'); + } + return { + channelName: user.channelName, + channelClaimId: user.channelClaimId + }; + } + // case: channelName or channel Id are provided with password instead of user token + if (!channelPassword) throw new Error('no channel password provided'); + return module.exports.authenticateChannelCredentials(channelName, channelId, channelPassword); + }, + authenticateChannelCredentials: function authenticateChannelCredentials(channelName, channelId, userPassword) { + return new Promise(function (resolve, reject) { + // hoisted variables + var channelData = void 0; + // build the params for finding the channel + var channelFindParams = {}; + if (channelName) channelFindParams['channelName'] = channelName; + if (channelId) channelFindParams['channelClaimId'] = channelId; + // find the channel + db.Channel.findOne({ + where: channelFindParams + }).then(function (channel) { + if (!channel) { + logger.debug('no channel found'); + throw new Error('Authentication failed, you do not have access to that channel'); + } + channelData = channel.get(); + logger.debug('channel data:', channelData); + return db.User.findOne({ + where: { userName: channelData.channelName.substring(1) } + }); + }).then(function (user) { + if (!user) { + logger.debug('no user found'); + throw new Error('Authentication failed, you do not have access to that channel'); + } + return user.comparePassword(userPassword); + }).then(function (isMatch) { + if (!isMatch) { + logger.debug('incorrect password'); + throw new Error('Authentication failed, you do not have access to that channel'); + } + logger.debug('...password was a match...'); + resolve(channelData); + }).catch(function (error) { + reject(error); + }); + }); + } +}; + +/***/ }), +/* 75 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var _require = __webpack_require__(5), + resolveUri = _require.resolveUri; + +var _require2 = __webpack_require__(1), + handleErrorResponse = _require2.handleErrorResponse; + +/* + + route to run a resolve request on the daemon + +*/ + +var claimResolve = function claimResolve(_ref, res) { + var headers = _ref.headers, + ip = _ref.ip, + originalUrl = _ref.originalUrl, + params = _ref.params; + + resolveUri(params.name + '#' + params.claimId).then(function (resolvedUri) { + res.status(200).json(resolvedUri); + }).catch(function (error) { + handleErrorResponse(originalUrl, ip, error, res); + }); +}; + +module.exports = claimResolve; + +/***/ }), +/* 76 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var _require = __webpack_require__(1), + handleErrorResponse = _require.handleErrorResponse; + +var db = __webpack_require__(2); + +/* + + route to get a short claim id from long claim Id + +*/ + +var claimShortId = function claimShortId(_ref, res) { + var ip = _ref.ip, + originalUrl = _ref.originalUrl, + body = _ref.body, + params = _ref.params; + + db.Claim.getShortClaimIdFromLongClaimId(params.longId, params.name).then(function (shortId) { + res.status(200).json({ success: true, data: shortId }); + }).catch(function (error) { + handleErrorResponse(originalUrl, ip, error, res); + }); +}; + +module.exports = claimShortId; + +/***/ }), +/* 77 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var _require = __webpack_require__(5), + getClaimList = _require.getClaimList; + +var _require2 = __webpack_require__(1), + handleErrorResponse = _require2.handleErrorResponse; + +/* + + route to get list of claims + +*/ + +var claimList = function claimList(_ref, res) { + var ip = _ref.ip, + originalUrl = _ref.originalUrl, + params = _ref.params; + + getClaimList(params.name).then(function (claimsList) { + res.status(200).json(claimsList); + }).catch(function (error) { + handleErrorResponse(originalUrl, ip, error, res); + }); +}; + +module.exports = claimList; + +/***/ }), +/* 78 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var _require = __webpack_require__(1), + handleErrorResponse = _require.handleErrorResponse; + +var db = __webpack_require__(2); + +/* + + route to see if asset is available locally + +*/ + +var fileAvailability = function fileAvailability(_ref, res) { + var ip = _ref.ip, + originalUrl = _ref.originalUrl, + params = _ref.params; + + var name = params.name; + var claimId = params.claimId; + db.File.findOne({ + where: { + name: name, + claimId: claimId + } + }).then(function (result) { + if (result) { + return res.status(200).json({ success: true, data: true }); + } + res.status(200).json({ success: true, data: false }); + }).catch(function (error) { + handleErrorResponse(originalUrl, ip, error, res); + }); +}; + +module.exports = fileAvailability; + +/***/ }), +/* 79 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var multipart = __webpack_require__(80); + +var _require = __webpack_require__(3), + uploadDirectory = _require.publishing.uploadDirectory; + +var multipartMiddleware = multipart({ uploadDir: uploadDirectory }); + +module.exports = multipartMiddleware; + +/***/ }), +/* 80 */ +/***/ (function(module, exports) { + +module.exports = require("connect-multiparty"); + +/***/ }), +/* 81 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var handlePageRequest = __webpack_require__(82); +var handleEmbedRequest = __webpack_require__(83); +var redirect = __webpack_require__(84); + +module.exports = function (app) { + app.get('/', handlePageRequest); + app.get('/login', handlePageRequest); + app.get('/about', handlePageRequest); + app.get('/trending', redirect('/popular')); + app.get('/popular', handlePageRequest); + app.get('/new', handlePageRequest); + app.get('/embed/:claimId/:name', handleEmbedRequest); // route to send embedable video player (for twitter) +}; + +/***/ }), +/* 82 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var handlePageRender = __webpack_require__(14); + +var sendReactApp = function sendReactApp(req, res) { + handlePageRender(req, res); +}; + +module.exports = sendReactApp; + +/***/ }), +/* 83 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var _require = __webpack_require__(3), + host = _require.details.host; + +var sendEmbedPage = function sendEmbedPage(_ref, res) { + var params = _ref.params; + + var claimId = params.claimId; + var name = params.name; + // get and render the content + res.status(200).render('embed', { layout: 'embed', host: host, claimId: claimId, name: name }); +}; + +module.exports = sendEmbedPage; + +/***/ }), +/* 84 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var redirect = function redirect(route) { + return function (req, res) { + res.status(301).redirect(route); + }; +}; + +module.exports = redirect; + +/***/ }), +/* 85 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var serveAssetByClaim = __webpack_require__(86); +var serveAssetByIdentifierAndClaim = __webpack_require__(89); + +module.exports = function (app, db) { + app.get('/:identifier/:claim', serveAssetByIdentifierAndClaim); + app.get('/:claim', serveAssetByClaim); +}; + +/***/ }), +/* 86 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var _require = __webpack_require__(4), + sendGAServeEvent = _require.sendGAServeEvent; + +var _require2 = __webpack_require__(23), + determineResponseType = _require2.determineResponseType, + logRequestData = _require2.logRequestData, + getClaimIdAndServeAsset = _require2.getClaimIdAndServeAsset; + +var lbryUri = __webpack_require__(24); +var handleShowRender = __webpack_require__(25); +var SERVE = 'SERVE'; + +/* + + route to serve an asset or the react app via the claim name only + +*/ + +var serverAssetByClaim = function serverAssetByClaim(req, res) { + var headers = req.headers, + ip = req.ip, + originalUrl = req.originalUrl, + params = req.params; + // decide if this is a show request + + var hasFileExtension = void 0; + try { + var _lbryUri$parseModifie = lbryUri.parseModifier(params.claim); + + hasFileExtension = _lbryUri$parseModifie.hasFileExtension; + } catch (error) { + return res.status(400).json({ success: false, message: error.message }); + } + var responseType = determineResponseType(hasFileExtension, headers); + if (responseType !== SERVE) { + return handleShowRender(req, res); + } + // handle serve request + // send google analytics + sendGAServeEvent(headers, ip, originalUrl); + // parse the claim + var claimName = void 0; + try { + var _lbryUri$parseClaim = lbryUri.parseClaim(params.claim); + + claimName = _lbryUri$parseClaim.claimName; + } catch (error) { + return res.status(400).json({ success: false, message: error.message }); + } + // log the request data for debugging + logRequestData(responseType, claimName, null, null); + // get the claim Id and then serve the asset + getClaimIdAndServeAsset(null, null, claimName, null, originalUrl, ip, res); +}; + +module.exports = serverAssetByClaim; + +/***/ }), +/* 87 */ +/***/ (function(module, exports) { + +module.exports = require("redux-saga"); + +/***/ }), +/* 88 */ +/***/ (function(module, exports) { + +module.exports = require("redux-saga/effects"); + +/***/ }), +/* 89 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"]) _i["return"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError("Invalid attempt to destructure non-iterable instance"); } }; }(); + +var _require = __webpack_require__(4), + sendGAServeEvent = _require.sendGAServeEvent; + +var _require2 = __webpack_require__(23), + determineResponseType = _require2.determineResponseType, + flipClaimNameAndIdForBackwardsCompatibility = _require2.flipClaimNameAndIdForBackwardsCompatibility, + logRequestData = _require2.logRequestData, + getClaimIdAndServeAsset = _require2.getClaimIdAndServeAsset; + +var lbryUri = __webpack_require__(24); +var handleShowRender = __webpack_require__(25); + +var SERVE = 'SERVE'; + +/* + + route to serve an asset or the react app via the claim name and an identifier + +*/ + +var serverAssetByIdentifierAndClaim = function serverAssetByIdentifierAndClaim(req, res) { + var headers = req.headers, + ip = req.ip, + originalUrl = req.originalUrl, + params = req.params; + // decide if this is a show request + + var hasFileExtension = void 0; + try { + var _lbryUri$parseModifie = lbryUri.parseModifier(params.claim); + + hasFileExtension = _lbryUri$parseModifie.hasFileExtension; + } catch (error) { + return res.status(400).json({ success: false, message: error.message }); + } + var responseType = determineResponseType(hasFileExtension, headers); + if (responseType !== SERVE) { + return handleShowRender(req, res); + } + // handle serve request + // send google analytics + sendGAServeEvent(headers, ip, originalUrl); + // parse the claim + var claimName = void 0; + try { + var _lbryUri$parseClaim = lbryUri.parseClaim(params.claim); + + claimName = _lbryUri$parseClaim.claimName; + } catch (error) { + return res.status(400).json({ success: false, message: error.message }); + } + // parse the identifier + var isChannel = void 0, + channelName = void 0, + channelClaimId = void 0, + claimId = void 0; + try { + var _lbryUri$parseIdentif = lbryUri.parseIdentifier(params.identifier); + + isChannel = _lbryUri$parseIdentif.isChannel; + channelName = _lbryUri$parseIdentif.channelName; + channelClaimId = _lbryUri$parseIdentif.channelClaimId; + claimId = _lbryUri$parseIdentif.claimId; + } catch (error) { + return res.status(400).json({ success: false, message: error.message }); + } + if (!isChannel) { + var _flipClaimNameAndIdFo = flipClaimNameAndIdForBackwardsCompatibility(claimId, claimName); + + var _flipClaimNameAndIdFo2 = _slicedToArray(_flipClaimNameAndIdFo, 2); + + claimId = _flipClaimNameAndIdFo2[0]; + claimName = _flipClaimNameAndIdFo2[1]; + } + // log the request data for debugging + logRequestData(responseType, claimName, channelName, claimId); + // get the claim Id and then serve the asset + getClaimIdAndServeAsset(channelName, channelClaimId, claimName, claimId, originalUrl, ip, res); +}; + +module.exports = serverAssetByIdentifierAndClaim; + +/***/ }), +/* 90 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var handlePageRequest = __webpack_require__(91); + +module.exports = function (app) { + app.get('*', handlePageRequest); +}; + +/***/ }), +/* 91 */ +/***/ (function(module, exports, __webpack_require__) { + +"use strict"; + + +var handlePageRender = __webpack_require__(14); + +var sendReactApp = function sendReactApp(req, res) { + handlePageRender(req, res); +}; + +module.exports = sendReactApp; + +/***/ }) +/******/ ]); +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///webpack/bootstrap 82ebf2ca30129849e4ef","webpack:///external \"winston\"","webpack:///./server/helpers/errorHandlers.js","webpack:///./server/models/index.js","webpack:///./config/siteConfig.js","webpack:///./server/helpers/googleAnalytics.js","webpack:///./server/helpers/lbryApi.js","webpack:///./server/controllers/serveController.js","webpack:///./server/speechPassport/index.js","webpack:///./server/controllers/publishController.js","webpack:///./server/helpers/publishHelpers.js","webpack:///./config/mysqlConfig.js","webpack:///external \"passport-local\"","webpack:///./server/helpers/sequelizeHelpers.js","webpack:///external \"sequelize\"","webpack:///./server/helpers/handlePageRender.jsx","webpack:///external \"react\"","webpack:///external \"react-dom/server\"","webpack:///external \"redux\"","webpack:///external \"react-redux\"","webpack:///external \"react-router-dom\"","webpack:///external \"spee.ch-components\"","webpack:///./server/helpers/renderFullPage.js","webpack:///external \"react-helmet\"","webpack:///./server/helpers/serveHelpers.js","webpack:///./server/helpers/lbryUri.js","webpack:///./server/helpers/handleShowRender.jsx","webpack:///external \"babel-polyfill\"","webpack:///external \"whatwg-fetch\"","webpack:///./speech.js","webpack:///./server/index.js","webpack:///external \"express\"","webpack:///external \"body-parser\"","webpack:///external \"express-handlebars\"","webpack:///external \"handlebars\"","webpack:///external \"helmet\"","webpack:///external \"cookie-session\"","webpack:///external \"http\"","webpack:///./server/middleware/requestLogger.js","webpack:///external \"path\"","webpack:///./config/loggerConfig.js","webpack:///./config/slackConfig.js","webpack:///external \"winston-slack-webhook\"","webpack:///external \"passport\"","webpack:///./server/speechPassport/local-login.js","webpack:///./server/models/certificate.js","webpack:///./server/models/channel.js","webpack:///./server/models/claim.js","webpack:///./server/models/file.js","webpack:///./server/models/request.js","webpack:///./server/models/user.js","webpack:///external \"bcrypt\"","webpack:///./server/speechPassport/local-signup.js","webpack:///external \"axios\"","webpack:///./config/lbryConfig.js","webpack:///external \"universal-analytics\"","webpack:///./server/helpers/authHelpers.js","webpack:///./server/routes/auth/index.js","webpack:///./server/routes/auth/signup.js","webpack:///./server/routes/auth/login.js","webpack:///./server/routes/auth/logout.js","webpack:///./server/routes/auth/user.js","webpack:///./server/routes/api/index.js","webpack:///./server/routes/api/channelAvailability.js","webpack:///external \"fs\"","webpack:///./server/routes/api/channelClaims.js","webpack:///./server/helpers/channelPagination.js","webpack:///./server/routes/api/channelData.js","webpack:///./server/routes/api/channelShortId.js","webpack:///./server/routes/api/claimAvailability.js","webpack:///./server/routes/api/claimData.js","webpack:///./server/routes/api/claimGet.js","webpack:///./server/routes/api/claimLongId.js","webpack:///./server/routes/api/claimPublish.js","webpack:///./server/auth/authentication.js","webpack:///./server/routes/api/claimResolve.js","webpack:///./server/routes/api/claimShortId.js","webpack:///./server/routes/api/claimList.js","webpack:///./server/routes/api/fileAvailability.js","webpack:///./server/helpers/multipartMiddleware.js","webpack:///external \"connect-multiparty\"","webpack:///./server/routes/pages/index.js","webpack:///./server/routes/pages/sendReactApp.js","webpack:///./server/routes/pages/sendEmbedPage.js","webpack:///./server/routes/pages/redirect.js","webpack:///./server/routes/assets/index.js","webpack:///./server/routes/assets/serveAssetByClaim.js","webpack:///external \"redux-saga\"","webpack:///external \"redux-saga/effects\"","webpack:///./server/routes/assets/serveAssetByIdentifierAndClaim.js","webpack:///./server/routes/fallback/index.js","webpack:///./server/routes/fallback/sendReactApp.js"],"names":["logger","require","module","exports","handleErrorResponse","originalUrl","ip","error","res","useObjectPropertiesIfNoKeys","returnErrorMessageAndStatus","status","message","json","createErrorResponsePayload","code","err","Object","keys","length","newErrorObject","getOwnPropertyNames","forEach","key","success","Certificate","Channel","Claim","File","Request","User","Sequelize","database","username","password","sequelize","host","dialect","dialectOptions","decimalNumbers","logging","pool","max","min","idle","acquire","authenticate","then","info","catch","db","import","modelName","associate","upsert","Model","values","condition","tableName","findOne","where","obj","debug","update","create","SiteConfig","analytics","googleId","assetDefaults","description","thumbnail","title","auth","sessionKey","customComponents","components","containers","pages","details","port","twitter","publishing","additionalClaimAddresses","disabled","disabledMessage","primaryClaimAddress","thumbnailChannel","thumbnailChannelId","uploadDirectory","routes","config","console","log","ua","createServeEventParams","headers","eventCategory","eventAction","eventLabel","ipOverride","userAgentOverride","createPublishTimingEventParams","category","variable","label","startTime","endTime","duration","userTimingCategory","userTimingVariableName","userTimingTime","userTimingLabel","sendGoogleAnalyticsEvent","params","visitorId","replace","visitor","strictCidFormat","https","event","sendGoogleAnalyticsTiming","timing","sendGAServeEvent","sendGATimingEvent","chooseGaLbrynetPublishLabel","channelName","channel_name","channelId","channel_id","axios","api","apiHost","apiPort","lbryApiUri","handleLbrynetResponse","resolve","reject","data","result","Error","JSON","stringify","publishClaim","publishParams","name","gaStartTime","Date","now","Promise","post","method","response","getClaim","uri","timeout","getClaimList","claimName","resolveUri","getDownloadDirectory","download_directory","createChannel","amount","returnPaginatedChannelClaims","NO_CHANNEL","NO_CLAIM","NO_FILE","getClaimId","channelClaimId","claimId","getClaimIdByChannel","getClaimIdByClaim","getLongClaimId","longClaimId","getLongChannelId","longChannelId","all","getClaimIdByLongChannelId","getChannelData","page","longChannelClaimId","getShortChannelIdFromLongChannelId","shortChannelClaimId","getChannelClaims","getAllChannelClaims","channelClaimsArray","paginatedChannelViewData","getLocalFileRecord","file","dataValues","passport","localLoginStrategy","localSignupStrategy","serializeSpeechUser","deserializeSpeechUser","deserializeUser","serializeUser","use","lbryApi","publishHelpers","Op","publish","fileName","fileType","publishResults","certificateId","tx","channel","fileRecord","claim_id","metadata","address","claim_address","outpoint","txid","nout","height","filePath","file_path","nsfw","claimRecord","contentType","bid","upsertCriteria","claim","setClaim","setFile","deleteTemporaryFile","claimNameIsAvailable","claimAddresses","push","findAll","attributes","or","checkChannelAvailability","fs","parsePublishApiRequestBody","license","invalidNameCharacters","exec","parsePublishApiRequestFiles","path","type","size","test","validateFileTypeAndSize","thumbnailFileName","thumbnailFilePath","thumbnailFileType","createBasicPublishParams","trim","author","language","createThumbnailPublishParams","unlink","addGetResultsToFileData","fileInfo","getResult","file_name","download_path","createFileData","mysql","warn","returnShortId","claimsArray","longId","claimIndex","shortId","substring","shortIdLength","findIndex","element","possibleMatches","slice","filter","siteConfig","req","context","customizedReducers","store","html","url","helmet","renderStatic","redirect","preloadedState","getState","send","toString","meta","link","SERVE","SHOW","clientAcceptsHtml","accept","match","requestIsFromBrowser","clientWantsAsset","range","imageIsWanted","videoIsWanted","isValidClaimId","isValidShortId","isValidShortIdOrClaimId","input","serveAssetToClient","verbose","sendFileOptions","sendFile","getClaimIdAndServeAsset","fullClaimId","determineResponseType","hasFileExtension","responseType","flipClaimNameAndIdForBackwardsCompatibility","identifier","tempName","logRequestData","REGEXP_INVALID_CLAIM","REGEXP_INVALID_CHANNEL","REGEXP_ADDRESS","CHANNEL_CHAR","parseIdentifier","componentsRegex","RegExp","map","proto","value","modifierSeperator","modifier","isChannel","startsWith","nameBadChars","join","parseClaim","parseModifier","returnSagaWithParams","saga","sagaMiddleware","middleware","action","onHandleShowPageUri","handleShowPageUri","run","done","Server","express","bodyParser","expressHandlebars","Handlebars","cookieSession","http","requestLogger","Path","loggerConfig","mysqlConfig","slackConfig","configureLogger","userConfig","configureMysql","configureSiteDetails","configureSlack","configureClientBundle","configureModels","configureRoutes","createApp","app","enable","publicFolder","process","cwd","static","publicPath","__dirname","urlencoded","extended","speechPassport","maxAge","initialize","session","hbs","defaultLayout","handlebars","engine","set","server","start","PORT","sync","listen","next","LoggerConfig","logLevel","configure","transports","Console","level","timestamp","colorize","prettyPrint","handleExceptions","humanReadableUnhandledException","silly","winstonSlackWebHook","SlackWebHook","winston","SlackConfig","slackWebHook","slackErrorChannel","slackInfoChannel","add","webhookUrl","iconEmoji","PassportLocalStrategy","Strategy","returnUserAndChannelInfo","userInstance","userInfo","id","userName","getChannel","shortChannelId","usernameField","passwordField","user","comparePassword","isMatch","STRING","BOOLEAN","INTEGER","TEXT","DECIMAL","define","default","claimSequence","decodedClaim","depth","effectiveAmount","hasSignature","hex","validAtHeight","valueVersion","claimType","certificateVersion","keyType","publicKey","freezeTableName","belongsTo","foreignKey","allowNull","order","getLongChannelIdFromShortChannelId","$like","getLongChannelIdFromChannelName","validateLongChannelId","hasOne","defaultThumbnail","determineFileExtensionFromContentType","determineThumbnail","storedThumbnail","prepareClaimData","licenseUrl","preview","metadataVersion","source","sourceType","sourceVersion","streamVersion","getShortClaimIdFromLongClaimId","raw","getLongClaimIdFromShortClaimId","getTopFreeClaimIdByClaimName","validateLongClaimId","resolveClaim","claimArray","defaultValue","trendingEligible","hasMany","getRecentClaims","limit","ipAddress","bcrypt","prototype","compare","changePassword","newPassword","genSalt","saltError","salt","hash","hashError","hook","options","userData","channelData","certificateData","newUser","newChannel","newCertificate","setChannel","setUser","lbryConfig","handleSignupRequest","handleLoginRequest","handleLogoutRequest","handleUserRequest","get","signup","login","logIn","logout","channelAvailability","channelClaims","channelShortId","claimAvailability","claimData","claimGet","claimLongId","claimPublish","claimResolve","claimShortId","claimList","fileAvailability","multipartMiddleware","availableName","body","CLAIMS_PER_PAGE","claims","totalPages","determineTotalPages","paginationPage","getPageFromQuery","viewData","extractPageFromClaims","previousPage","determinePreviousPage","currentPage","nextPage","determineNextPage","totalResults","determineTotalClaims","parseInt","pageNumber","claimStartIndex","claimEndIndex","pageOfClaims","totalClaims","fullPages","Math","floor","remainder","channelShortIdRoute","claimInfo","resolveResult","fileData","completed","authenticateUser","files","channelPassword","validatedClaimName","thumbnailPublishParams","lbryTx","authenticateChannelCredentials","userPassword","channelFindParams","resolvedUri","claimsList","multipart","uploadDir","handlePageRequest","handleEmbedRequest","handlePageRender","sendReactApp","sendEmbedPage","render","layout","route","serveAssetByClaim","serveAssetByIdentifierAndClaim","lbryUri","handleShowRender","serverAssetByClaim","serverAssetByIdentifierAndClaim"],"mappings":"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","file":"index.js","sourcesContent":[" \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId]) {\n \t\t\treturn installedModules[moduleId].exports;\n \t\t}\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\ti: moduleId,\n \t\t\tl: false,\n \t\t\texports: {}\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.l = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// define getter function for harmony exports\n \t__webpack_require__.d = function(exports, name, getter) {\n \t\tif(!__webpack_require__.o(exports, name)) {\n \t\t\tObject.defineProperty(exports, name, {\n \t\t\t\tconfigurable: false,\n \t\t\t\tenumerable: true,\n \t\t\t\tget: getter\n \t\t\t});\n \t\t}\n \t};\n\n \t// getDefaultExport function for compatibility with non-harmony modules\n \t__webpack_require__.n = function(module) {\n \t\tvar getter = module && module.__esModule ?\n \t\t\tfunction getDefault() { return module['default']; } :\n \t\t\tfunction getModuleExports() { return module; };\n \t\t__webpack_require__.d(getter, 'a', getter);\n \t\treturn getter;\n \t};\n\n \t// Object.prototype.hasOwnProperty.call\n \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"/\";\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(__webpack_require__.s = 26);\n\n\n\n// WEBPACK FOOTER //\n// webpack/bootstrap 82ebf2ca30129849e4ef","module.exports = require(\"winston\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"winston\"\n// module id = 0\n// module chunks = 0","const logger = require('winston');\n\nmodule.exports = {\n  handleErrorResponse: function (originalUrl, ip, error, res) {\n    logger.error(`Error on ${originalUrl}`, module.exports.useObjectPropertiesIfNoKeys(error));\n    const [status, message] = module.exports.returnErrorMessageAndStatus(error);\n    res\n      .status(status)\n      .json(module.exports.createErrorResponsePayload(status, message));\n  },\n  returnErrorMessageAndStatus: function (error) {\n    let status, message;\n    // check for daemon being turned off\n    if (error.code === 'ECONNREFUSED') {\n      status = 503;\n      message = 'Connection refused.  The daemon may not be running.';\n      // fallback for everything else\n    } else {\n      status = 400;\n      if (error.message) {\n        message = error.message;\n      } else {\n        message = error;\n      };\n    };\n    return [status, message];\n  },\n  useObjectPropertiesIfNoKeys: function (err) {\n    if (Object.keys(err).length === 0) {\n      let newErrorObject = {};\n      Object.getOwnPropertyNames(err).forEach((key) => {\n        newErrorObject[key] = err[key];\n      });\n      return newErrorObject;\n    }\n    return err;\n  },\n  createErrorResponsePayload (status, message) {\n    return {\n      status,\n      success: false,\n      message,\n    };\n  },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/errorHandlers.js","const Certificate = require('models/certificate.js');\nconst Channel = require('models/channel.js');\nconst Claim = require('models/claim.js');\nconst File = require('models/file.js');\nconst Request = require('models/request.js');\nconst User = require('models/user.js');\n\nconst Sequelize = require('sequelize');\nconst logger = require('winston');\n\nconst {database, username, password} = require('mysqlConfig.js');\n\n// set sequelize options\nconst sequelize = new Sequelize(database, username, password, {\n  host          : 'localhost',\n  dialect       : 'mysql',\n  dialectOptions: {decimalNumbers: true},\n  logging       : false,\n  pool          : {\n    max    : 5,\n    min    : 0,\n    idle   : 10000,\n    acquire: 10000,\n  },\n});\n\n// establish mysql connection\nsequelize\n  .authenticate()\n  .then(() => {\n    logger.info('Sequelize has established mysql connection successfully.');\n  })\n  .catch(err => {\n    logger.error('Sequelize was unable to connect to the database:', err);\n  });\n\n// manually add each model to the db object (note: make this dynamic)\nconst db = {};\ndb['Certificate'] = sequelize.import('Certificate', Certificate);\ndb['Channel'] = sequelize.import('Channel', Channel);\ndb['Claim'] = sequelize.import('Claim', Claim);\ndb['File'] = sequelize.import('File', File);\ndb['Request'] = sequelize.import('Request', Request);\ndb['User'] = sequelize.import('User', User);\n\n// run model.association for each model in the db object that has an association\nlogger.info('associating db models...');\nObject.keys(db).forEach(modelName => {\n  if (db[modelName].associate) {\n    logger.info('Associating model:', modelName);\n    db[modelName].associate(db);\n  }\n});\n\n// add sequelize/Sequelize to db\ndb.sequelize = sequelize;\ndb.Sequelize = Sequelize;\n// add an 'upsert' method to the db object\ndb.upsert = (Model, values, condition, tableName) => {\n  return Model\n    .findOne({\n      where: condition,\n    })\n    .then(obj => {\n      if (obj) {  // update\n        logger.debug(`updating record in db.${tableName}`);\n        return obj.update(values);\n      } else {  // insert\n        logger.debug(`creating record in db.${tableName}`);\n        return Model.create(values);\n      }\n    })\n    .catch(function (error) {\n      logger.error(`${tableName}.upsert error`, error);\n      throw error;\n    });\n};\n\nmodule.exports = db;\n\n\n\n// WEBPACK FOOTER //\n// ./server/models/index.js","function SiteConfig () {\n  this.analytics = {\n    googleId: 'default',\n  };\n  this.assetDefaults = {\n    description: 'An asset published on Spee.ch',\n    thumbnail  : 'https://spee.ch/assets/img/video_thumb_default.png',\n    title      : 'Spee.ch',\n  };\n  this.auth = {\n    sessionKey: 'default',\n  };\n  this.customComponents = {\n    components: {},\n    containers: {},\n    pages     : {},\n  };\n  this.details = {\n    description: 'Open-source, decentralized image and video sharing.',\n    host       : 'default',\n    port       : 3000,\n    title      : 'Spee.ch',\n    twitter    : '@spee_ch',\n  };\n  this.publishing = {\n    additionalClaimAddresses: [],\n    disabled                : false,\n    disabledMessage         : 'Please check back soon.',\n    primaryClaimAddress     : 'default',\n    thumbnailChannel        : 'default',\n    thumbnailChannelId      : 'default',\n    uploadDirectory         : '/home/lbry/Uploads',\n  };\n  this.routes = {};\n  this.update = (config) => {\n    if (!config) {\n      return console.log('No site config received.');\n    }\n    const { analytics, assetDefaults, auth, customComponents, details, publishing, routes } = config;\n    console.log('Configuring site details...');\n    this.analytics = analytics;\n    this.assetDefaults = assetDefaults;\n    this.auth = auth;\n    this.details = details;\n    this.publishing = publishing;\n    this.customComponents = customComponents;\n    this.routes = routes;\n  };\n};\n\nmodule.exports = new SiteConfig();\n\n\n\n// WEBPACK FOOTER //\n// ./config/siteConfig.js","const logger = require('winston');\nconst ua = require('universal-analytics');\nconst { analytics : { googleId }, details: { title } } = require('../../config/siteConfig.js');\n\nfunction createServeEventParams (headers, ip, originalUrl) {\n  return {\n    eventCategory    : 'client requests',\n    eventAction      : 'serve request',\n    eventLabel       : originalUrl,\n    ipOverride       : ip,\n    userAgentOverride: headers['user-agent'],\n  };\n};\n\nfunction createPublishTimingEventParams (category, variable, label, startTime, endTime) {\n  const duration = endTime - startTime;\n  return {\n    userTimingCategory    : category,\n    userTimingVariableName: variable,\n    userTimingTime        : duration,\n    userTimingLabel       : label,\n  };\n};\n\nfunction sendGoogleAnalyticsEvent (ip, params) {\n  const visitorId = ip.replace(/\\./g, '-');\n  const visitor = ua(googleId, visitorId, { strictCidFormat: false, https: true });\n  visitor.event(params, (err) => {\n    if (err) {\n      logger.error('Google Analytics Event Error >>', err);\n    }\n  });\n};\n\nfunction sendGoogleAnalyticsTiming (visitorId, params) {\n  const visitor = ua(googleId, visitorId, { strictCidFormat: false, https: true });\n  visitor.timing(params, (err) => {\n    if (err) {\n      logger.error('Google Analytics Event Error >>', err);\n    }\n    logger.debug(`Timing event successfully sent to google analytics`);\n  });\n};\n\nmodule.exports = {\n  sendGAServeEvent (headers, ip, originalUrl) {\n    const params = createServeEventParams(headers, ip, originalUrl);\n    sendGoogleAnalyticsEvent(ip, params);\n  },\n  sendGATimingEvent (category, variable, label, startTime, endTime) {\n    const params = createPublishTimingEventParams(category, variable, label, startTime, endTime);\n    sendGoogleAnalyticsTiming(title, params);\n  },\n  chooseGaLbrynetPublishLabel ({ channel_name: channelName, channel_id: channelId }) {\n    return (channelName || channelId ? 'PUBLISH_IN_CHANNEL_CLAIM' : 'PUBLISH_ANONYMOUS_CLAIM');\n  },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/googleAnalytics.js","const axios = require('axios');\nconst logger = require('winston');\nconst { api: { apiHost, apiPort } } = require('../../config/lbryConfig.js');\nconst lbryApiUri = 'http://' + apiHost + ':' + apiPort;\nconst { chooseGaLbrynetPublishLabel, sendGATimingEvent } = require('./googleAnalytics.js');\n\nconst handleLbrynetResponse = ({ data }, resolve, reject) => {\n  logger.debug('lbry api data:', data);\n  if (data.result) {\n    // check for an error\n    if (data.result.error) {\n      logger.debug('Lbrynet api error:', data.result.error);\n      reject(new Error(data.result.error));\n      return;\n    };\n    resolve(data.result);\n    return;\n  }\n  // fallback in case it just timed out\n  reject(JSON.stringify(data));\n};\n\nmodule.exports = {\n  publishClaim (publishParams) {\n    logger.debug(`lbryApi >> Publishing claim to \"${publishParams.name}\"`);\n    const gaStartTime = Date.now();\n    return new Promise((resolve, reject) => {\n      axios\n        .post(lbryApiUri, {\n          method: 'publish',\n          params: publishParams,\n        })\n        .then(response => {\n          sendGATimingEvent('lbrynet', 'publish', chooseGaLbrynetPublishLabel(publishParams), gaStartTime, Date.now());\n          handleLbrynetResponse(response, resolve, reject);\n        })\n        .catch(error => {\n          reject(error);\n        });\n    });\n  },\n  getClaim (uri) {\n    logger.debug(`lbryApi >> Getting Claim for \"${uri}\"`);\n    const gaStartTime = Date.now();\n    return new Promise((resolve, reject) => {\n      axios\n        .post(lbryApiUri, {\n          method: 'get',\n          params: { uri, timeout: 20 },\n        })\n        .then(response => {\n          sendGATimingEvent('lbrynet', 'getClaim', 'GET', gaStartTime, Date.now());\n          handleLbrynetResponse(response, resolve, reject);\n        })\n        .catch(error => {\n          reject(error);\n        });\n    });\n  },\n  getClaimList (claimName) {\n    logger.debug(`lbryApi >> Getting claim_list for \"${claimName}\"`);\n    const gaStartTime = Date.now();\n    return new Promise((resolve, reject) => {\n      axios\n        .post(lbryApiUri, {\n          method: 'claim_list',\n          params: { name: claimName },\n        })\n        .then(response => {\n          sendGATimingEvent('lbrynet', 'getClaimList', 'CLAIM_LIST', gaStartTime, Date.now());\n          handleLbrynetResponse(response, resolve, reject);\n        })\n        .catch(error => {\n          reject(error);\n        });\n    });\n  },\n  resolveUri (uri) {\n    logger.debug(`lbryApi >> Resolving URI for \"${uri}\"`);\n    const gaStartTime = Date.now();\n    return new Promise((resolve, reject) => {\n      axios\n        .post(lbryApiUri, {\n          method: 'resolve',\n          params: { uri },\n        })\n        .then(({ data }) => {\n          sendGATimingEvent('lbrynet', 'resolveUri', 'RESOLVE', gaStartTime, Date.now());\n          if (data.result[uri].error) {  // check for errors\n            reject(data.result[uri].error);\n          } else {  // if no errors, resolve\n            resolve(data.result[uri]);\n          }\n        })\n        .catch(error => {\n          reject(error);\n        });\n    });\n  },\n  getDownloadDirectory () {\n    logger.debug('lbryApi >> Retrieving the download directory path from lbry daemon...');\n    const gaStartTime = Date.now();\n    return new Promise((resolve, reject) => {\n      axios\n        .post(lbryApiUri, {\n          method: 'settings_get',\n        })\n        .then(({ data }) => {\n          sendGATimingEvent('lbrynet', 'getDownloadDirectory', 'SETTINGS_GET', gaStartTime, Date.now());\n          if (data.result) {\n            resolve(data.result.download_directory);\n          } else {\n            return new Error('Successfully connected to lbry daemon, but unable to retrieve the download directory.');\n          }\n        })\n        .catch(error => {\n          logger.error('Lbrynet Error:', error);\n          resolve('/home/lbry/Downloads/');\n        });\n    });\n  },\n  createChannel (name) {\n    logger.debug(`lbryApi >> Creating channel for ${name}...`);\n    const gaStartTime = Date.now();\n    return new Promise((resolve, reject) => {\n      axios\n        .post(lbryApiUri, {\n          method: 'channel_new',\n          params: {\n            channel_name: name,\n            amount      : 0.1,\n          },\n        })\n        .then(response => {\n          sendGATimingEvent('lbrynet', 'createChannel', 'CHANNEL_NEW', gaStartTime, Date.now());\n          handleLbrynetResponse(response, resolve, reject);\n        })\n        .catch(error => {\n          reject(error);\n        });\n    });\n  },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/lbryApi.js","const db = require('models');\nconst logger = require('winston');\nconst { returnPaginatedChannelClaims } = require('helpers/channelPagination.js');\n\nconst NO_CHANNEL = 'NO_CHANNEL';\nconst NO_CLAIM = 'NO_CLAIM';\nconst NO_FILE = 'NO_FILE';\n\nmodule.exports = {\n  getClaimId (channelName, channelClaimId, name, claimId) {\n    if (channelName) {\n      return module.exports.getClaimIdByChannel(channelName, channelClaimId, name);\n    } else {\n      return module.exports.getClaimIdByClaim(name, claimId);\n    }\n  },\n  getClaimIdByClaim (claimName, claimId) {\n    logger.debug(`getClaimIdByClaim(${claimName}, ${claimId})`);\n    return new Promise((resolve, reject) => {\n      db.Claim.getLongClaimId(claimName, claimId)\n        .then(longClaimId => {\n          if (!longClaimId) {\n            resolve(NO_CLAIM);\n          }\n          resolve(longClaimId);\n        })\n        .catch(error => {\n          reject(error);\n        });\n    });\n  },\n  getClaimIdByChannel (channelName, channelClaimId, claimName) {\n    logger.debug(`getClaimIdByChannel(${channelName}, ${channelClaimId}, ${claimName})`);\n    return new Promise((resolve, reject) => {\n      db.Certificate.getLongChannelId(channelName, channelClaimId) // 1. get the long channel id\n        .then(longChannelId => {\n          if (!longChannelId) {\n            return [null, null];\n          }\n          return Promise.all([longChannelId, db.Claim.getClaimIdByLongChannelId(longChannelId, claimName)]);  // 2. get the long claim id\n        })\n        .then(([longChannelId, longClaimId]) => {\n          if (!longChannelId) {\n            return resolve(NO_CHANNEL);\n          }\n          if (!longClaimId) {\n            return resolve(NO_CLAIM);\n          }\n          resolve(longClaimId);\n        })\n        .catch(error => {\n          reject(error);\n        });\n    });\n  },\n  getChannelData (channelName, channelClaimId, page) {\n    return new Promise((resolve, reject) => {\n      // 1. get the long channel Id (make sure channel exists)\n      db.Certificate.getLongChannelId(channelName, channelClaimId)\n        .then(longChannelClaimId => {\n          if (!longChannelClaimId) {\n            return [null, null, null];\n          }\n          // 2. get the short ID and all claims for that channel\n          return Promise.all([longChannelClaimId, db.Certificate.getShortChannelIdFromLongChannelId(longChannelClaimId, channelName)]);\n        })\n        .then(([longChannelClaimId, shortChannelClaimId]) => {\n          if (!longChannelClaimId) {\n            return resolve(NO_CHANNEL);\n          }\n          // 3. return all the channel information\n          resolve({\n            channelName,\n            longChannelClaimId,\n            shortChannelClaimId,\n          });\n        })\n        .catch(error => {\n          reject(error);\n        });\n    });\n  },\n  getChannelClaims (channelName, channelClaimId, page) {\n    return new Promise((resolve, reject) => {\n      // 1. get the long channel Id (make sure channel exists)\n      db.Certificate.getLongChannelId(channelName, channelClaimId)\n        .then(longChannelClaimId => {\n          if (!longChannelClaimId) {\n            return [null, null, null];\n          }\n          // 2. get the short ID and all claims for that channel\n          return Promise.all([longChannelClaimId, db.Claim.getAllChannelClaims(longChannelClaimId)]);\n        })\n        .then(([longChannelClaimId, channelClaimsArray]) => {\n          if (!longChannelClaimId) {\n            return resolve(NO_CHANNEL);\n          }\n          // 3. format the data for the view, including pagination\n          let paginatedChannelViewData = returnPaginatedChannelClaims(channelName, longChannelClaimId, channelClaimsArray, page);\n          // 4. return all the channel information and contents\n          resolve(paginatedChannelViewData);\n        })\n        .catch(error => {\n          reject(error);\n        });\n    });\n  },\n  getLocalFileRecord (claimId, name) {\n    return db.File.findOne({where: {claimId, name}})\n      .then(file => {\n        if (!file) {\n          return NO_FILE;\n        }\n        return file.dataValues;\n      });\n  },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/controllers/serveController.js","const passport = require('passport');\nconst localLoginStrategy = require('./local-login.js');\nconst localSignupStrategy = require('./local-signup.js');\nconst { serializeSpeechUser, deserializeSpeechUser } = require('helpers/authHelpers.js');\n\npassport.deserializeUser(deserializeSpeechUser);\npassport.serializeUser(serializeSpeechUser);\npassport.use('local-login', localLoginStrategy);\npassport.use('local-signup', localSignupStrategy);\n\nmodule.exports = passport;\n\n\n\n// WEBPACK FOOTER //\n// ./server/speechPassport/index.js","const logger = require('winston');\nconst db = require('models');\nconst lbryApi = require('helpers/lbryApi.js');\nconst publishHelpers = require('helpers/publishHelpers.js');\nconst { publishing: { primaryClaimAddress, additionalClaimAddresses } } = require('siteConfig.js');\nconst Sequelize = require('sequelize');\nconst Op = Sequelize.Op;\n\nmodule.exports = {\n  publish (publishParams, fileName, fileType) {\n    return new Promise((resolve, reject) => {\n      let publishResults, certificateId, channelName;\n      // publish the file\n      return lbryApi.publishClaim(publishParams)\n        .then(tx => {\n          logger.info(`Successfully published ${publishParams.name} ${fileName}`, tx);\n          publishResults = tx;\n          // get the channel information\n          if (publishParams.channel_name) {\n            logger.debug(`this claim was published in channel: ${publishParams.channel_name}`);\n            return db.Channel.findOne({\n              where: {\n                channelName: publishParams.channel_name,\n              },\n            });\n          } else {\n            logger.debug('this claim was not published in a channel');\n            return null;\n          }\n        })\n        .then(channel => {\n        // set channel information\n          certificateId = null;\n          channelName = null;\n          if (channel) {\n            certificateId = channel.channelClaimId;\n            channelName = channel.channelName;\n          }\n          logger.debug(`certificateId: ${certificateId}`);\n        })\n        .then(() => {\n        // create the File record\n          const fileRecord = {\n            name       : publishParams.name,\n            claimId    : publishResults.claim_id,\n            title      : publishParams.metadata.title,\n            description: publishParams.metadata.description,\n            address    : publishParams.claim_address,\n            outpoint   : `${publishResults.txid}:${publishResults.nout}`,\n            height     : 0,\n            fileName,\n            filePath   : publishParams.file_path,\n            fileType,\n            nsfw       : publishParams.metadata.nsfw,\n          };\n          // create the Claim record\n          const claimRecord = {\n            name       : publishParams.name,\n            claimId    : publishResults.claim_id,\n            title      : publishParams.metadata.title,\n            description: publishParams.metadata.description,\n            address    : publishParams.claim_address,\n            thumbnail  : publishParams.metadata.thumbnail,\n            outpoint   : `${publishResults.txid}:${publishResults.nout}`,\n            height     : 0,\n            contentType: fileType,\n            nsfw       : publishParams.metadata.nsfw,\n            amount     : publishParams.bid,\n            certificateId,\n            channelName,\n          };\n          // upsert criteria\n          const upsertCriteria = {\n            name   : publishParams.name,\n            claimId: publishResults.claim_id,\n          };\n          // upsert the records\n          return Promise.all([db.upsert(db.File, fileRecord, upsertCriteria, 'File'), db.upsert(db.Claim, claimRecord, upsertCriteria, 'Claim')]);\n        })\n        .then(([file, claim]) => {\n          logger.debug('File and Claim records successfully created');\n          return Promise.all([file.setClaim(claim), claim.setFile(file)]);\n        })\n        .then(() => {\n          logger.debug('File and Claim records successfully associated');\n          resolve(publishResults); // resolve the promise with the result from lbryApi.publishClaim;\n        })\n        .catch(error => {\n          logger.error('PUBLISH ERROR', error);\n          publishHelpers.deleteTemporaryFile(publishParams.file_path); // delete the local file\n          reject(error);\n        });\n    });\n  },\n  claimNameIsAvailable (name) {\n    const claimAddresses = additionalClaimAddresses || [];\n    claimAddresses.push(primaryClaimAddress);\n    // find any records where the name is used\n    return db.Claim\n      .findAll({\n        attributes: ['address'],\n        where     : {\n          name,\n          address: {\n            [Op.or]: claimAddresses,\n          },\n        },\n      })\n      .then(result => {\n        if (result.length >= 1) {\n          throw new Error('That claim is already in use');\n        };\n        return name;\n      })\n      .catch(error => {\n        throw error;\n      });\n  },\n  checkChannelAvailability (name) {\n    return db.Channel\n      .findAll({\n        where: { channelName: name },\n      })\n      .then(result => {\n        if (result.length >= 1) {\n          throw new Error('That channel has already been claimed');\n        }\n        return name;\n      })\n      .catch(error => {\n        throw error;\n      });\n  },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/controllers/publishController.js","const logger = require('winston');\nconst fs = require('fs');\n\nconst { details, publishing } = require('../../config/siteConfig.js');\n\nmodule.exports = {\n  parsePublishApiRequestBody ({name, nsfw, license, title, description, thumbnail}) {\n    // validate name\n    if (!name) {\n      throw new Error('no name field found in request');\n    }\n    const invalidNameCharacters = /[^A-Za-z0-9,-]/.exec(name);\n    if (invalidNameCharacters) {\n      throw new Error('The claim name you provided is not allowed.  Only the following characters are allowed: A-Z, a-z, 0-9, and \"-\"');\n    }\n    // optional parameters\n    nsfw = (nsfw === 'true');\n    license = license || null;\n    title = title || null;\n    description = description || null;\n    thumbnail = thumbnail || null;\n    // return results\n    return {\n      name,\n      nsfw,\n      license,\n      title,\n      description,\n      thumbnail,\n    };\n  },\n  parsePublishApiRequestFiles ({file, thumbnail}) {\n    // make sure a file was provided\n    if (!file) {\n      throw new Error('no file with key of [file] found in request');\n    }\n    if (!file.path) {\n      throw new Error('no file path found');\n    }\n    if (!file.type) {\n      throw new Error('no file type found');\n    }\n    if (!file.size) {\n      throw new Error('no file type found');\n    }\n    // validate the file name\n    if (/'/.test(file.name)) {\n      throw new Error('apostrophes are not allowed in the file name');\n    }\n    // validate the file\n    module.exports.validateFileTypeAndSize(file);\n    // return results\n    return {\n      fileName         : file.name,\n      filePath         : file.path,\n      fileType         : file.type,\n      thumbnailFileName: (thumbnail ? thumbnail.name : null),\n      thumbnailFilePath: (thumbnail ? thumbnail.path : null),\n      thumbnailFileType: (thumbnail ? thumbnail.type : null),\n    };\n  },\n  validateFileTypeAndSize (file) {\n    // check file type and size\n    switch (file.type) {\n      case 'image/jpeg':\n      case 'image/jpg':\n      case 'image/png':\n        if (file.size > 10000000) {\n          logger.debug('publish > file validation > .jpeg/.jpg/.png was too big');\n          throw new Error('Sorry, images are limited to 10 megabytes.');\n        }\n        break;\n      case 'image/gif':\n        if (file.size > 50000000) {\n          logger.debug('publish > file validation > .gif was too big');\n          throw new Error('Sorry, .gifs are limited to 50 megabytes.');\n        }\n        break;\n      case 'video/mp4':\n        if (file.size > 50000000) {\n          logger.debug('publish > file validation > .mp4 was too big');\n          throw new Error('Sorry, videos are limited to 50 megabytes.');\n        }\n        break;\n      default:\n        logger.debug('publish > file validation > unrecognized file type');\n        throw new Error('The ' + file.type + ' content type is not supported.  Only, .jpeg, .png, .gif, and .mp4 files are currently supported.');\n    }\n    return file;\n  },\n  createBasicPublishParams (filePath, name, title, description, license, nsfw, thumbnail) {\n    logger.debug(`Creating Publish Parameters`);\n    // provide defaults for title\n    if (title === null || title.trim() === '') {\n      title = name;\n    }\n    // provide default for description\n    if (description === null || description.trim() === '') {\n      description = '';\n    }\n    // provide default for license\n    if (license === null || license.trim() === '') {\n      license = ' ';  // default to empty string\n    }\n    // create the publish params\n    const publishParams = {\n      name,\n      file_path: filePath,\n      bid      : 0.01,\n      metadata : {\n        description,\n        title,\n        author  : details.title,\n        language: 'en',\n        license,\n        nsfw,\n      },\n      claim_address: publishing.primaryClaimAddress,\n    };\n    // add thumbnail to channel if video\n    if (thumbnail) {\n      publishParams['metadata']['thumbnail'] = thumbnail;\n    }\n    return publishParams;\n  },\n  createThumbnailPublishParams (thumbnailFilePath, claimName, license, nsfw) {\n    if (!thumbnailFilePath) {\n      return;\n    }\n    logger.debug(`Creating Thumbnail Publish Parameters`);\n    // create the publish params\n    return {\n      name     : `${claimName}-thumb`,\n      file_path: thumbnailFilePath,\n      bid      : 0.01,\n      metadata : {\n        title      : `${claimName} thumbnail`,\n        description: `a thumbnail for ${claimName}`,\n        author     : details.title,\n        language   : 'en',\n        license,\n        nsfw,\n      },\n      claim_address: publishing.primaryClaimAddress,\n      channel_name : publishing.thumbnailChannel,\n      channel_id   : publishing.thumbnailChannelId,\n    };\n  },\n  deleteTemporaryFile (filePath) {\n    fs.unlink(filePath, err => {\n      if (err) {\n        logger.error(`error deleting temporary file ${filePath}`);\n        throw err;\n      }\n      logger.debug(`successfully deleted ${filePath}`);\n    });\n  },\n  addGetResultsToFileData (fileInfo, getResult) {\n    fileInfo.fileName = getResult.file_name;\n    fileInfo.filePath = getResult.download_path;\n    return fileInfo;\n  },\n  createFileData ({ name, claimId, outpoint, height, address, nsfw, contentType }) {\n    return {\n      name,\n      claimId,\n      outpoint,\n      height,\n      address,\n      fileName: '',\n      filePath: '',\n      fileType: contentType,\n      nsfw,\n    };\n  },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/publishHelpers.js","const logger = require('winston');\n\nfunction mysql () {\n  this.database = 'default';\n  this.username = 'default';\n  this.password = 'default';\n  this.update = (config) => {\n    if (!config) {\n      return logger.warn('No MySQL config received.');\n    }\n    // configure credentials\n    logger.info('configuring mysql...');\n    const { database, username, password } = config;\n    this.database = database;\n    this.username = username;\n    this.password = password;\n  };\n};\n\nmodule.exports = new mysql();\n\n\n\n// WEBPACK FOOTER //\n// ./config/mysqlConfig.js","module.exports = require(\"passport-local\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"passport-local\"\n// module id = 11\n// module chunks = 0","module.exports = {\n  returnShortId: function (claimsArray, longId) {\n    let claimIndex;\n    let shortId = longId.substring(0, 1); // default short id is the first letter\n    let shortIdLength = 0;\n    // find the index of this claim id\n    claimIndex = claimsArray.findIndex(element => {\n      return element.claimId === longId;\n    });\n    if (claimIndex < 0) {\n      throw new Error('claim id not found in claims list');\n    }\n    // get an array of all claims with lower height\n    let possibleMatches = claimsArray.slice(0, claimIndex);\n    // remove certificates with the same prefixes until none are left.\n    while (possibleMatches.length > 0) {\n      shortIdLength += 1;\n      shortId = longId.substring(0, shortIdLength);\n      possibleMatches = possibleMatches.filter(element => {\n        return (element.claimId && (element.claimId.substring(0, shortIdLength) === shortId));\n      });\n    }\n    return shortId;\n  },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/sequelizeHelpers.js","module.exports = require(\"sequelize\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"sequelize\"\n// module id = 13\n// module chunks = 0","import React from 'react';\nimport { renderToString } from 'react-dom/server';\nimport { createStore } from 'redux';\nimport { Provider } from 'react-redux';\nimport { StaticRouter } from 'react-router-dom';\nimport { Reducers, GAListener, App } from 'spee.ch-components';\nimport renderFullPage from './renderFullPage.js';\nimport Helmet from 'react-helmet';\n\nconst siteConfig = require('siteConfig.js');\n\nmodule.exports = (req, res) => {\n  let context = {};\n\n  // customize the reducer by passing in intial state configs\n  const customizedReducers = Reducers(siteConfig);\n\n  // create a new Redux store instance\n  const store = createStore(customizedReducers);\n\n  // render component to a string\n  const html = renderToString(\n    <Provider store={store}>\n      <StaticRouter location={req.url} context={context}>\n        <GAListener>\n          <App />\n        </GAListener>\n      </StaticRouter>\n    </Provider>\n  );\n\n  // get head tags from helmet\n  const helmet = Helmet.renderStatic();\n\n  // check for a redirect\n  if (context.url) {\n    // Somewhere a `<Redirect>` was rendered\n    return res.redirect(301, context.url);\n  } else {\n    // we're good, send the response\n  }\n\n  // get the initial state from our Redux store\n  const preloadedState = store.getState();\n\n  // send the rendered page back to the client\n  res.send(renderFullPage(helmet, html, preloadedState));\n\n  console.log('hello from spee.ch handlePageRender.jsx');\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/handlePageRender.jsx","module.exports = require(\"react\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"react\"\n// module id = 15\n// module chunks = 0","module.exports = require(\"react-dom/server\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"react-dom/server\"\n// module id = 16\n// module chunks = 0","module.exports = require(\"redux\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"redux\"\n// module id = 17\n// module chunks = 0","module.exports = require(\"react-redux\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"react-redux\"\n// module id = 18\n// module chunks = 0","module.exports = require(\"react-router-dom\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"react-router-dom\"\n// module id = 19\n// module chunks = 0","module.exports = require(\"spee.ch-components\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"spee.ch-components\"\n// module id = 20\n// module chunks = 0","module.exports = (helmet, html, preloadedState) => {\n  // take the html and preloadedState and return the full page\n  return `\n    <!DOCTYPE html>\n    <html lang=\"en\" prefix=\"og: http://ogp.me/ns# fb: http://ogp.me/ns/fb#\">\n        <head>\n            <meta charset=\"UTF-8\">\n            <meta name=\"viewport\" content=\"width=device-width, initial-scale=1.0, maximum-scale=1, user-scalable=no\">\n            <meta http-equiv=\"X-UA-Compatible\" content=\"ie=edge\">\n            <!--helmet-->\n            ${helmet.title.toString()}\n            ${helmet.meta.toString()}\n            ${helmet.link.toString()}\n            <!--style sheets-->\n            <link rel=\"stylesheet\" href=\"/static/assets/css/reset.css\" type=\"text/css\">\n            <link rel=\"stylesheet\" href=\"/static/assets/css/general.css\" type=\"text/css\">\n            <link rel=\"stylesheet\" href=\"/static/assets/css/mediaQueries.css\" type=\"text/css\">\n            <!--google font-->\n            <link href=\"https://fonts.googleapis.com/css?family=Roboto:300\" rel=\"stylesheet\">\n        </head>\n        <body id=\"main-body\">\n            <div class=\"row row--tall flex-container--column\">\n                <div id=\"react-app\" class=\"row row--tall flex-container--column\">${html}</div>\n            </div>\n            <script>\n                window.__PRELOADED_STATE__ = ${JSON.stringify(preloadedState).replace(/</g, '\\\\\\u003c')}\n            </script>\n            <script src=\"/static/bundle/bundle.js\"></script>\n        </body>\n    </html>\n  `;\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/renderFullPage.js","module.exports = require(\"react-helmet\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"react-helmet\"\n// module id = 22\n// module chunks = 0","const logger = require('winston');\nconst { getClaimId, getLocalFileRecord } = require('../controllers/serveController.js');\nconst { handleErrorResponse } = require('./errorHandlers.js');\n\nconst SERVE = 'SERVE';\nconst SHOW = 'SHOW';\nconst NO_FILE = 'NO_FILE';\nconst NO_CHANNEL = 'NO_CHANNEL';\nconst NO_CLAIM = 'NO_CLAIM';\n\nfunction clientAcceptsHtml ({accept}) {\n  return accept && accept.match(/text\\/html/);\n};\n\nfunction requestIsFromBrowser (headers) {\n  return headers['user-agent'] && headers['user-agent'].match(/Mozilla/);\n};\n\nfunction clientWantsAsset ({accept, range}) {\n  const imageIsWanted = accept && accept.match(/image\\/.*/) && !accept.match(/text\\/html/) && !accept.match(/text\\/\\*/);\n  const videoIsWanted = accept && range;\n  return imageIsWanted || videoIsWanted;\n};\n\nfunction isValidClaimId (claimId) {\n  return ((claimId.length === 40) && !/[^A-Za-z0-9]/g.test(claimId));\n};\n\nfunction isValidShortId (claimId) {\n  return claimId.length === 1;  // it should really evaluate the short url itself\n};\n\nfunction isValidShortIdOrClaimId (input) {\n  return (isValidClaimId(input) || isValidShortId(input));\n};\n\nfunction serveAssetToClient (claimId, name, res) {\n  return getLocalFileRecord(claimId, name)\n    .then(fileRecord => {\n      // check that a local record was found\n      if (fileRecord === NO_FILE) {\n        return res.status(307).redirect(`/api/claim/get/${name}/${claimId}`);\n      }\n      // serve the file\n      const {filePath, fileType} = fileRecord;\n      logger.verbose(`serving file: ${filePath}`);\n      const sendFileOptions = {\n        headers: {\n          'X-Content-Type-Options': 'nosniff',\n          'Content-Type'          : fileType || 'image/jpeg',\n        },\n      };\n      res.status(200).sendFile(filePath, sendFileOptions);\n    })\n    .catch(error => {\n      throw error;\n    });\n};\n\nmodule.exports = {\n  getClaimIdAndServeAsset (channelName, channelClaimId, claimName, claimId, originalUrl, ip, res) {\n    // get the claim Id and then serve the asset\n    getClaimId(channelName, channelClaimId, claimName, claimId)\n      .then(fullClaimId => {\n        if (fullClaimId === NO_CLAIM) {\n          return res.status(404).json({success: false, message: 'no claim id could be found'});\n        } else if (fullClaimId === NO_CHANNEL) {\n          return res.status(404).json({success: false, message: 'no channel id could be found'});\n        }\n        serveAssetToClient(fullClaimId, claimName, res);\n        // postToStats(responseType, originalUrl, ip, claimName, fullClaimId, 'success');\n      })\n      .catch(error => {\n        handleErrorResponse(originalUrl, ip, error, res);\n        // postToStats(responseType, originalUrl, ip, claimName, fullClaimId, 'fail');\n      });\n  },\n  determineResponseType (hasFileExtension, headers) {\n    let responseType;\n    if (hasFileExtension) {\n      responseType = SERVE;  // assume a serve request if file extension is present\n      if (clientAcceptsHtml(headers)) {  // if the request comes from a browser, change it to a show request\n        responseType = SHOW;\n      }\n    } else {\n      responseType = SHOW;\n      if (clientWantsAsset(headers) && requestIsFromBrowser(headers)) {  // this is in case someone embeds a show url\n        logger.debug('Show request came from browser but wants an image/video. Changing response to serve...');\n        responseType = SERVE;\n      }\n    }\n    return responseType;\n  },\n  flipClaimNameAndIdForBackwardsCompatibility (identifier, name) {\n    // this is a patch for backwards compatability with '/name/claim_id' url format\n    if (isValidShortIdOrClaimId(name) && !isValidShortIdOrClaimId(identifier)) {\n      const tempName = name;\n      name = identifier;\n      identifier = tempName;\n    }\n    return [identifier, name];\n  },\n  logRequestData (responseType, claimName, channelName, claimId) {\n    logger.debug('responseType ===', responseType);\n    logger.debug('claim name === ', claimName);\n    logger.debug('channel name ===', channelName);\n    logger.debug('claim id ===', claimId);\n  },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/serveHelpers.js","const logger = require('winston');\n\nmodule.exports = {\n  REGEXP_INVALID_CLAIM  : /[^A-Za-z0-9-]/g,\n  REGEXP_INVALID_CHANNEL: /[^A-Za-z0-9-@]/g,\n  REGEXP_ADDRESS        : /^b(?=[^0OIl]{32,33})[0-9A-Za-z]{32,33}$/,\n  CHANNEL_CHAR          : '@',\n  parseIdentifier       : function (identifier) {\n    logger.debug('parsing identifier:', identifier);\n    const componentsRegex = new RegExp(\n      '([^:$#/]*)' + // value (stops at the first separator or end)\n      '([:$#]?)([^/]*)' // modifier separator, modifier (stops at the first path separator or end)\n    );\n    const [proto, value, modifierSeperator, modifier] = componentsRegex\n      .exec(identifier)\n      .map(match => match || null);\n    logger.debug(`${proto}, ${value}, ${modifierSeperator}, ${modifier}`);\n\n    // Validate and process name\n    if (!value) {\n      throw new Error(`Check your url.  No channel name provided before \"${modifierSeperator}\"`);\n    }\n    const isChannel = value.startsWith(module.exports.CHANNEL_CHAR);\n    const channelName = isChannel ? value : null;\n    let claimId;\n    if (isChannel) {\n      if (!channelName) {\n        throw new Error('No channel name after @.');\n      }\n      const nameBadChars = (channelName).match(module.exports.REGEXP_INVALID_CHANNEL);\n      if (nameBadChars) {\n        throw new Error(`Invalid characters in channel name: ${nameBadChars.join(', ')}.`);\n      }\n    } else {\n      claimId = value;\n    }\n\n    // Validate and process modifier\n    let channelClaimId;\n    if (modifierSeperator) {\n      if (!modifier) {\n        throw new Error(`No modifier provided after separator \"${modifierSeperator}\"`);\n      }\n\n      if (modifierSeperator === ':') {\n        channelClaimId = modifier;\n      } else {\n        throw new Error(`The \"${modifierSeperator}\" modifier is not currently supported`);\n      }\n    }\n    return {\n      isChannel,\n      channelName,\n      channelClaimId,\n      claimId,\n    };\n  },\n  parseClaim: function (claim) {\n    logger.debug('parsing name:', claim);\n    const componentsRegex = new RegExp(\n      '([^:$#/.]*)' + // name (stops at the first modifier)\n      '([:$#.]?)([^/]*)' // modifier separator, modifier (stops at the first path separator or end)\n    );\n    const [proto, claimName, modifierSeperator, modifier] = componentsRegex\n      .exec(claim)\n      .map(match => match || null);\n    logger.debug(`${proto}, ${claimName}, ${modifierSeperator}, ${modifier}`);\n\n    // Validate and process name\n    if (!claimName) {\n      throw new Error('No claim name provided before .');\n    }\n    const nameBadChars = (claimName).match(module.exports.REGEXP_INVALID_CLAIM);\n    if (nameBadChars) {\n      throw new Error(`Invalid characters in claim name: ${nameBadChars.join(', ')}.`);\n    }\n    // Validate and process modifier\n    if (modifierSeperator) {\n      if (!modifier) {\n        throw new Error(`No file extension provided after separator ${modifierSeperator}.`);\n      }\n      if (modifierSeperator !== '.') {\n        throw new Error(`The ${modifierSeperator} modifier is not supported in the claim name`);\n      }\n    }\n    // return results\n    return {\n      claimName,\n    };\n  },\n  parseModifier: function (claim) {\n    logger.debug('parsing modifier:', claim);\n    const componentsRegex = new RegExp(\n      '([^:$#/.]*)' + // name (stops at the first modifier)\n      '([:$#.]?)([^/]*)' // modifier separator, modifier (stops at the first path separator or end)\n    );\n    const [proto, claimName, modifierSeperator, modifier] = componentsRegex\n      .exec(claim)\n      .map(match => match || null);\n    logger.debug(`${proto}, ${claimName}, ${modifierSeperator}, ${modifier}`);\n    // Validate and process modifier\n    let hasFileExtension = false;\n    if (modifierSeperator) {\n      hasFileExtension = true;\n    }\n    return {\n      hasFileExtension,\n    };\n  },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/lbryUri.js","import React from 'react';\nimport { renderToString } from 'react-dom/server';\nimport { createStore, applyMiddleware } from 'redux';\nimport { Provider } from 'react-redux';\nimport { StaticRouter } from 'react-router-dom';\nimport renderFullPage from './renderFullPage';\nimport createSagaMiddleware from 'redux-saga';\nimport { call } from 'redux-saga/effects';\nimport { Reducers, GAListener, App, Sagas, Actions } from 'spee.ch-components';\n\nimport Helmet from 'react-helmet';\n\n// configure the reducers by passing initial state configs\nconst siteConfig = require('siteConfig.js');\nconst customizedReducers = Reducers(siteConfig);\n\nconst returnSagaWithParams = (saga, params) => {\n  return function * () {\n    yield call(saga, params);\n  };\n};\n\nmodule.exports = (req, res) => {\n  let context = {};\n\n  // create and apply middleware\n  const sagaMiddleware = createSagaMiddleware();\n  const middleware = applyMiddleware(sagaMiddleware);\n\n  // create a new Redux store instance\n  const store = createStore(customizedReducers, middleware);\n\n  // create saga\n  const action = Actions.onHandleShowPageUri(req.params);\n  const saga = returnSagaWithParams(Sagas.handleShowPageUri, action);\n\n  // run the saga middleware\n  sagaMiddleware\n    .run(saga)\n    .done\n    .then(() => {\n      // render component to a string\n      const html = renderToString(\n        <Provider store={store}>\n          <StaticRouter location={req.url} context={context}>\n            <GAListener>\n              <App />\n            </GAListener>\n          </StaticRouter>\n        </Provider>\n      );\n\n      // get head tags from helmet\n      const helmet = Helmet.renderStatic();\n\n      // check for a redirect\n      if (context.url) {\n        return res.redirect(301, context.url);\n      }\n\n      // get the initial state from our Redux store\n      const preloadedState = store.getState();\n\n      // send the rendered page back to the client\n      res.send(renderFullPage(helmet, html, preloadedState));\n    });\n\n  console.log('hello from spee.ch handleShowRender.jsx');\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/handleShowRender.jsx","module.exports = require(\"babel-polyfill\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"babel-polyfill\"\n// module id = 27\n// module chunks = 0","module.exports = require(\"whatwg-fetch\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"whatwg-fetch\"\n// module id = 28\n// module chunks = 0","const Server = require('./server');\n\nconst exports = {\n  Server,\n};\n\nmodule.exports = exports;\n\n\n\n// WEBPACK FOOTER //\n// ./speech.js","// app dependencies\nconst express = require('express');\nconst bodyParser = require('body-parser');\nconst expressHandlebars = require('express-handlebars');\nconst Handlebars = require('handlebars');\nconst helmet = require('helmet');\nconst cookieSession = require('cookie-session');\nconst http = require('http');\nconst logger = require('winston');\nconst requestLogger = require('middleware/requestLogger.js');\nconst Path = require('path');\nconst loggerConfig = require('loggerConfig.js');\nconst mysqlConfig = require('mysqlConfig.js');\nconst siteConfig = require('siteConfig.js');\nconst slackConfig = require('slackConfig.js');\n\nfunction Server () {\n  this.configureLogger = (userConfig) => {\n    loggerConfig.update(userConfig);\n  };\n  this.configureMysql = (userConfig) => {\n    mysqlConfig.update(userConfig);\n  };\n  this.configureSiteDetails = (userConfig) => {\n    siteConfig.update(userConfig);\n  };\n  this.configureSlack = (userConfig) => {\n    slackConfig.update(userConfig);\n  };\n  this.configureClientBundle = () => {\n    logger.debug('configure the client here by passing in the bundle and configuring it, or better yet: taking in the components to use dynamically from here.');\n  };\n  this.configureModels = () => {\n    logger.debug('here is where you could add/overwrite the default models')\n  };\n  this.configureRoutes = () => {\n    logger.debug('here is where you could add/overwrite the default routes')\n  };\n  this.createApp = () => {\n    // create an Express application\n    const app = express();\n\n    // trust the proxy to get ip address for us\n    app.enable('trust proxy');\n\n    /* add middleware */\n    // set HTTP headers to protect against well-known web vulnerabilties\n    app.use(helmet());\n    // 'express.static' to serve static files from public directory\n    if (siteConfig.routes.publicFolder) {\n      // take in a different public folder, so it can serve it's own bundle if needed\n      const publicFolder = Path.resolve(process.cwd(), siteConfig.routes.publicFolder);\n      app.use('/static', express.static(publicFolder));\n      logger.info('serving static files from custom path:', publicFolder);\n    } else {\n      const publicPath = Path.resolve(__dirname, 'public');\n      app.use('/static', express.static(publicPath));\n      logger.info('serving static files from default path:', publicPath);\n    };\n    // 'body parser' for parsing application/json\n    app.use(bodyParser.json());\n    // 'body parser' for parsing application/x-www-form-urlencoded\n    app.use(bodyParser.urlencoded({ extended: true }));\n\n    // add custom middleware (note: build out to accept dynamically use what is in server/middleware/\n    app.use(requestLogger);\n\n    // configure passport\n    const speechPassport = require('speechPassport');\n    // initialize passport\n    const sessionKey = siteConfig.auth.sessionKey;\n    app.use(cookieSession({\n      name  : 'session',\n      keys  : [sessionKey],\n      maxAge: 24 * 60 * 60 * 1000, // i.e. 24 hours\n    }));\n    app.use(speechPassport.initialize());\n    app.use(speechPassport.session());\n\n    // configure handlebars & register it with express app\n    const hbs = expressHandlebars.create({\n      defaultLayout: 'embed',\n      handlebars   : Handlebars,\n    });\n    app.engine('handlebars', hbs.engine);\n    app.set('view engine', 'handlebars');\n\n    // set the routes on the app\n    require('./routes/auth/')(app);\n    require('./routes/api/')(app);\n    require('./routes/pages/')(app);\n    require('./routes/assets/')(app);\n    require('./routes/fallback/')(app);\n\n    this.app = app;\n  };\n  this.initialize = () => {\n    this.createApp();\n    this.server = http.Server(this.app);\n  };\n  this.start = () => {\n    const db = require('models');\n    const PORT = siteConfig.details.port;\n    // sync sequelize\n    db.sequelize.sync()\n    // start the server\n      .then(() => {\n        this.server.listen(PORT, () => {\n          logger.info(`Server is listening on PORT ${PORT}`);\n        });\n      })\n      .catch((error) => {\n        logger.error(`Startup Error:`, error);\n      });\n  };\n};\n\nmodule.exports = Server;\n\n\n\n// WEBPACK FOOTER //\n// ./server/index.js","module.exports = require(\"express\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"express\"\n// module id = 31\n// module chunks = 0","module.exports = require(\"body-parser\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"body-parser\"\n// module id = 32\n// module chunks = 0","module.exports = require(\"express-handlebars\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"express-handlebars\"\n// module id = 33\n// module chunks = 0","module.exports = require(\"handlebars\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"handlebars\"\n// module id = 34\n// module chunks = 0","module.exports = require(\"helmet\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"helmet\"\n// module id = 35\n// module chunks = 0","module.exports = require(\"cookie-session\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"cookie-session\"\n// module id = 36\n// module chunks = 0","module.exports = require(\"http\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"http\"\n// module id = 37\n// module chunks = 0","const logger = require('winston');\n\nconst requestLogger = (req, res, next) => {  // custom logging middleware to log all incoming http requests\n  logger.verbose(`Request on ${req.originalUrl} from ${req.ip}`);\n  next();\n};\n\nmodule.exports = requestLogger;\n\n\n\n// WEBPACK FOOTER //\n// ./server/middleware/requestLogger.js","module.exports = require(\"path\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"path\"\n// module id = 39\n// module chunks = 0","const logger = require('winston');\n\nfunction LoggerConfig () {\n  this.logLevel = 'debug';\n  this.update = (config) => {\n    if (!config) {\n      return logger.warn('No logger config received.');\n    }\n    logger.info('configuring winston logger...');\n    // update values with local config params\n    const {logLevel} = config;\n    this.logLevel = logLevel;\n    // configure the winston logger\n    logger.configure({\n      transports: [\n        new (logger.transports.Console)({\n          level                          : this.logLevel,\n          timestamp                      : false,\n          colorize                       : true,\n          prettyPrint                    : true,\n          handleExceptions               : true,\n          humanReadableUnhandledException: true,\n        }),\n      ],\n    });\n    // test all the log levels\n    logger.info('testing winston log levels...');\n    logger.error('Level 0');\n    logger.warn('Level 1');\n    logger.info('Level 2');\n    logger.verbose('Level 3');\n    logger.debug('Level 4');\n    logger.silly('Level 5');\n  };\n};\n\nmodule.exports = new LoggerConfig();\n\n\n\n// WEBPACK FOOTER //\n// ./config/loggerConfig.js","const winstonSlackWebHook = require('winston-slack-webhook').SlackWebHook;\nconst winston = require('winston');\n\nfunction SlackConfig () {\n  this.slackWebHook      = 'default';\n  this.slackErrorChannel = 'default';\n  this.slackInfoChannel  = 'default';\n  this.update = (config) => {\n    if (!config) {\n      return winston.warn('No slack config received');\n    }\n    // update variables\n    winston.info('configuring slack logger...');\n    const {slackWebHook, slackErrorChannel, slackInfoChannel} = config;\n    this.slackWebHook = slackWebHook;\n    this.slackErrorChannel = slackErrorChannel;\n    this.slackInfoChannel = slackInfoChannel;\n    // update slack webhook settings\n    if (this.slackWebHook) {\n      // add a transport for errors to slack\n      if (this.slackErrorChannel) {\n        winston.add(winstonSlackWebHook, {\n          name      : 'slack-errors-transport',\n          level     : 'warn',\n          webhookUrl: this.slackWebHook,\n          channel   : this.slackErrorChannel,\n          username  : 'spee.ch',\n          iconEmoji : ':face_with_head_bandage:',\n        });\n      };\n      if (slackInfoChannel) {\n        winston.add(winstonSlackWebHook, {\n          name      : 'slack-info-transport',\n          level     : 'info',\n          webhookUrl: this.slackWebHook,\n          channel   : this.slackInfoChannel,\n          username  : 'spee.ch',\n          iconEmoji : ':nerd_face:',\n        });\n      };\n      // send test messages\n      winston.info('testing slack logger...');\n      winston.error('Slack \"error\" logging is online.');\n      winston.info('Slack \"info\" logging is online.');\n    } else {\n      winston.warn('Slack logging is not enabled because no slackWebHook config var provided.');\n    }\n  };\n};\n\nmodule.exports = new SlackConfig();\n\n\n\n// WEBPACK FOOTER //\n// ./config/slackConfig.js","module.exports = require(\"winston-slack-webhook\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"winston-slack-webhook\"\n// module id = 42\n// module chunks = 0","module.exports = require(\"passport\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"passport\"\n// module id = 43\n// module chunks = 0","const PassportLocalStrategy = require('passport-local').Strategy;\nconst logger = require('winston');\nconst db = require('models');\n\nconst returnUserAndChannelInfo = (userInstance) => {\n  return new Promise((resolve, reject) => {\n    let userInfo = {};\n    userInfo['id'] = userInstance.id;\n    userInfo['userName'] = userInstance.userName;\n    userInstance\n      .getChannel()\n      .then(({channelName, channelClaimId}) => {\n        userInfo['channelName'] = channelName;\n        userInfo['channelClaimId'] = channelClaimId;\n        return db.Certificate.getShortChannelIdFromLongChannelId(channelClaimId, channelName);\n      })\n      .then(shortChannelId => {\n        userInfo['shortChannelId'] = shortChannelId;\n        resolve(userInfo);\n      })\n      .catch(error => {\n        reject(error);\n      });\n  });\n};\n\nmodule.exports = new PassportLocalStrategy(\n  {\n    usernameField: 'username',\n    passwordField: 'password',\n  },\n  (username, password, done) => {\n    return db.User\n      .findOne({\n        where: {userName: username},\n      })\n      .then(user => {\n        if (!user) {\n          logger.debug('no user found');\n          return done(null, false, {message: 'Incorrect username or password'});\n        }\n        return user.comparePassword(password)\n          .then(isMatch => {\n            if (!isMatch) {\n              logger.debug('incorrect password');\n              return done(null, false, {message: 'Incorrect username or password'});\n            }\n            logger.debug('Password was a match, returning User');\n            return returnUserAndChannelInfo(user)\n              .then(userInfo => {\n                return done(null, userInfo);\n              })\n              .catch(error => {\n                return error;\n              });\n          })\n          .catch(error => {\n            return error;\n          });\n      })\n      .catch(error => {\n        return done(error);\n      });\n  },\n);\n\n\n\n// WEBPACK FOOTER //\n// ./server/speechPassport/local-login.js","const logger = require('winston');\nconst { returnShortId } = require('../helpers/sequelizeHelpers.js');\n\nmodule.exports = (sequelize, { STRING, BOOLEAN, INTEGER, TEXT, DECIMAL }) => {\n  const Certificate = sequelize.define(\n    'Certificate',\n    {\n      address: {\n        type   : STRING,\n        default: null,\n      },\n      amount: {\n        type   : DECIMAL(19, 8),\n        default: null,\n      },\n      claimId: {\n        type   : STRING,\n        default: null,\n      },\n      claimSequence: {\n        type   : INTEGER,\n        default: null,\n      },\n      decodedClaim: {\n        type   : BOOLEAN,\n        default: null,\n      },\n      depth: {\n        type   : INTEGER,\n        default: null,\n      },\n      effectiveAmount: {\n        type   : DECIMAL(19, 8),\n        default: null,\n      },\n      hasSignature: {\n        type   : BOOLEAN,\n        default: null,\n      },\n      height: {\n        type   : INTEGER,\n        default: null,\n      },\n      hex: {\n        type   : TEXT('long'),\n        default: null,\n      },\n      name: {\n        type   : STRING,\n        default: null,\n      },\n      nout: {\n        type   : INTEGER,\n        default: null,\n      },\n      txid: {\n        type   : STRING,\n        default: null,\n      },\n      validAtHeight: {\n        type   : INTEGER,\n        default: null,\n      },\n      outpoint: {\n        type   : STRING,\n        default: null,\n      },\n      valueVersion: {\n        type   : STRING,\n        default: null,\n      },\n      claimType: {\n        type   : STRING,\n        default: null,\n      },\n      certificateVersion: {\n        type   : STRING,\n        default: null,\n      },\n      keyType: {\n        type   : STRING,\n        default: null,\n      },\n      publicKey: {\n        type   : TEXT('long'),\n        default: null,\n      },\n    },\n    {\n      freezeTableName: true,\n    }\n  );\n\n  Certificate.associate = db => {\n    Certificate.belongsTo(db.Channel, {\n      foreignKey: {\n        allowNull: true,\n      },\n    });\n  };\n\n  Certificate.getShortChannelIdFromLongChannelId = function (longChannelId, channelName) {\n    logger.debug(`getShortChannelIdFromLongChannelId ${channelName}:${longChannelId}`);\n    return new Promise((resolve, reject) => {\n      this\n        .findAll({\n          where: {name: channelName},\n          order: [['height', 'ASC']],\n        })\n        .then(result => {\n          switch (result.length) {\n            case 0:\n              throw new Error('No channel(s) found with that channel name');\n            default:\n              return resolve(returnShortId(result, longChannelId));\n          }\n        })\n        .catch(error => {\n          reject(error);\n        });\n    });\n  };\n\n  Certificate.getLongChannelIdFromShortChannelId = function (channelName, channelClaimId) {\n    logger.debug(`getLongChannelIdFromShortChannelId(${channelName}, ${channelClaimId})`);\n    return new Promise((resolve, reject) => {\n      this\n        .findAll({\n          where: {\n            name   : channelName,\n            claimId: {\n              $like: `${channelClaimId}%`,\n            },\n          },\n          order: [['height', 'ASC']],\n        })\n        .then(result => {\n          switch (result.length) {\n            case 0:\n              return resolve(null);\n            default: // note results must be sorted\n              return resolve(result[0].claimId);\n          }\n        })\n        .catch(error => {\n          reject(error);\n        });\n    });\n  };\n\n  Certificate.getLongChannelIdFromChannelName = function (channelName) {\n    logger.debug(`getLongChannelIdFromChannelName(${channelName})`);\n    return new Promise((resolve, reject) => {\n      this\n        .findAll({\n          where: { name: channelName },\n          order: [['effectiveAmount', 'DESC'], ['height', 'ASC']],\n        })\n        .then(result => {\n          switch (result.length) {\n            case 0:\n              return resolve(null);\n            default:\n              return resolve(result[0].claimId);\n          }\n        })\n        .catch(error => {\n          reject(error);\n        });\n    });\n  };\n\n  Certificate.validateLongChannelId = function (name, claimId) {\n    logger.debug(`validateLongChannelId(${name}, ${claimId})`);\n    return new Promise((resolve, reject) => {\n      this.findOne({\n        where: {name, claimId},\n      })\n        .then(result => {\n          if (!result) {\n            return resolve(null);\n          };\n          resolve(claimId);\n        })\n        .catch(error => {\n          reject(error);\n        });\n    });\n  };\n\n  Certificate.getLongChannelId = function (channelName, channelClaimId) {\n    logger.debug(`getLongChannelId(${channelName}, ${channelClaimId})`);\n    if (channelClaimId && (channelClaimId.length === 40)) {  // if a full channel id is provided\n      return this.validateLongChannelId(channelName, channelClaimId);\n    } else if (channelClaimId && channelClaimId.length < 40) {  // if a short channel id is provided\n      return this.getLongChannelIdFromShortChannelId(channelName, channelClaimId);\n    } else {\n      return this.getLongChannelIdFromChannelName(channelName);  // if no channel id provided\n    }\n  };\n\n  return Certificate;\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/models/certificate.js","module.exports = (sequelize, { STRING }) => {\n  const Channel = sequelize.define(\n    'Channel',\n    {\n      channelName: {\n        type     : STRING,\n        allowNull: false,\n      },\n      channelClaimId: {\n        type     : STRING,\n        allowNull: false,\n      },\n    },\n    {\n      freezeTableName: true,\n    }\n  );\n\n  Channel.associate = db => {\n    Channel.belongsTo(db.User);\n    Channel.hasOne(db.Certificate);\n  };\n\n  return Channel;\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/models/channel.js","const logger = require('winston');\nconst { returnShortId } = require('../helpers/sequelizeHelpers.js');\nconst { assetDefaults: { thumbnail: defaultThumbnail }, details: { host } } = require('../../config/siteConfig.js');\n\nfunction determineFileExtensionFromContentType (contentType) {\n  switch (contentType) {\n    case 'image/jpeg':\n    case 'image/jpg':\n      return 'jpeg';\n    case 'image/png':\n      return 'png';\n    case 'image/gif':\n      return 'gif';\n    case 'video/mp4':\n      return 'mp4';\n    default:\n      logger.debug('setting unknown file type as file extension jpeg');\n      return 'jpeg';\n  }\n};\n\nfunction determineThumbnail (storedThumbnail, defaultThumbnail) {\n  if (storedThumbnail === '') {\n    return defaultThumbnail;\n  }\n  return storedThumbnail;\n};\n\nfunction prepareClaimData (claim) {\n  // logger.debug('preparing claim data based on resolved data:', claim);\n  claim['thumbnail'] = determineThumbnail(claim.thumbnail, defaultThumbnail);\n  claim['fileExt'] = determineFileExtensionFromContentType(claim.contentType);\n  claim['host'] = host;\n  return claim;\n};\n\nmodule.exports = (sequelize, { STRING, BOOLEAN, INTEGER, TEXT, DECIMAL }) => {\n  const Claim = sequelize.define(\n    'Claim',\n    {\n      address: {\n        type   : STRING,\n        default: null,\n      },\n      amount: {\n        type   : DECIMAL(19, 8),\n        default: null,\n      },\n      claimId: {\n        type   : STRING,\n        default: null,\n      },\n      claimSequence: {\n        type   : INTEGER,\n        default: null,\n      },\n      decodedClaim: {\n        type   : BOOLEAN,\n        default: null,\n      },\n      depth: {\n        type   : INTEGER,\n        default: null,\n      },\n      effectiveAmount: {\n        type   : DECIMAL(19, 8),\n        default: null,\n      },\n      hasSignature: {\n        type   : BOOLEAN,\n        default: null,\n      },\n      height: {\n        type   : INTEGER,\n        default: null,\n      },\n      hex: {\n        type   : TEXT('long'),\n        default: null,\n      },\n      name: {\n        type   : STRING,\n        default: null,\n      },\n      nout: {\n        type   : INTEGER,\n        default: null,\n      },\n      txid: {\n        type   : STRING,\n        default: null,\n      },\n      validAtHeight: {\n        type   : INTEGER,\n        default: null,\n      },\n      outpoint: {\n        type   : STRING,\n        default: null,\n      },\n      claimType: {\n        type   : STRING,\n        default: null,\n      },\n      certificateId: {\n        type   : STRING,\n        default: null,\n      },\n      author: {\n        type   : STRING,\n        default: null,\n      },\n      description: {\n        type   : TEXT('long'),\n        default: null,\n      },\n      language: {\n        type   : STRING,\n        default: null,\n      },\n      license: {\n        type   : STRING,\n        default: null,\n      },\n      licenseUrl: {\n        type   : STRING,\n        default: null,\n      },\n      nsfw: {\n        type   : BOOLEAN,\n        default: null,\n      },\n      preview: {\n        type   : STRING,\n        default: null,\n      },\n      thumbnail: {\n        type   : STRING,\n        default: null,\n      },\n      title: {\n        type   : STRING,\n        default: null,\n      },\n      metadataVersion: {\n        type   : STRING,\n        default: null,\n      },\n      contentType: {\n        type   : STRING,\n        default: null,\n      },\n      source: {\n        type   : STRING,\n        default: null,\n      },\n      sourceType: {\n        type   : STRING,\n        default: null,\n      },\n      sourceVersion: {\n        type   : STRING,\n        default: null,\n      },\n      streamVersion: {\n        type   : STRING,\n        default: null,\n      },\n      valueVersion: {\n        type   : STRING,\n        default: null,\n      },\n      channelName: {\n        type     : STRING,\n        allowNull: true,\n        default  : null,\n      },\n    },\n    {\n      freezeTableName: true,\n    }\n  );\n\n  Claim.associate = db => {\n    Claim.belongsTo(db.File, {\n      foreignKey: {\n        allowNull: true,\n      },\n    });\n  };\n\n  Claim.getShortClaimIdFromLongClaimId = function (claimId, claimName) {\n    logger.debug(`Claim.getShortClaimIdFromLongClaimId for ${claimName}#${claimId}`);\n    return new Promise((resolve, reject) => {\n      this\n        .findAll({\n          where: { name: claimName },\n          order: [['height', 'ASC']],\n        })\n        .then(result => {\n          switch (result.length) {\n            case 0:\n              throw new Error('No claim(s) found with that claim name');\n            default:\n              resolve(returnShortId(result, claimId));\n          }\n        })\n        .catch(error => {\n          reject(error);\n        });\n    });\n  };\n\n  Claim.getAllChannelClaims = function (channelClaimId) {\n    logger.debug(`Claim.getAllChannelClaims for ${channelClaimId}`);\n    return new Promise((resolve, reject) => {\n      this\n        .findAll({\n          where: { certificateId: channelClaimId },\n          order: [['height', 'ASC']],\n          raw  : true,  // returns an array of only data, not an array of instances\n        })\n        .then(channelClaimsArray => {\n          // logger.debug('channelclaimsarray length:', channelClaimsArray.length);\n          switch (channelClaimsArray.length) {\n            case 0:\n              return resolve(null);\n            default:\n              channelClaimsArray.forEach(claim => {\n                claim['fileExt'] = determineFileExtensionFromContentType(claim.contentType);\n                claim['thumbnail'] = determineThumbnail(claim.thumbnail, defaultThumbnail);\n                return claim;\n              });\n              return resolve(channelClaimsArray);\n          }\n        })\n        .catch(error => {\n          reject(error);\n        });\n    });\n  };\n\n  Claim.getClaimIdByLongChannelId = function (channelClaimId, claimName) {\n    logger.debug(`finding claim id for claim ${claimName} from channel ${channelClaimId}`);\n    return new Promise((resolve, reject) => {\n      this\n        .findAll({\n          where: { name: claimName, certificateId: channelClaimId },\n          order: [['id', 'ASC']],\n        })\n        .then(result => {\n          switch (result.length) {\n            case 0:\n              return resolve(null);\n            case 1:\n              return resolve(result[0].claimId);\n            default:\n              logger.error(`${result.length} records found for \"${claimName}\" in channel \"${channelClaimId}\"`);\n              return resolve(result[0].claimId);\n          }\n        })\n        .catch(error => {\n          reject(error);\n        });\n    });\n  };\n\n  Claim.getLongClaimIdFromShortClaimId = function (name, shortId) {\n    return new Promise((resolve, reject) => {\n      this\n        .findAll({\n          where: {\n            name,\n            claimId: {\n              $like: `${shortId}%`,\n            }},\n          order: [['height', 'ASC']],\n        })\n        .then(result => {\n          switch (result.length) {\n            case 0:\n              return resolve(null);\n            default: // note results must be sorted\n              return resolve(result[0].claimId);\n          }\n        })\n        .catch(error => {\n          reject(error);\n        });\n    });\n  };\n\n  Claim.getTopFreeClaimIdByClaimName = function (name) {\n    return new Promise((resolve, reject) => {\n      this\n        .findAll({\n          where: { name },\n          order: [['effectiveAmount', 'DESC'], ['height', 'ASC']],  // note: maybe height and effective amount need to switch?\n        })\n        .then(result => {\n          logger.debug('length of result', result.length);\n          switch (result.length) {\n            case 0:\n              return resolve(null);\n            default:\n              return resolve(result[0].dataValues.claimId);\n          }\n        })\n        .catch(error => {\n          reject(error);\n        });\n    });\n  };\n\n  Claim.validateLongClaimId = function (name, claimId) {\n    return new Promise((resolve, reject) => {\n      this.findOne({\n        where: {name, claimId},\n      })\n        .then(result => {\n          if (!result) {\n            return resolve(null);\n          };\n          resolve(claimId);\n        })\n        .catch(error => {\n          reject(error);\n        });\n    });\n  };\n\n  Claim.getLongClaimId = function (claimName, claimId) {\n    logger.debug(`getLongClaimId(${claimName}, ${claimId})`);\n    if (claimId && (claimId.length === 40)) {  // if a full claim id is provided\n      return this.validateLongClaimId(claimName, claimId);\n    } else if (claimId && claimId.length < 40) {\n      return this.getLongClaimIdFromShortClaimId(claimName, claimId);  // if a short claim id is provided\n    } else {\n      return this.getTopFreeClaimIdByClaimName(claimName);  // if no claim id is provided\n    }\n  };\n\n  Claim.resolveClaim = function (name, claimId) {\n    logger.debug(`Claim.resolveClaim: ${name} ${claimId}`);\n    return new Promise((resolve, reject) => {\n      this\n        .findAll({\n          where: { name, claimId },\n        })\n        .then(claimArray => {\n          switch (claimArray.length) {\n            case 0:\n              return resolve(null);\n            case 1:\n              return resolve(prepareClaimData(claimArray[0].dataValues));\n            default:\n              logger.error(`more than one record matches ${name}#${claimId} in db.Claim`);\n              return resolve(prepareClaimData(claimArray[0].dataValues));\n          }\n        })\n        .catch(error => {\n          reject(error);\n        });\n    });\n  };\n\n  return Claim;\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/models/claim.js","module.exports = (sequelize, { STRING, BOOLEAN, INTEGER }) => {\n  const File = sequelize.define(\n    'File',\n    {\n      name: {\n        type     : STRING,\n        allowNull: false,\n      },\n      claimId: {\n        type     : STRING,\n        allowNull: false,\n      },\n      address: {\n        type     : STRING,\n        allowNull: false,\n      },\n      outpoint: {\n        type     : STRING,\n        allowNull: false,\n      },\n      height: {\n        type     : INTEGER,\n        allowNull: false,\n        default  : 0,\n      },\n      fileName: {\n        type     : STRING,\n        allowNull: false,\n      },\n      filePath: {\n        type     : STRING,\n        allowNull: false,\n      },\n      fileType: {\n        type: STRING,\n      },\n      nsfw: {\n        type        : BOOLEAN,\n        allowNull   : false,\n        defaultValue: false,\n      },\n      trendingEligible: {\n        type        : BOOLEAN,\n        allowNull   : false,\n        defaultValue: true,\n      },\n    },\n    {\n      freezeTableName: true,\n    }\n  );\n\n  File.associate = db => {\n    File.hasMany(db.Request);\n    File.hasOne(db.Claim);\n  };\n\n  File.getRecentClaims = function () {\n    return this.findAll({\n      where: { nsfw: false, trendingEligible: true },\n      order: [['createdAt', 'DESC']],\n      limit: 25,\n    });\n  };\n\n  return File;\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/models/file.js","module.exports = (sequelize, { STRING, BOOLEAN, TEXT }) => {\n  const Request = sequelize.define(\n    'Request',\n    {\n      action: {\n        type     : STRING,\n        allowNull: false,\n      },\n      url: {\n        type     : STRING,\n        allowNull: false,\n      },\n      ipAddress: {\n        type     : STRING,\n        allowNull: true,\n      },\n      result: {\n        type     : TEXT('long'),\n        allowNull: true,\n        default  : null,\n      },\n    },\n    {\n      freezeTableName: true,\n    }\n  );\n\n  Request.associate = db => {\n    Request.belongsTo(db.File, {\n      foreignKey: {\n        allowNull: true,\n      },\n    });\n  };\n\n  return Request;\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/models/request.js","'use strict';\nconst bcrypt = require('bcrypt');\nconst logger = require('winston');\n\nmodule.exports = (sequelize, { STRING }) => {\n  const User = sequelize.define(\n    'User',\n    {\n      userName: {\n        type     : STRING,\n        allowNull: false,\n      },\n      password: {\n        type     : STRING,\n        allowNull: false,\n      },\n    },\n    {\n      freezeTableName: true,\n    }\n  );\n\n  User.associate = db => {\n    User.hasOne(db.Channel);\n  };\n\n  User.prototype.comparePassword = function (password) {\n    return bcrypt.compare(password, this.password);\n  };\n\n  User.prototype.changePassword = function (newPassword) {\n    return new Promise((resolve, reject) => {\n      // generate a salt string to use for hashing\n      bcrypt.genSalt((saltError, salt) => {\n        if (saltError) {\n          logger.error('salt error', saltError);\n          reject(saltError);\n          return;\n        }\n        // generate a hashed version of the user's password\n        bcrypt.hash(newPassword, salt, (hashError, hash) => {\n          // if there is an error with the hash generation return the error\n          if (hashError) {\n            logger.error('hash error', hashError);\n            reject(hashError);\n            return;\n          }\n          // replace the current password with the new hash\n          this\n            .update({password: hash})\n            .then(() => {\n              resolve();\n            })\n            .catch(error => {\n              reject(error);\n            });\n        });\n      });\n    });\n  };\n\n  // pre-save hook method to hash the user's password before the user's info is saved to the db.\n  User.hook('beforeCreate', (user, options) => {\n    logger.debug('User.beforeCreate hook...');\n    return new Promise((resolve, reject) => {\n      // generate a salt string to use for hashing\n      bcrypt.genSalt((saltError, salt) => {\n        if (saltError) {\n          logger.error('salt error', saltError);\n          reject(saltError);\n          return;\n        }\n        // generate a hashed version of the user's password\n        bcrypt.hash(user.password, salt, (hashError, hash) => {\n          // if there is an error with the hash generation return the error\n          if (hashError) {\n            logger.error('hash error', hashError);\n            reject(hashError);\n            return;\n          }\n          // replace the password string with the hash password value\n          user.password = hash;\n          resolve();\n        });\n      });\n    });\n  });\n\n  return User;\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/models/user.js","module.exports = require(\"bcrypt\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"bcrypt\"\n// module id = 51\n// module chunks = 0","const PassportLocalStrategy = require('passport-local').Strategy;\nconst lbryApi = require('../helpers/lbryApi.js');\nconst logger = require('winston');\nconst db = require('models');\n\nmodule.exports = new PassportLocalStrategy(\n  {\n    usernameField: 'username',\n    passwordField: 'password',\n  },\n  (username, password, done) => {\n    logger.verbose(`new channel signup request. user: ${username} pass: ${password} .`);\n    let userInfo = {};\n    // server-side validaton of inputs (username, password)\n\n    // create the channel and retrieve the metadata\n    return lbryApi.createChannel(`@${username}`)\n      .then(tx => {\n        // create user record\n        const userData = {\n          userName: username,\n          password: password,\n        };\n        logger.verbose('userData >', userData);\n        // create user record\n        const channelData = {\n          channelName   : `@${username}`,\n          channelClaimId: tx.claim_id,\n        };\n        logger.verbose('channelData >', channelData);\n        // create certificate record\n        const certificateData = {\n          claimId: tx.claim_id,\n          name   : `@${username}`,\n          // address,\n        };\n        logger.verbose('certificateData >', certificateData);\n        // save user and certificate to db\n        return Promise.all([db.User.create(userData), db.Channel.create(channelData), db.Certificate.create(certificateData)]);\n      })\n      .then(([newUser, newChannel, newCertificate]) => {\n        logger.verbose('user and certificate successfully created');\n        // store the relevant newUser info to be passed back for req.User\n        userInfo['id'] = newUser.id;\n        userInfo['userName'] = newUser.userName;\n        userInfo['channelName'] = newChannel.channelName;\n        userInfo['channelClaimId'] = newChannel.channelClaimId;\n        // associate the instances\n        return Promise.all([newCertificate.setChannel(newChannel), newChannel.setUser(newUser)]);\n      })\n      .then(() => {\n        logger.verbose('user and certificate successfully associated');\n        return db.Certificate.getShortChannelIdFromLongChannelId(userInfo.channelClaimId, userInfo.channelName);\n      })\n      .then(shortChannelId => {\n        userInfo['shortChannelId'] = shortChannelId;\n        return done(null, userInfo);\n      })\n      .catch(error => {\n        logger.error('signup error', error);\n        return done(error);\n      });\n  }\n);\n\n\n\n// WEBPACK FOOTER //\n// ./server/speechPassport/local-signup.js","module.exports = require(\"axios\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"axios\"\n// module id = 53\n// module chunks = 0","const lbryConfig = {\n  api: {\n    apiHost: 'localhost',\n    apiPort: '5279',\n  },\n};\n\nmodule.exports = lbryConfig;\n\n\n\n// WEBPACK FOOTER //\n// ./config/lbryConfig.js","module.exports = require(\"universal-analytics\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"universal-analytics\"\n// module id = 55\n// module chunks = 0","module.exports = {\n  serializeSpeechUser (user, done) {  // returns user data to be serialized into session\n    console.log('serializing user');\n    done(null, user);\n  },\n  deserializeSpeechUser (user, done) {  // deserializes session and populates additional info to req.user\n    console.log('deserializing user');\n    done(null, user);\n  },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/authHelpers.js","const speechPassport = require('speechPassport');\nconst handleSignupRequest = require('./signup');\nconst handleLoginRequest = require('./login');\nconst handleLogoutRequest = require('./logout');\nconst handleUserRequest = require('./user');\n\nmodule.exports = (app) => {\n  app.post('/signup', speechPassport.authenticate('local-signup'), handleSignupRequest);\n  app.post('/login', handleLoginRequest);\n  app.get('/logout', handleLogoutRequest);\n  app.get('/user', handleUserRequest);\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/auth/index.js","const signup = (req, res) => {\n  res.status(200).json({\n    success       : true,\n    channelName   : req.user.channelName,\n    channelClaimId: req.user.channelClaimId,\n    shortChannelId: req.user.shortChannelId,\n  });\n};\n\nmodule.exports = signup;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/auth/signup.js","const speechPassport = require('speechPassport');\n\nconst login = (req, res, next) => {\n  speechPassport.authenticate('local-login', (err, user, info) => {\n    if (err) {\n      return next(err);\n    }\n    if (!user) {\n      return res.status(400).json({\n        success: false,\n        message: info.message,\n      });\n    }\n    req.logIn(user, (err) => {\n      if (err) {\n        return next(err);\n      }\n      return res.status(200).json({\n        success       : true,\n        channelName   : req.user.channelName,\n        channelClaimId: req.user.channelClaimId,\n        shortChannelId: req.user.shortChannelId,\n      });\n    });\n  })(req, res, next);\n};\n\nmodule.exports = login;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/auth/login.js","const logout = (req, res) => {\n  req.logout();\n  res.status(200).json({success: true, message: 'you successfully logged out'});\n};\n\nmodule.exports = logout;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/auth/logout.js","const user = (req, res) => {\n  if (req.user) {\n    res.status(200).json({success: true, data: req.user});\n  } else {\n    res.status(401).json({success: false, message: 'user is not logged in'});\n  }\n};\n\nmodule.exports = user;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/auth/user.js","const channelAvailability = require('./channelAvailability');\nconst channelClaims = require('./channelClaims');\nconst channelData = require('./channelData');\nconst channelShortId = require('./channelShortId');\nconst claimAvailability = require('./claimAvailability');\nconst claimData = require('./claimData');\nconst claimGet = require('./claimGet');\nconst claimLongId = require('./claimLongId');\nconst claimPublish = require('./claimPublish');\nconst claimResolve = require('./claimResolve');\nconst claimShortId = require('./claimShortId');\nconst claimList = require('./claimList');\nconst fileAvailability = require('./fileAvailability');\n\nconst multipartMiddleware = require('helpers/multipartMiddleware');\n\nmodule.exports = (app) => {\n  // channel routes\n  app.get('/api/channel/availability/:name', channelAvailability);\n  app.get('/api/channel/short-id/:longId/:name', channelShortId);\n  app.get('/api/channel/data/:channelName/:channelClaimId', channelData);\n  app.get('/api/channel/claims/:channelName/:channelClaimId/:page', channelClaims);\n  // claim routes\n  app.get('/api/claim/list/:name', claimList);\n  app.get('/api/claim/get/:name/:claimId', claimGet);\n  app.get('/api/claim/availability/:name', claimAvailability);\n  app.get('/api/claim/resolve/:name/:claimId', claimResolve);\n  app.post('/api/claim/publish', multipartMiddleware, claimPublish);\n  app.get('/api/claim/short-id/:longId/:name', claimShortId);\n  app.post('/api/claim/long-id', claimLongId);\n  app.get('/api/claim/data/:claimName/:claimId', claimData);\n  // file routes\n  app.get('/api/file/availability/:name/:claimId', fileAvailability);\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/index.js","const { checkChannelAvailability } = require('controllers/publishController.js');\nconst { sendGATimingEvent } = require('helpers/googleAnalytics.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\n\n/*\n\n  route to check whether site has published to a channel\n\n*/\n\nconst channelAvailability = ({ ip, originalUrl, params: { name } }, res) => {\n  const gaStartTime = Date.now();\n  checkChannelAvailability(name)\n    .then(availableName => {\n      res.status(200).json(availableName);\n      sendGATimingEvent('end-to-end', 'claim name availability', name, gaStartTime, Date.now());\n    })\n    .catch(error => {\n      handleErrorResponse(originalUrl, ip, error, res);\n    });\n};\n\nmodule.exports = channelAvailability;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/channelAvailability.js","module.exports = require(\"fs\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"fs\"\n// module id = 64\n// module chunks = 0","const { getChannelClaims } = require('controllers/serveController.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\n\nconst NO_CHANNEL = 'NO_CHANNEL';\n\n/*\n\n  route to get all claims for channel\n\n*/\n\nconst channelClaims = ({ ip, originalUrl, body, params }, res) => {\n  const channelName = params.channelName;\n  let channelClaimId = params.channelClaimId;\n  if (channelClaimId === 'none') channelClaimId = null;\n  const page = params.page;\n  getChannelClaims(channelName, channelClaimId, page)\n    .then(data => {\n      if (data === NO_CHANNEL) {\n        return res.status(404).json({success: false, message: 'No matching channel was found'});\n      }\n      res.status(200).json({success: true, data});\n    })\n    .catch(error => {\n      handleErrorResponse(originalUrl, ip, error, res);\n    });\n};\n\nmodule.exports = channelClaims;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/channelClaims.js","const CLAIMS_PER_PAGE = 12;\n\nmodule.exports = {\n  returnPaginatedChannelClaims (channelName, longChannelClaimId, claims, page) {\n    const totalPages = module.exports.determineTotalPages(claims);\n    const paginationPage = module.exports.getPageFromQuery(page);\n    const viewData = {\n      channelName       : channelName,\n      longChannelClaimId: longChannelClaimId,\n      claims            : module.exports.extractPageFromClaims(claims, paginationPage),\n      previousPage      : module.exports.determinePreviousPage(paginationPage),\n      currentPage       : paginationPage,\n      nextPage          : module.exports.determineNextPage(totalPages, paginationPage),\n      totalPages        : totalPages,\n      totalResults      : module.exports.determineTotalClaims(claims),\n    };\n    return viewData;\n  },\n  getPageFromQuery (page) {\n    if (page) {\n      return parseInt(page);\n    }\n    return 1;\n  },\n  extractPageFromClaims (claims, pageNumber) {\n    if (!claims) {\n      return [];  // if no claims, return this default\n    }\n    // logger.debug('claims is array?', Array.isArray(claims));\n    // logger.debug(`pageNumber ${pageNumber} is number?`, Number.isInteger(pageNumber));\n    const claimStartIndex = (pageNumber - 1) * CLAIMS_PER_PAGE;\n    const claimEndIndex = claimStartIndex + CLAIMS_PER_PAGE;\n    const pageOfClaims = claims.slice(claimStartIndex, claimEndIndex);\n    return pageOfClaims;\n  },\n  determineTotalPages (claims) {\n    if (!claims) {\n      return 0;\n    } else {\n      const totalClaims = claims.length;\n      if (totalClaims < CLAIMS_PER_PAGE) {\n        return 1;\n      }\n      const fullPages = Math.floor(totalClaims / CLAIMS_PER_PAGE);\n      const remainder = totalClaims % CLAIMS_PER_PAGE;\n      if (remainder === 0) {\n        return fullPages;\n      }\n      return fullPages + 1;\n    }\n  },\n  determinePreviousPage (currentPage) {\n    if (currentPage === 1) {\n      return null;\n    }\n    return currentPage - 1;\n  },\n  determineNextPage (totalPages, currentPage) {\n    if (currentPage === totalPages) {\n      return null;\n    }\n    return currentPage + 1;\n  },\n  determineTotalClaims (claims) {\n    if (!claims) {\n      return 0;\n    }\n    return claims.length;\n  },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/channelPagination.js","const { getChannelData } = require('controllers/serveController.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\n\nconst NO_CHANNEL = 'NO_CHANNEL';\n\n/*\n\n  route to get data for a channel\n\n*/\n\nconst channelData = ({ ip, originalUrl, body, params }, res) => {\n  const channelName = params.channelName;\n  let channelClaimId = params.channelClaimId;\n  if (channelClaimId === 'none') channelClaimId = null;\n  getChannelData(channelName, channelClaimId, 0)\n    .then(data => {\n      if (data === NO_CHANNEL) {\n        return res.status(404).json({success: false, message: 'No matching channel was found'});\n      }\n      res.status(200).json({success: true, data});\n    })\n    .catch(error => {\n      handleErrorResponse(originalUrl, ip, error, res);\n    });\n};\n\nmodule.exports = channelData;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/channelData.js","const { handleErrorResponse } = require('helpers/errorHandlers.js');\nconst db = require('models');\n\n/*\n\nroute to get a short channel id from long channel Id\n\n*/\n\nconst channelShortIdRoute = ({ ip, originalUrl, params }, res) => {\n  db.Certificate.getShortChannelIdFromLongChannelId(params.longId, params.name)\n    .then(shortId => {\n      res.status(200).json(shortId);\n    })\n    .catch(error => {\n      handleErrorResponse(originalUrl, ip, error, res);\n    });\n};\n\nmodule.exports = channelShortIdRoute;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/channelShortId.js","const { claimNameIsAvailable } = require('controllers/publishController.js');\nconst { sendGATimingEvent } = require('helpers/googleAnalytics.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\n\n/*\n\n  route to check whether this site published to a claim\n\n*/\n\nconst claimAvailability = ({ ip, originalUrl, params: { name } }, res) => {\n  const gaStartTime = Date.now();\n  claimNameIsAvailable(name)\n    .then(result => {\n      res.status(200).json(result);\n      sendGATimingEvent('end-to-end', 'claim name availability', name, gaStartTime, Date.now());\n    })\n    .catch(error => {\n      handleErrorResponse(originalUrl, ip, error, res);\n    });\n};\n\nmodule.exports = claimAvailability;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimAvailability.js","const { handleErrorResponse } = require('helpers/errorHandlers.js');\nconst db = require('models');\n\n/*\n\n  route to return data for a claim\n\n*/\n\nconst claimData = ({ ip, originalUrl, body, params }, res) => {\n  const claimName = params.claimName;\n  let claimId = params.claimId;\n  if (claimId === 'none') claimId = null;\n  db.Claim.resolveClaim(claimName, claimId)\n    .then(claimInfo => {\n      if (!claimInfo) {\n        return res.status(404).json({success: false, message: 'No claim could be found'});\n      }\n      res.status(200).json({success: true, data: claimInfo});\n    })\n    .catch(error => {\n      handleErrorResponse(originalUrl, ip, error, res);\n    });\n};\n\nmodule.exports = claimData;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimData.js","const { getClaim } = require('helpers/lbryApi.js');\nconst { addGetResultsToFileData, createFileData } = require('../../helpers/publishHelpers.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\nconst db = require('models');\n\n/*\n\n  route to get a claim\n\n*/\n\nconst claimGet = ({ ip, originalUrl, params }, res) => {\n  const name = params.name;\n  const claimId = params.claimId;\n  // resolve the claim\n  db.Claim.resolveClaim(name, claimId)\n    .then(resolveResult => {\n      // make sure a claim actually exists at that uri\n      if (!resolveResult) {\n        throw new Error('No matching uri found in Claim table');\n      }\n      let fileData = createFileData(resolveResult);\n      // get the claim\n      return Promise.all([fileData, getClaim(`${name}#${claimId}`)]);\n    })\n    .then(([ fileData, getResult ]) => {\n      fileData = addGetResultsToFileData(fileData, getResult);\n      return Promise.all([db.upsert(db.File, fileData, {name, claimId}, 'File'), getResult]);\n    })\n    .then(([ fileRecord, {message, completed} ]) => {\n      res.status(200).json({ success: true, message, completed });\n    })\n    .catch(error => {\n      handleErrorResponse(originalUrl, ip, error, res);\n    });\n};\n\nmodule.exports = claimGet;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimGet.js","const { getClaimId } = require('controllers/serveController.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\n\nconst NO_CHANNEL = 'NO_CHANNEL';\nconst NO_CLAIM = 'NO_CLAIM';\n\n/*\n\n  route to get a long claim id\n\n*/\n\nconst claimLongId = ({ ip, originalUrl, body, params }, res) => {\n  const channelName = body.channelName;\n  const channelClaimId = body.channelClaimId;\n  const claimName = body.claimName;\n  const claimId = body.claimId;\n  getClaimId(channelName, channelClaimId, claimName, claimId)\n    .then(result => {\n      if (result === NO_CHANNEL) {\n        return res.status(404).json({success: false, message: 'No matching channel could be found'});\n      }\n      if (result === NO_CLAIM) {\n        return res.status(404).json({success: false, message: 'No matching claim id could be found'});\n      }\n      res.status(200).json({success: true, data: result});\n    })\n    .catch(error => {\n      handleErrorResponse(originalUrl, ip, error, res);\n    });\n};\n\nmodule.exports = claimLongId;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimLongId.js","const { createBasicPublishParams, createThumbnailPublishParams, parsePublishApiRequestBody, parsePublishApiRequestFiles } = require('helpers/publishHelpers.js');\nconst { claimNameIsAvailable, publish } = require('controllers/publishController.js');\nconst { authenticateUser } = require('auth/authentication.js');\nconst { sendGATimingEvent } = require('helpers/googleAnalytics.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\nconst { details: { host } } = require('siteConfig.js');\n\n/*\n\n  route to publish a claim through the daemon\n\n*/\n\nconst claimPublish = ({ body, files, headers, ip, originalUrl, user }, res) => {\n  // define variables\n  let  channelName, channelId, channelPassword, description, fileName, filePath, fileType, gaStartTime, license, name, nsfw, thumbnail, thumbnailFileName, thumbnailFilePath, thumbnailFileType, title;\n  // record the start time of the request\n  gaStartTime = Date.now();\n  // validate the body and files of the request\n  try {\n    // validateApiPublishRequest(body, files);\n    ({name, nsfw, license, title, description, thumbnail} = parsePublishApiRequestBody(body));\n    ({fileName, filePath, fileType, thumbnailFileName, thumbnailFilePath, thumbnailFileType} = parsePublishApiRequestFiles(files));\n    ({channelName, channelId, channelPassword} = body);\n  } catch (error) {\n    return res.status(400).json({success: false, message: error.message});\n  }\n  // check channel authorization\n  Promise\n    .all([\n      authenticateUser(channelName, channelId, channelPassword, user),\n      claimNameIsAvailable(name),\n      createBasicPublishParams(filePath, name, title, description, license, nsfw, thumbnail),\n      createThumbnailPublishParams(thumbnailFilePath, name, license, nsfw),\n    ])\n    .then(([{channelName, channelClaimId}, validatedClaimName, publishParams, thumbnailPublishParams]) => {\n      // add channel details to the publish params\n      if (channelName && channelClaimId) {\n        publishParams['channel_name'] = channelName;\n        publishParams['channel_id'] = channelClaimId;\n      }\n      // publish the thumbnail\n      if (thumbnailPublishParams) {\n        publish(thumbnailPublishParams, thumbnailFileName, thumbnailFileType);\n      }\n      // publish the asset\n      return publish(publishParams, fileName, fileType);\n    })\n    .then(result => {\n      res.status(200).json({\n        success: true,\n        message: 'publish completed successfully',\n        data   : {\n          name,\n          claimId: result.claim_id,\n          url    : `${host}/${result.claim_id}/${name}`,\n          lbryTx : result,\n        },\n      });\n      // record the publish end time and send to google analytics\n      sendGATimingEvent('end-to-end', 'publish', fileType, gaStartTime, Date.now());\n    })\n    .catch(error => {\n      handleErrorResponse(originalUrl, ip, error, res);\n    });\n};\n\nmodule.exports = claimPublish;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimPublish.js","const db = require('models');\nconst logger = require('winston');\n\nmodule.exports = {\n  authenticateUser (channelName, channelId, channelPassword, user) {\n    // case: no channelName or channel Id are provided (anonymous), regardless of whether user token is provided\n    if (!channelName && !channelId) {\n      return {\n        channelName   : null,\n        channelClaimId: null,\n      };\n    }\n    // case: channelName or channel Id are provided with user token\n    if (user) {\n      if (channelName && channelName !== user.channelName) {\n        throw new Error('the provided channel name does not match user credentials');\n      }\n      if (channelId && channelId !== user.channelClaimId) {\n        throw new Error('the provided channel id does not match user credentials');\n      }\n      return {\n        channelName   : user.channelName,\n        channelClaimId: user.channelClaimId,\n      };\n    }\n    // case: channelName or channel Id are provided with password instead of user token\n    if (!channelPassword) throw new Error('no channel password provided');\n    return module.exports.authenticateChannelCredentials(channelName, channelId, channelPassword);\n  },\n  authenticateChannelCredentials (channelName, channelId, userPassword) {\n    return new Promise((resolve, reject) => {\n      // hoisted variables\n      let channelData;\n      // build the params for finding the channel\n      let channelFindParams = {};\n      if (channelName) channelFindParams['channelName'] = channelName;\n      if (channelId) channelFindParams['channelClaimId'] = channelId;\n      // find the channel\n      db.Channel\n        .findOne({\n          where: channelFindParams,\n        })\n        .then(channel => {\n          if (!channel) {\n            logger.debug('no channel found');\n            throw new Error('Authentication failed, you do not have access to that channel');\n          }\n          channelData = channel.get();\n          logger.debug('channel data:', channelData);\n          return db.User.findOne({\n            where: { userName: channelData.channelName.substring(1) },\n          });\n        })\n        .then(user => {\n          if (!user) {\n            logger.debug('no user found');\n            throw new Error('Authentication failed, you do not have access to that channel');\n          }\n          return user.comparePassword(userPassword);\n        })\n        .then(isMatch => {\n          if (!isMatch) {\n            logger.debug('incorrect password');\n            throw new Error('Authentication failed, you do not have access to that channel');\n          }\n          logger.debug('...password was a match...');\n          resolve(channelData);\n        })\n        .catch(error => {\n          reject(error);\n        });\n    });\n  },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/auth/authentication.js","const { resolveUri } = require('helpers/lbryApi.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\n\n/*\n\n  route to run a resolve request on the daemon\n\n*/\n\nconst claimResolve = ({ headers, ip, originalUrl, params }, res) => {\n  resolveUri(`${params.name}#${params.claimId}`)\n    .then(resolvedUri => {\n      res.status(200).json(resolvedUri);\n    })\n    .catch(error => {\n      handleErrorResponse(originalUrl, ip, error, res);\n    });\n};\n\nmodule.exports = claimResolve;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimResolve.js","const { handleErrorResponse } = require('helpers/errorHandlers.js');\nconst db = require('models');\n\n/*\n\n  route to get a short claim id from long claim Id\n\n*/\n\nconst claimShortId = ({ ip, originalUrl, body, params }, res) => {\n  db.Claim.getShortClaimIdFromLongClaimId(params.longId, params.name)\n    .then(shortId => {\n      res.status(200).json({success: true, data: shortId});\n    })\n    .catch(error => {\n      handleErrorResponse(originalUrl, ip, error, res);\n    });\n};\n\nmodule.exports = claimShortId;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimShortId.js","const { getClaimList } = require('helpers/lbryApi.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\n\n/*\n\n  route to get list of claims\n\n*/\n\nconst claimList = ({ ip, originalUrl, params }, res) => {\n  getClaimList(params.name)\n    .then(claimsList => {\n      res.status(200).json(claimsList);\n    })\n    .catch(error => {\n      handleErrorResponse(originalUrl, ip, error, res);\n    });\n};\n\nmodule.exports = claimList;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimList.js","const { handleErrorResponse } = require('helpers/errorHandlers.js');\nconst db = require('models');\n\n/*\n\n  route to see if asset is available locally\n\n*/\n\nconst fileAvailability = ({ ip, originalUrl, params }, res) => {\n  const name = params.name;\n  const claimId = params.claimId;\n  db.File\n    .findOne({\n      where: {\n        name,\n        claimId,\n      },\n    })\n    .then(result => {\n      if (result) {\n        return res.status(200).json({success: true, data: true});\n      }\n      res.status(200).json({success: true, data: false});\n    })\n    .catch(error => {\n      handleErrorResponse(originalUrl, ip, error, res);\n    });\n};\n\nmodule.exports = fileAvailability;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/fileAvailability.js","const multipart = require('connect-multiparty');\r\nconst { publishing: { uploadDirectory } } = require('siteConfig.js');\r\nconst multipartMiddleware = multipart({uploadDir: uploadDirectory});\r\n\r\nmodule.exports = multipartMiddleware;\r\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/multipartMiddleware.js","module.exports = require(\"connect-multiparty\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"connect-multiparty\"\n// module id = 80\n// module chunks = 0","const handlePageRequest = require('./sendReactApp');\nconst handleEmbedRequest = require('./sendEmbedPage');\nconst redirect = require('./redirect');\n\nmodule.exports = (app) => {\n  app.get('/', handlePageRequest);\n  app.get('/login', handlePageRequest);\n  app.get('/about', handlePageRequest);\n  app.get('/trending', redirect('/popular'));\n  app.get('/popular', handlePageRequest);\n  app.get('/new', handlePageRequest);\n  app.get('/embed/:claimId/:name', handleEmbedRequest);  // route to send embedable video player (for twitter)\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/pages/index.js","const handlePageRender = require('helpers/handlePageRender.jsx');\n\nconst sendReactApp = (req, res) => {\n  handlePageRender(req, res);\n};\n\nmodule.exports = sendReactApp;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/pages/sendReactApp.js","const { details: { host } } = require('siteConfig.js');\n\nconst sendEmbedPage = ({ params }, res) => {\n  const claimId = params.claimId;\n  const name = params.name;\n  // get and render the content\n  res.status(200).render('embed', { layout: 'embed', host, claimId, name });\n};\n\nmodule.exports = sendEmbedPage;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/pages/sendEmbedPage.js","const redirect = (route) => {\n  return (req, res) => {\n    res.status(301).redirect(route);\n  };\n};\n\nmodule.exports = redirect;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/pages/redirect.js","const serveAssetByClaim = require('./serveAssetByClaim');\nconst serveAssetByIdentifierAndClaim = require('./serveAssetByIdentifierAndClaim');\n\nmodule.exports = (app, db) => {\n  app.get('/:identifier/:claim', serveAssetByIdentifierAndClaim);\n  app.get('/:claim', serveAssetByClaim);\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/assets/index.js","const { sendGAServeEvent } = require('helpers/googleAnalytics');\nconst { determineResponseType, logRequestData, getClaimIdAndServeAsset } = require('helpers/serveHelpers.js');\nconst lbryUri = require('helpers/lbryUri.js');\nconst handleShowRender = require('helpers/handleShowRender.jsx');\nconst SERVE = 'SERVE';\n\n/*\n\n  route to serve an asset or the react app via the claim name only\n\n*/\n\nconst serverAssetByClaim = (req, res) => {\n  const { headers, ip, originalUrl, params } = req;\n  // decide if this is a show request\n  let hasFileExtension;\n  try {\n    ({ hasFileExtension } = lbryUri.parseModifier(params.claim));\n  } catch (error) {\n    return res.status(400).json({success: false, message: error.message});\n  }\n  let responseType = determineResponseType(hasFileExtension, headers);\n  if (responseType !== SERVE) {\n    return handleShowRender(req, res);\n  }\n  // handle serve request\n  // send google analytics\n  sendGAServeEvent(headers, ip, originalUrl);\n  // parse the claim\n  let claimName;\n  try {\n    ({claimName} = lbryUri.parseClaim(params.claim));\n  } catch (error) {\n    return res.status(400).json({success: false, message: error.message});\n  }\n  // log the request data for debugging\n  logRequestData(responseType, claimName, null, null);\n  // get the claim Id and then serve the asset\n  getClaimIdAndServeAsset(null, null, claimName, null, originalUrl, ip, res);\n};\n\nmodule.exports = serverAssetByClaim;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/assets/serveAssetByClaim.js","module.exports = require(\"redux-saga\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"redux-saga\"\n// module id = 87\n// module chunks = 0","module.exports = require(\"redux-saga/effects\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"redux-saga/effects\"\n// module id = 88\n// module chunks = 0","const { sendGAServeEvent } = require('helpers/googleAnalytics');\nconst {\n  determineResponseType,\n  flipClaimNameAndIdForBackwardsCompatibility,\n  logRequestData,\n  getClaimIdAndServeAsset,\n} = require('helpers/serveHelpers.js');\nconst lbryUri = require('helpers/lbryUri.js');\nconst handleShowRender = require('helpers/handleShowRender.jsx');\n\nconst SERVE = 'SERVE';\n\n/*\n\n  route to serve an asset or the react app via the claim name and an identifier\n\n*/\n\nconst serverAssetByIdentifierAndClaim = (req, res) => {\n  const { headers, ip, originalUrl, params } = req;\n  // decide if this is a show request\n  let hasFileExtension;\n  try {\n    ({ hasFileExtension } = lbryUri.parseModifier(params.claim));\n  } catch (error) {\n    return res.status(400).json({success: false, message: error.message});\n  }\n  let responseType = determineResponseType(hasFileExtension, headers);\n  if (responseType !== SERVE) {\n    return handleShowRender(req, res);\n  }\n  // handle serve request\n  // send google analytics\n  sendGAServeEvent(headers, ip, originalUrl);\n  // parse the claim\n  let claimName;\n  try {\n    ({ claimName } = lbryUri.parseClaim(params.claim));\n  } catch (error) {\n    return res.status(400).json({success: false, message: error.message});\n  }\n  // parse the identifier\n  let isChannel, channelName, channelClaimId, claimId;\n  try {\n    ({ isChannel, channelName, channelClaimId, claimId } = lbryUri.parseIdentifier(params.identifier));\n  } catch (error) {\n    return res.status(400).json({success: false, message: error.message});\n  }\n  if (!isChannel) {\n    [claimId, claimName] = flipClaimNameAndIdForBackwardsCompatibility(claimId, claimName);\n  }\n  // log the request data for debugging\n  logRequestData(responseType, claimName, channelName, claimId);\n  // get the claim Id and then serve the asset\n  getClaimIdAndServeAsset(channelName, channelClaimId, claimName, claimId, originalUrl, ip, res);\n};\n\nmodule.exports = serverAssetByIdentifierAndClaim;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/assets/serveAssetByIdentifierAndClaim.js","const handlePageRequest = require('./sendReactApp');\r\n\r\nmodule.exports = (app) => {\r\n  app.get('*', handlePageRequest);\r\n};\r\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/fallback/index.js","const handlePageRender = require('helpers/handlePageRender.jsx');\n\nconst sendReactApp = (req, res) => {\n  handlePageRender(req, res);\n};\n\nmodule.exports = sendReactApp;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/fallback/sendReactApp.js"],"sourceRoot":""} \ No newline at end of file diff --git a/index.js.map b/index.js.map index e3976587..b73e3880 100644 --- a/index.js.map +++ b/index.js.map @@ -1 +1 @@ -{"version":3,"sources":["webpack:///index.js","webpack:///webpack/bootstrap 997e9b26fb4c0319bb2a","webpack:///external \"winston\"","webpack:///./server/helpers/errorHandlers.js","webpack:///./server/models/index.js","webpack:///./config/siteConfig.js","webpack:///./server/helpers/googleAnalytics.js","webpack:///./server/helpers/lbryApi.js","webpack:///./server/controllers/serveController.js","webpack:///./server/speechPassport/index.js","webpack:///./server/controllers/publishController.js","webpack:///./server/helpers/publishHelpers.js","webpack:///./config/mysqlConfig.js","webpack:///external \"passport-local\"","webpack:///./server/helpers/sequelizeHelpers.js","webpack:///external \"sequelize\"","webpack:///./server/helpers/handlePageRender.jsx","webpack:///external \"react\"","webpack:///external \"react-dom/server\"","webpack:///external \"redux\"","webpack:///external \"react-redux\"","webpack:///external \"react-router-dom\"","webpack:///external \"spee.ch-components\"","webpack:///./server/helpers/renderFullPage.js","webpack:///external \"react-helmet\"","webpack:///./server/helpers/serveHelpers.js","webpack:///./server/helpers/lbryUri.js","webpack:///./server/helpers/handleShowRender.jsx","webpack:///external \"babel-polyfill\"","webpack:///external \"whatwg-fetch\"","webpack:///./speech.js","webpack:///./server/index.js","webpack:///external \"express\"","webpack:///external \"body-parser\"","webpack:///external \"express-handlebars\"","webpack:///external \"handlebars\"","webpack:///external \"helmet\"","webpack:///external \"cookie-session\"","webpack:///external \"http\"","webpack:///./server/middleware/requestLogger.js","webpack:///external \"path\"","webpack:///./config/loggerConfig.js","webpack:///./config/slackConfig.js","webpack:///external \"winston-slack-webhook\"","webpack:///external \"passport\"","webpack:///./server/speechPassport/local-login.js","webpack:///./server/models/certificate.js","webpack:///./server/models/channel.js","webpack:///./server/models/claim.js","webpack:///./server/models/file.js","webpack:///./server/models/request.js","webpack:///./server/models/user.js","webpack:///external \"bcrypt\"","webpack:///./server/speechPassport/local-signup.js","webpack:///external \"axios\"","webpack:///./config/lbryConfig.js","webpack:///external \"universal-analytics\"","webpack:///./server/helpers/authHelpers.js","webpack:///./server/routes/auth/index.js","webpack:///./server/routes/auth/signup.js","webpack:///./server/routes/auth/login.js","webpack:///./server/routes/auth/logout.js","webpack:///./server/routes/auth/user.js","webpack:///./server/routes/api/index.js","webpack:///./server/routes/api/channelAvailability.js","webpack:///external \"fs\"","webpack:///./server/routes/api/channelClaims.js","webpack:///./server/helpers/channelPagination.js","webpack:///./server/routes/api/channelData.js","webpack:///./server/routes/api/channelShortId.js","webpack:///./server/routes/api/claimAvailability.js","webpack:///./server/routes/api/claimData.js","webpack:///./server/routes/api/claimGet.js","webpack:///./server/routes/api/claimLongId.js","webpack:///./server/routes/api/claimPublish.js","webpack:///./server/auth/authentication.js","webpack:///./server/routes/api/claimResolve.js","webpack:///./server/routes/api/claimShortId.js","webpack:///./server/routes/api/claimList.js","webpack:///./server/routes/api/fileAvailability.js","webpack:///./server/helpers/multipartMiddleware.js","webpack:///external \"connect-multiparty\"","webpack:///./server/routes/pages/index.js","webpack:///./server/routes/pages/sendReactApp.js","webpack:///./server/routes/pages/sendEmbedPage.js","webpack:///./server/routes/pages/redirect.js","webpack:///./server/routes/assets/index.js","webpack:///./server/routes/assets/serveAssetByClaim.js","webpack:///external \"redux-saga\"","webpack:///external \"redux-saga/effects\"","webpack:///./server/routes/assets/serveAssetByIdentifierAndClaim.js","webpack:///./server/routes/fallback/index.js","webpack:///./server/routes/fallback/sendReactApp.js"],"names":["module","exports","modules","__webpack_require__","moduleId","installedModules","i","l","call","m","c","d","name","getter","o","Object","defineProperty","configurable","enumerable","get","n","__esModule","object","property","prototype","hasOwnProperty","p","s","require","_slicedToArray","sliceIterator","arr","_arr","_n","_d","_e","undefined","_s","_i","Symbol","iterator","next","done","push","value","length","err","Array","isArray","TypeError","logger","handleErrorResponse","originalUrl","ip","error","res","useObjectPropertiesIfNoKeys","_module$exports$retur","returnErrorMessageAndStatus","_module$exports$retur2","status","message","json","createErrorResponsePayload","code","keys","newErrorObject","getOwnPropertyNames","forEach","key","success","Certificate","Channel","Claim","File","Request","User","Sequelize","_require","database","username","password","sequelize","host","dialect","dialectOptions","decimalNumbers","logging","pool","max","min","idle","acquire","authenticate","then","info","catch","db","import","modelName","associate","upsert","Model","values","condition","tableName","findOne","where","obj","debug","update","create","SiteConfig","_this","this","analytics","googleId","assetDefaults","description","thumbnail","title","auth","sessionKey","customComponents","components","containers","pages","details","port","twitter","publishing","additionalClaimAddresses","disabled","disabledMessage","primaryClaimAddress","thumbnailChannel","thumbnailChannelId","uploadDirectory","routes","config","console","log","createServeEventParams","headers","eventCategory","eventAction","eventLabel","ipOverride","userAgentOverride","createPublishTimingEventParams","category","variable","label","startTime","endTime","userTimingCategory","userTimingVariableName","userTimingTime","userTimingLabel","sendGoogleAnalyticsEvent","params","visitorId","replace","ua","strictCidFormat","https","event","sendGoogleAnalyticsTiming","timing","sendGAServeEvent","sendGATimingEvent","chooseGaLbrynetPublishLabel","_ref","channelName","channel_name","channelId","channel_id","axios","_require$api","api","apiHost","apiPort","lbryApiUri","_require2","handleLbrynetResponse","resolve","reject","data","result","Error","JSON","stringify","publishClaim","publishParams","gaStartTime","Date","now","Promise","post","method","response","getClaim","uri","timeout","getClaimList","claimName","resolveUri","_ref2","getDownloadDirectory","_ref3","download_directory","createChannel","amount","returnPaginatedChannelClaims","getClaimId","channelClaimId","claimId","getClaimIdByChannel","getClaimIdByClaim","getLongClaimId","longClaimId","getLongChannelId","longChannelId","all","getClaimIdByLongChannelId","getChannelData","page","longChannelClaimId","getShortChannelIdFromLongChannelId","_ref4","shortChannelClaimId","getChannelClaims","getAllChannelClaims","_ref5","_ref6","channelClaimsArray","paginatedChannelViewData","getLocalFileRecord","file","dataValues","passport","localLoginStrategy","localSignupStrategy","serializeSpeechUser","deserializeSpeechUser","deserializeUser","serializeUser","use","_defineProperty","writable","lbryApi","publishHelpers","_require$publishing","Op","publish","fileName","fileType","publishResults","certificateId","tx","channel","fileRecord","claim_id","metadata","address","claim_address","outpoint","txid","nout","height","filePath","file_path","nsfw","claimRecord","contentType","bid","upsertCriteria","claim","setClaim","setFile","deleteTemporaryFile","claimNameIsAvailable","claimAddresses","findAll","attributes","or","checkChannelAvailability","fs","parsePublishApiRequestBody","license","exec","parsePublishApiRequestFiles","path","type","size","test","validateFileTypeAndSize","thumbnailFileName","thumbnailFilePath","thumbnailFileType","createBasicPublishParams","trim","author","language","createThumbnailPublishParams","unlink","addGetResultsToFileData","fileInfo","getResult","file_name","download_path","createFileData","mysql","warn","returnShortId","claimsArray","longId","claimIndex","shortId","substring","shortIdLength","findIndex","element","possibleMatches","slice","filter","_interopRequireDefault","default","_react","_react2","_server","_redux","_reactRedux","_reactRouterDom","_spee","_renderFullPage","_renderFullPage2","_reactHelmet","_reactHelmet2","req","context","store","createStore","Reducers","html","renderToString","createElement","Provider","StaticRouter","location","url","GAListener","App","helmet","renderStatic","redirect","preloadedState","getState","send","toString","meta","link","clientAcceptsHtml","accept","match","requestIsFromBrowser","clientWantsAsset","range","imageIsWanted","videoIsWanted","isValidClaimId","isValidShortId","isValidShortIdOrClaimId","input","serveAssetToClient","NO_FILE","verbose","sendFileOptions","X-Content-Type-Options","Content-Type","sendFile","getClaimIdAndServeAsset","fullClaimId","determineResponseType","hasFileExtension","responseType","flipClaimNameAndIdForBackwardsCompatibility","identifier","tempName","logRequestData","REGEXP_INVALID_CLAIM","REGEXP_INVALID_CHANNEL","REGEXP_ADDRESS","CHANNEL_CHAR","parseIdentifier","componentsRegex","RegExp","_componentsRegex$exec","map","_componentsRegex$exec2","proto","modifierSeperator","modifier","isChannel","startsWith","nameBadChars","join","parseClaim","_componentsRegex$exec3","_componentsRegex$exec4","parseModifier","_componentsRegex$exec5","_componentsRegex$exec6","_reduxSaga","_reduxSaga2","_effects","returnSagaWithParams","saga","regeneratorRuntime","mark","_callee","wrap","_context","prev","stop","sagaMiddleware","middleware","applyMiddleware","action","Actions","onHandleShowPageUri","Sagas","handleShowPageUri","run","Server","configureLogger","userConfig","loggerConfig","configureMysql","mysqlConfig","configureSiteDetails","siteConfig","configureSlack","slackConfig","configureClientBundle","configureModels","configureRoutes","createApp","app","express","enable","publicFolder","Path","process","cwd","static","publicPath","__dirname","bodyParser","urlencoded","extended","requestLogger","speechPassport","cookieSession","maxAge","initialize","session","hbs","expressHandlebars","defaultLayout","handlebars","Handlebars","engine","set","server","http","start","PORT","sync","listen","LoggerConfig","logLevel","configure","transports","Console","level","timestamp","colorize","prettyPrint","handleExceptions","humanReadableUnhandledException","silly","SlackConfig","slackWebHook","slackErrorChannel","slackInfoChannel","winston","add","winstonSlackWebHook","webhookUrl","iconEmoji","SlackWebHook","PassportLocalStrategy","Strategy","returnUserAndChannelInfo","userInstance","userInfo","id","userName","getChannel","shortChannelId","usernameField","passwordField","user","comparePassword","isMatch","STRING","BOOLEAN","INTEGER","TEXT","DECIMAL","define","claimSequence","decodedClaim","depth","effectiveAmount","hasSignature","hex","validAtHeight","valueVersion","claimType","certificateVersion","keyType","publicKey","freezeTableName","belongsTo","foreignKey","allowNull","order","getLongChannelIdFromShortChannelId","_this2","$like","getLongChannelIdFromChannelName","_this3","validateLongChannelId","_this4","hasOne","determineFileExtensionFromContentType","determineThumbnail","storedThumbnail","defaultThumbnail","prepareClaimData","licenseUrl","preview","metadataVersion","source","sourceType","sourceVersion","streamVersion","getShortClaimIdFromLongClaimId","raw","getLongClaimIdFromShortClaimId","getTopFreeClaimIdByClaimName","_this5","validateLongClaimId","_this6","resolveClaim","_this7","claimArray","defaultValue","trendingEligible","hasMany","getRecentClaims","limit","ipAddress","bcrypt","compare","changePassword","newPassword","genSalt","saltError","salt","hash","hashError","hook","options","userData","channelData","certificateData","newUser","newChannel","newCertificate","setChannel","setUser","lbryConfig","handleSignupRequest","handleLoginRequest","handleLogoutRequest","handleUserRequest","signup","login","logIn","logout","channelAvailability","channelClaims","channelShortId","claimAvailability","claimData","claimGet","claimLongId","claimPublish","claimResolve","claimShortId","claimList","fileAvailability","multipartMiddleware","_require3","availableName","body","claims","totalPages","determineTotalPages","paginationPage","getPageFromQuery","extractPageFromClaims","previousPage","determinePreviousPage","currentPage","nextPage","determineNextPage","totalResults","determineTotalClaims","parseInt","pageNumber","claimStartIndex","claimEndIndex","totalClaims","fullPages","Math","floor","channelShortIdRoute","claimInfo","resolveResult","fileData","_ref5$","completed","authenticateUser","_require4","_require5","_require6","files","channelPassword","_parsePublishApiReque","_parsePublishApiReque2","_ref3$","thumbnailPublishParams","lbryTx","authenticateChannelCredentials","userPassword","channelFindParams","resolvedUri","claimsList","multipart","uploadDir","handlePageRequest","handleEmbedRequest","handlePageRender","sendReactApp","sendEmbedPage","render","layout","route","serveAssetByClaim","serveAssetByIdentifierAndClaim","lbryUri","handleShowRender","serverAssetByClaim","serverAssetByIdentifierAndClaim","_lbryUri$parseIdentif","_flipClaimNameAndIdFo","_flipClaimNameAndIdFo2"],"mappings":"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","file":"index.js","sourcesContent":["module.exports =\n/******/ (function(modules) { // webpackBootstrap\n/******/ \t// The module cache\n/******/ \tvar installedModules = {};\n/******/\n/******/ \t// The require function\n/******/ \tfunction __webpack_require__(moduleId) {\n/******/\n/******/ \t\t// Check if module is in cache\n/******/ \t\tif(installedModules[moduleId]) {\n/******/ \t\t\treturn installedModules[moduleId].exports;\n/******/ \t\t}\n/******/ \t\t// Create a new module (and put it into the cache)\n/******/ \t\tvar module = installedModules[moduleId] = {\n/******/ \t\t\ti: moduleId,\n/******/ \t\t\tl: false,\n/******/ \t\t\texports: {}\n/******/ \t\t};\n/******/\n/******/ \t\t// Execute the module function\n/******/ \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n/******/\n/******/ \t\t// Flag the module as loaded\n/******/ \t\tmodule.l = true;\n/******/\n/******/ \t\t// Return the exports of the module\n/******/ \t\treturn module.exports;\n/******/ \t}\n/******/\n/******/\n/******/ \t// expose the modules object (__webpack_modules__)\n/******/ \t__webpack_require__.m = modules;\n/******/\n/******/ \t// expose the module cache\n/******/ \t__webpack_require__.c = installedModules;\n/******/\n/******/ \t// define getter function for harmony exports\n/******/ \t__webpack_require__.d = function(exports, name, getter) {\n/******/ \t\tif(!__webpack_require__.o(exports, name)) {\n/******/ \t\t\tObject.defineProperty(exports, name, {\n/******/ \t\t\t\tconfigurable: false,\n/******/ \t\t\t\tenumerable: true,\n/******/ \t\t\t\tget: getter\n/******/ \t\t\t});\n/******/ \t\t}\n/******/ \t};\n/******/\n/******/ \t// getDefaultExport function for compatibility with non-harmony modules\n/******/ \t__webpack_require__.n = function(module) {\n/******/ \t\tvar getter = module && module.__esModule ?\n/******/ \t\t\tfunction getDefault() { return module['default']; } :\n/******/ \t\t\tfunction getModuleExports() { return module; };\n/******/ \t\t__webpack_require__.d(getter, 'a', getter);\n/******/ \t\treturn getter;\n/******/ \t};\n/******/\n/******/ \t// Object.prototype.hasOwnProperty.call\n/******/ \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n/******/\n/******/ \t// __webpack_public_path__\n/******/ \t__webpack_require__.p = \"/\";\n/******/\n/******/ \t// Load entry module and return exports\n/******/ \treturn __webpack_require__(__webpack_require__.s = 26);\n/******/ })\n/************************************************************************/\n/******/ ([\n/* 0 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"winston\");\n\n/***/ }),\n/* 1 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i[\"return\"]) _i[\"return\"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError(\"Invalid attempt to destructure non-iterable instance\"); } }; }();\n\nvar logger = __webpack_require__(0);\n\nmodule.exports = {\n handleErrorResponse: function handleErrorResponse(originalUrl, ip, error, res) {\n logger.error('Error on ' + originalUrl, module.exports.useObjectPropertiesIfNoKeys(error));\n\n var _module$exports$retur = module.exports.returnErrorMessageAndStatus(error),\n _module$exports$retur2 = _slicedToArray(_module$exports$retur, 2),\n status = _module$exports$retur2[0],\n message = _module$exports$retur2[1];\n\n res.status(status).json(module.exports.createErrorResponsePayload(status, message));\n },\n returnErrorMessageAndStatus: function returnErrorMessageAndStatus(error) {\n var status = void 0,\n message = void 0;\n // check for daemon being turned off\n if (error.code === 'ECONNREFUSED') {\n status = 503;\n message = 'Connection refused. The daemon may not be running.';\n // fallback for everything else\n } else {\n status = 400;\n if (error.message) {\n message = error.message;\n } else {\n message = error;\n };\n };\n return [status, message];\n },\n useObjectPropertiesIfNoKeys: function useObjectPropertiesIfNoKeys(err) {\n if (Object.keys(err).length === 0) {\n var newErrorObject = {};\n Object.getOwnPropertyNames(err).forEach(function (key) {\n newErrorObject[key] = err[key];\n });\n return newErrorObject;\n }\n return err;\n },\n createErrorResponsePayload: function createErrorResponsePayload(status, message) {\n return {\n status: status,\n success: false,\n message: message\n };\n }\n};\n\n/***/ }),\n/* 2 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar Certificate = __webpack_require__(45);\nvar Channel = __webpack_require__(46);\nvar Claim = __webpack_require__(47);\nvar File = __webpack_require__(48);\nvar Request = __webpack_require__(49);\nvar User = __webpack_require__(50);\n\nvar Sequelize = __webpack_require__(13);\nvar logger = __webpack_require__(0);\n\nvar _require = __webpack_require__(10),\n database = _require.database,\n username = _require.username,\n password = _require.password;\n\n// set sequelize options\n\n\nvar sequelize = new Sequelize(database, username, password, {\n host: 'localhost',\n dialect: 'mysql',\n dialectOptions: { decimalNumbers: true },\n logging: false,\n pool: {\n max: 5,\n min: 0,\n idle: 10000,\n acquire: 10000\n }\n});\n\n// establish mysql connection\nsequelize.authenticate().then(function () {\n logger.info('Sequelize has established mysql connection successfully.');\n}).catch(function (err) {\n logger.error('Sequelize was unable to connect to the database:', err);\n});\n\n// manually add each model to the db object (note: make this dynamic)\nvar db = {};\ndb['Certificate'] = sequelize.import('Certificate', Certificate);\ndb['Channel'] = sequelize.import('Channel', Channel);\ndb['Claim'] = sequelize.import('Claim', Claim);\ndb['File'] = sequelize.import('File', File);\ndb['Request'] = sequelize.import('Request', Request);\ndb['User'] = sequelize.import('User', User);\n\n// run model.association for each model in the db object that has an association\nlogger.info('associating db models...');\nObject.keys(db).forEach(function (modelName) {\n if (db[modelName].associate) {\n logger.info('Associating model:', modelName);\n db[modelName].associate(db);\n }\n});\n\n// add sequelize/Sequelize to db\ndb.sequelize = sequelize;\ndb.Sequelize = Sequelize;\n// add an 'upsert' method to the db object\ndb.upsert = function (Model, values, condition, tableName) {\n return Model.findOne({\n where: condition\n }).then(function (obj) {\n if (obj) {\n // update\n logger.debug('updating record in db.' + tableName);\n return obj.update(values);\n } else {\n // insert\n logger.debug('creating record in db.' + tableName);\n return Model.create(values);\n }\n }).catch(function (error) {\n logger.error(tableName + '.upsert error', error);\n throw error;\n });\n};\n\nmodule.exports = db;\n\n/***/ }),\n/* 3 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nfunction SiteConfig() {\n var _this = this;\n\n this.analytics = {\n googleId: 'default'\n };\n this.assetDefaults = {\n description: 'An asset published on Spee.ch',\n thumbnail: 'https://spee.ch/assets/img/video_thumb_default.png',\n title: 'Spee.ch'\n };\n this.auth = {\n sessionKey: 'default'\n };\n this.customComponents = {\n components: {},\n containers: {},\n pages: {}\n };\n this.details = {\n description: 'Open-source, decentralized image and video sharing.',\n host: 'default',\n port: 3000,\n title: 'Spee.ch',\n twitter: '@spee_ch'\n };\n this.publishing = {\n additionalClaimAddresses: [],\n disabled: false,\n disabledMessage: 'Please check back soon.',\n primaryClaimAddress: 'default',\n thumbnailChannel: 'default',\n thumbnailChannelId: 'default',\n uploadDirectory: '/home/lbry/Uploads'\n };\n this.routes = {};\n this.update = function (config) {\n if (!config) {\n return console.log('No site config received.');\n }\n var analytics = config.analytics,\n assetDefaults = config.assetDefaults,\n auth = config.auth,\n customComponents = config.customComponents,\n details = config.details,\n publishing = config.publishing,\n routes = config.routes;\n\n console.log('Configuring site details...');\n _this.analytics = analytics;\n _this.assetDefaults = assetDefaults;\n _this.auth = auth;\n _this.details = details;\n _this.publishing = publishing;\n _this.customComponents = customComponents;\n _this.routes = routes;\n };\n};\n\nmodule.exports = new SiteConfig();\n\n/***/ }),\n/* 4 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar logger = __webpack_require__(0);\nvar ua = __webpack_require__(55);\n\nvar _require = __webpack_require__(3),\n googleId = _require.analytics.googleId,\n title = _require.details.title;\n\nfunction createServeEventParams(headers, ip, originalUrl) {\n return {\n eventCategory: 'client requests',\n eventAction: 'serve request',\n eventLabel: originalUrl,\n ipOverride: ip,\n userAgentOverride: headers['user-agent']\n };\n};\n\nfunction createPublishTimingEventParams(category, variable, label, startTime, endTime) {\n var duration = endTime - startTime;\n return {\n userTimingCategory: category,\n userTimingVariableName: variable,\n userTimingTime: duration,\n userTimingLabel: label\n };\n};\n\nfunction sendGoogleAnalyticsEvent(ip, params) {\n var visitorId = ip.replace(/\\./g, '-');\n var visitor = ua(googleId, visitorId, { strictCidFormat: false, https: true });\n visitor.event(params, function (err) {\n if (err) {\n logger.error('Google Analytics Event Error >>', err);\n }\n });\n};\n\nfunction sendGoogleAnalyticsTiming(visitorId, params) {\n var visitor = ua(googleId, visitorId, { strictCidFormat: false, https: true });\n visitor.timing(params, function (err) {\n if (err) {\n logger.error('Google Analytics Event Error >>', err);\n }\n logger.debug('Timing event successfully sent to google analytics');\n });\n};\n\nmodule.exports = {\n sendGAServeEvent: function sendGAServeEvent(headers, ip, originalUrl) {\n var params = createServeEventParams(headers, ip, originalUrl);\n sendGoogleAnalyticsEvent(ip, params);\n },\n sendGATimingEvent: function sendGATimingEvent(category, variable, label, startTime, endTime) {\n var params = createPublishTimingEventParams(category, variable, label, startTime, endTime);\n sendGoogleAnalyticsTiming(title, params);\n },\n chooseGaLbrynetPublishLabel: function chooseGaLbrynetPublishLabel(_ref) {\n var channelName = _ref.channel_name,\n channelId = _ref.channel_id;\n\n return channelName || channelId ? 'PUBLISH_IN_CHANNEL_CLAIM' : 'PUBLISH_ANONYMOUS_CLAIM';\n }\n};\n\n/***/ }),\n/* 5 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar axios = __webpack_require__(53);\nvar logger = __webpack_require__(0);\n\nvar _require = __webpack_require__(54),\n _require$api = _require.api,\n apiHost = _require$api.apiHost,\n apiPort = _require$api.apiPort;\n\nvar lbryApiUri = 'http://' + apiHost + ':' + apiPort;\n\nvar _require2 = __webpack_require__(4),\n chooseGaLbrynetPublishLabel = _require2.chooseGaLbrynetPublishLabel,\n sendGATimingEvent = _require2.sendGATimingEvent;\n\nvar handleLbrynetResponse = function handleLbrynetResponse(_ref, resolve, reject) {\n var data = _ref.data;\n\n logger.debug('lbry api data:', data);\n if (data.result) {\n // check for an error\n if (data.result.error) {\n logger.debug('Lbrynet api error:', data.result.error);\n reject(new Error(data.result.error));\n return;\n };\n resolve(data.result);\n return;\n }\n // fallback in case it just timed out\n reject(JSON.stringify(data));\n};\n\nmodule.exports = {\n publishClaim: function publishClaim(publishParams) {\n logger.debug('lbryApi >> Publishing claim to \"' + publishParams.name + '\"');\n var gaStartTime = Date.now();\n return new Promise(function (resolve, reject) {\n axios.post(lbryApiUri, {\n method: 'publish',\n params: publishParams\n }).then(function (response) {\n sendGATimingEvent('lbrynet', 'publish', chooseGaLbrynetPublishLabel(publishParams), gaStartTime, Date.now());\n handleLbrynetResponse(response, resolve, reject);\n }).catch(function (error) {\n reject(error);\n });\n });\n },\n getClaim: function getClaim(uri) {\n logger.debug('lbryApi >> Getting Claim for \"' + uri + '\"');\n var gaStartTime = Date.now();\n return new Promise(function (resolve, reject) {\n axios.post(lbryApiUri, {\n method: 'get',\n params: { uri: uri, timeout: 20 }\n }).then(function (response) {\n sendGATimingEvent('lbrynet', 'getClaim', 'GET', gaStartTime, Date.now());\n handleLbrynetResponse(response, resolve, reject);\n }).catch(function (error) {\n reject(error);\n });\n });\n },\n getClaimList: function getClaimList(claimName) {\n logger.debug('lbryApi >> Getting claim_list for \"' + claimName + '\"');\n var gaStartTime = Date.now();\n return new Promise(function (resolve, reject) {\n axios.post(lbryApiUri, {\n method: 'claim_list',\n params: { name: claimName }\n }).then(function (response) {\n sendGATimingEvent('lbrynet', 'getClaimList', 'CLAIM_LIST', gaStartTime, Date.now());\n handleLbrynetResponse(response, resolve, reject);\n }).catch(function (error) {\n reject(error);\n });\n });\n },\n resolveUri: function resolveUri(uri) {\n logger.debug('lbryApi >> Resolving URI for \"' + uri + '\"');\n var gaStartTime = Date.now();\n return new Promise(function (resolve, reject) {\n axios.post(lbryApiUri, {\n method: 'resolve',\n params: { uri: uri }\n }).then(function (_ref2) {\n var data = _ref2.data;\n\n sendGATimingEvent('lbrynet', 'resolveUri', 'RESOLVE', gaStartTime, Date.now());\n if (data.result[uri].error) {\n // check for errors\n reject(data.result[uri].error);\n } else {\n // if no errors, resolve\n resolve(data.result[uri]);\n }\n }).catch(function (error) {\n reject(error);\n });\n });\n },\n getDownloadDirectory: function getDownloadDirectory() {\n logger.debug('lbryApi >> Retrieving the download directory path from lbry daemon...');\n var gaStartTime = Date.now();\n return new Promise(function (resolve, reject) {\n axios.post(lbryApiUri, {\n method: 'settings_get'\n }).then(function (_ref3) {\n var data = _ref3.data;\n\n sendGATimingEvent('lbrynet', 'getDownloadDirectory', 'SETTINGS_GET', gaStartTime, Date.now());\n if (data.result) {\n resolve(data.result.download_directory);\n } else {\n return new Error('Successfully connected to lbry daemon, but unable to retrieve the download directory.');\n }\n }).catch(function (error) {\n logger.error('Lbrynet Error:', error);\n resolve('/home/lbry/Downloads/');\n });\n });\n },\n createChannel: function createChannel(name) {\n logger.debug('lbryApi >> Creating channel for ' + name + '...');\n var gaStartTime = Date.now();\n return new Promise(function (resolve, reject) {\n axios.post(lbryApiUri, {\n method: 'channel_new',\n params: {\n channel_name: name,\n amount: 0.1\n }\n }).then(function (response) {\n sendGATimingEvent('lbrynet', 'createChannel', 'CHANNEL_NEW', gaStartTime, Date.now());\n handleLbrynetResponse(response, resolve, reject);\n }).catch(function (error) {\n reject(error);\n });\n });\n }\n};\n\n/***/ }),\n/* 6 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i[\"return\"]) _i[\"return\"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError(\"Invalid attempt to destructure non-iterable instance\"); } }; }();\n\nvar db = __webpack_require__(2);\nvar logger = __webpack_require__(0);\n\nvar _require = __webpack_require__(66),\n returnPaginatedChannelClaims = _require.returnPaginatedChannelClaims;\n\nvar NO_CHANNEL = 'NO_CHANNEL';\nvar NO_CLAIM = 'NO_CLAIM';\nvar NO_FILE = 'NO_FILE';\n\nmodule.exports = {\n getClaimId: function getClaimId(channelName, channelClaimId, name, claimId) {\n if (channelName) {\n return module.exports.getClaimIdByChannel(channelName, channelClaimId, name);\n } else {\n return module.exports.getClaimIdByClaim(name, claimId);\n }\n },\n getClaimIdByClaim: function getClaimIdByClaim(claimName, claimId) {\n logger.debug('getClaimIdByClaim(' + claimName + ', ' + claimId + ')');\n return new Promise(function (resolve, reject) {\n db.Claim.getLongClaimId(claimName, claimId).then(function (longClaimId) {\n if (!longClaimId) {\n resolve(NO_CLAIM);\n }\n resolve(longClaimId);\n }).catch(function (error) {\n reject(error);\n });\n });\n },\n getClaimIdByChannel: function getClaimIdByChannel(channelName, channelClaimId, claimName) {\n logger.debug('getClaimIdByChannel(' + channelName + ', ' + channelClaimId + ', ' + claimName + ')');\n return new Promise(function (resolve, reject) {\n db.Certificate.getLongChannelId(channelName, channelClaimId) // 1. get the long channel id\n .then(function (longChannelId) {\n if (!longChannelId) {\n return [null, null];\n }\n return Promise.all([longChannelId, db.Claim.getClaimIdByLongChannelId(longChannelId, claimName)]); // 2. get the long claim id\n }).then(function (_ref) {\n var _ref2 = _slicedToArray(_ref, 2),\n longChannelId = _ref2[0],\n longClaimId = _ref2[1];\n\n if (!longChannelId) {\n return resolve(NO_CHANNEL);\n }\n if (!longClaimId) {\n return resolve(NO_CLAIM);\n }\n resolve(longClaimId);\n }).catch(function (error) {\n reject(error);\n });\n });\n },\n getChannelData: function getChannelData(channelName, channelClaimId, page) {\n return new Promise(function (resolve, reject) {\n // 1. get the long channel Id (make sure channel exists)\n db.Certificate.getLongChannelId(channelName, channelClaimId).then(function (longChannelClaimId) {\n if (!longChannelClaimId) {\n return [null, null, null];\n }\n // 2. get the short ID and all claims for that channel\n return Promise.all([longChannelClaimId, db.Certificate.getShortChannelIdFromLongChannelId(longChannelClaimId, channelName)]);\n }).then(function (_ref3) {\n var _ref4 = _slicedToArray(_ref3, 2),\n longChannelClaimId = _ref4[0],\n shortChannelClaimId = _ref4[1];\n\n if (!longChannelClaimId) {\n return resolve(NO_CHANNEL);\n }\n // 3. return all the channel information\n resolve({\n channelName: channelName,\n longChannelClaimId: longChannelClaimId,\n shortChannelClaimId: shortChannelClaimId\n });\n }).catch(function (error) {\n reject(error);\n });\n });\n },\n getChannelClaims: function getChannelClaims(channelName, channelClaimId, page) {\n return new Promise(function (resolve, reject) {\n // 1. get the long channel Id (make sure channel exists)\n db.Certificate.getLongChannelId(channelName, channelClaimId).then(function (longChannelClaimId) {\n if (!longChannelClaimId) {\n return [null, null, null];\n }\n // 2. get the short ID and all claims for that channel\n return Promise.all([longChannelClaimId, db.Claim.getAllChannelClaims(longChannelClaimId)]);\n }).then(function (_ref5) {\n var _ref6 = _slicedToArray(_ref5, 2),\n longChannelClaimId = _ref6[0],\n channelClaimsArray = _ref6[1];\n\n if (!longChannelClaimId) {\n return resolve(NO_CHANNEL);\n }\n // 3. format the data for the view, including pagination\n var paginatedChannelViewData = returnPaginatedChannelClaims(channelName, longChannelClaimId, channelClaimsArray, page);\n // 4. return all the channel information and contents\n resolve(paginatedChannelViewData);\n }).catch(function (error) {\n reject(error);\n });\n });\n },\n getLocalFileRecord: function getLocalFileRecord(claimId, name) {\n return db.File.findOne({ where: { claimId: claimId, name: name } }).then(function (file) {\n if (!file) {\n return NO_FILE;\n }\n return file.dataValues;\n });\n }\n};\n\n/***/ }),\n/* 7 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar passport = __webpack_require__(43);\nvar localLoginStrategy = __webpack_require__(44);\nvar localSignupStrategy = __webpack_require__(52);\n\nvar _require = __webpack_require__(56),\n serializeSpeechUser = _require.serializeSpeechUser,\n deserializeSpeechUser = _require.deserializeSpeechUser;\n\npassport.deserializeUser(deserializeSpeechUser);\npassport.serializeUser(serializeSpeechUser);\npassport.use('local-login', localLoginStrategy);\npassport.use('local-signup', localSignupStrategy);\n\nmodule.exports = passport;\n\n/***/ }),\n/* 8 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i[\"return\"]) _i[\"return\"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError(\"Invalid attempt to destructure non-iterable instance\"); } }; }();\n\nfunction _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }\n\nvar logger = __webpack_require__(0);\nvar db = __webpack_require__(2);\nvar lbryApi = __webpack_require__(5);\nvar publishHelpers = __webpack_require__(9);\n\nvar _require = __webpack_require__(3),\n _require$publishing = _require.publishing,\n primaryClaimAddress = _require$publishing.primaryClaimAddress,\n additionalClaimAddresses = _require$publishing.additionalClaimAddresses;\n\nvar Sequelize = __webpack_require__(13);\nvar Op = Sequelize.Op;\n\nmodule.exports = {\n publish: function publish(publishParams, fileName, fileType) {\n return new Promise(function (resolve, reject) {\n var publishResults = void 0,\n certificateId = void 0,\n channelName = void 0;\n // publish the file\n return lbryApi.publishClaim(publishParams).then(function (tx) {\n logger.info('Successfully published ' + publishParams.name + ' ' + fileName, tx);\n publishResults = tx;\n // get the channel information\n if (publishParams.channel_name) {\n logger.debug('this claim was published in channel: ' + publishParams.channel_name);\n return db.Channel.findOne({\n where: {\n channelName: publishParams.channel_name\n }\n });\n } else {\n logger.debug('this claim was not published in a channel');\n return null;\n }\n }).then(function (channel) {\n // set channel information\n certificateId = null;\n channelName = null;\n if (channel) {\n certificateId = channel.channelClaimId;\n channelName = channel.channelName;\n }\n logger.debug('certificateId: ' + certificateId);\n }).then(function () {\n // create the File record\n var fileRecord = {\n name: publishParams.name,\n claimId: publishResults.claim_id,\n title: publishParams.metadata.title,\n description: publishParams.metadata.description,\n address: publishParams.claim_address,\n outpoint: publishResults.txid + ':' + publishResults.nout,\n height: 0,\n fileName: fileName,\n filePath: publishParams.file_path,\n fileType: fileType,\n nsfw: publishParams.metadata.nsfw\n };\n // create the Claim record\n var claimRecord = {\n name: publishParams.name,\n claimId: publishResults.claim_id,\n title: publishParams.metadata.title,\n description: publishParams.metadata.description,\n address: publishParams.claim_address,\n thumbnail: publishParams.metadata.thumbnail,\n outpoint: publishResults.txid + ':' + publishResults.nout,\n height: 0,\n contentType: fileType,\n nsfw: publishParams.metadata.nsfw,\n amount: publishParams.bid,\n certificateId: certificateId,\n channelName: channelName\n };\n // upsert criteria\n var upsertCriteria = {\n name: publishParams.name,\n claimId: publishResults.claim_id\n };\n // upsert the records\n return Promise.all([db.upsert(db.File, fileRecord, upsertCriteria, 'File'), db.upsert(db.Claim, claimRecord, upsertCriteria, 'Claim')]);\n }).then(function (_ref) {\n var _ref2 = _slicedToArray(_ref, 2),\n file = _ref2[0],\n claim = _ref2[1];\n\n logger.debug('File and Claim records successfully created');\n return Promise.all([file.setClaim(claim), claim.setFile(file)]);\n }).then(function () {\n logger.debug('File and Claim records successfully associated');\n resolve(publishResults); // resolve the promise with the result from lbryApi.publishClaim;\n }).catch(function (error) {\n logger.error('PUBLISH ERROR', error);\n publishHelpers.deleteTemporaryFile(publishParams.file_path); // delete the local file\n reject(error);\n });\n });\n },\n claimNameIsAvailable: function claimNameIsAvailable(name) {\n var claimAddresses = additionalClaimAddresses || [];\n claimAddresses.push(primaryClaimAddress);\n // find any records where the name is used\n return db.Claim.findAll({\n attributes: ['address'],\n where: {\n name: name,\n address: _defineProperty({}, Op.or, claimAddresses)\n }\n }).then(function (result) {\n if (result.length >= 1) {\n throw new Error('That claim is already in use');\n };\n return name;\n }).catch(function (error) {\n throw error;\n });\n },\n checkChannelAvailability: function checkChannelAvailability(name) {\n return db.Channel.findAll({\n where: { channelName: name }\n }).then(function (result) {\n if (result.length >= 1) {\n throw new Error('That channel has already been claimed');\n }\n return name;\n }).catch(function (error) {\n throw error;\n });\n }\n};\n\n/***/ }),\n/* 9 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar logger = __webpack_require__(0);\nvar fs = __webpack_require__(64);\n\nvar _require = __webpack_require__(3),\n details = _require.details,\n publishing = _require.publishing;\n\nmodule.exports = {\n parsePublishApiRequestBody: function parsePublishApiRequestBody(_ref) {\n var name = _ref.name,\n nsfw = _ref.nsfw,\n license = _ref.license,\n title = _ref.title,\n description = _ref.description,\n thumbnail = _ref.thumbnail;\n\n // validate name\n if (!name) {\n throw new Error('no name field found in request');\n }\n var invalidNameCharacters = /[^A-Za-z0-9,-]/.exec(name);\n if (invalidNameCharacters) {\n throw new Error('The claim name you provided is not allowed. Only the following characters are allowed: A-Z, a-z, 0-9, and \"-\"');\n }\n // optional parameters\n nsfw = nsfw === 'true';\n license = license || null;\n title = title || null;\n description = description || null;\n thumbnail = thumbnail || null;\n // return results\n return {\n name: name,\n nsfw: nsfw,\n license: license,\n title: title,\n description: description,\n thumbnail: thumbnail\n };\n },\n parsePublishApiRequestFiles: function parsePublishApiRequestFiles(_ref2) {\n var file = _ref2.file,\n thumbnail = _ref2.thumbnail;\n\n // make sure a file was provided\n if (!file) {\n throw new Error('no file with key of [file] found in request');\n }\n if (!file.path) {\n throw new Error('no file path found');\n }\n if (!file.type) {\n throw new Error('no file type found');\n }\n if (!file.size) {\n throw new Error('no file type found');\n }\n // validate the file name\n if (/'/.test(file.name)) {\n throw new Error('apostrophes are not allowed in the file name');\n }\n // validate the file\n module.exports.validateFileTypeAndSize(file);\n // return results\n return {\n fileName: file.name,\n filePath: file.path,\n fileType: file.type,\n thumbnailFileName: thumbnail ? thumbnail.name : null,\n thumbnailFilePath: thumbnail ? thumbnail.path : null,\n thumbnailFileType: thumbnail ? thumbnail.type : null\n };\n },\n validateFileTypeAndSize: function validateFileTypeAndSize(file) {\n // check file type and size\n switch (file.type) {\n case 'image/jpeg':\n case 'image/jpg':\n case 'image/png':\n if (file.size > 10000000) {\n logger.debug('publish > file validation > .jpeg/.jpg/.png was too big');\n throw new Error('Sorry, images are limited to 10 megabytes.');\n }\n break;\n case 'image/gif':\n if (file.size > 50000000) {\n logger.debug('publish > file validation > .gif was too big');\n throw new Error('Sorry, .gifs are limited to 50 megabytes.');\n }\n break;\n case 'video/mp4':\n if (file.size > 50000000) {\n logger.debug('publish > file validation > .mp4 was too big');\n throw new Error('Sorry, videos are limited to 50 megabytes.');\n }\n break;\n default:\n logger.debug('publish > file validation > unrecognized file type');\n throw new Error('The ' + file.type + ' content type is not supported. Only, .jpeg, .png, .gif, and .mp4 files are currently supported.');\n }\n return file;\n },\n createBasicPublishParams: function createBasicPublishParams(filePath, name, title, description, license, nsfw, thumbnail) {\n logger.debug('Creating Publish Parameters');\n // provide defaults for title\n if (title === null || title.trim() === '') {\n title = name;\n }\n // provide default for description\n if (description === null || description.trim() === '') {\n description = '';\n }\n // provide default for license\n if (license === null || license.trim() === '') {\n license = ' '; // default to empty string\n }\n // create the publish params\n var publishParams = {\n name: name,\n file_path: filePath,\n bid: 0.01,\n metadata: {\n description: description,\n title: title,\n author: details.title,\n language: 'en',\n license: license,\n nsfw: nsfw\n },\n claim_address: publishing.primaryClaimAddress\n };\n // add thumbnail to channel if video\n if (thumbnail) {\n publishParams['metadata']['thumbnail'] = thumbnail;\n }\n return publishParams;\n },\n createThumbnailPublishParams: function createThumbnailPublishParams(thumbnailFilePath, claimName, license, nsfw) {\n if (!thumbnailFilePath) {\n return;\n }\n logger.debug('Creating Thumbnail Publish Parameters');\n // create the publish params\n return {\n name: claimName + '-thumb',\n file_path: thumbnailFilePath,\n bid: 0.01,\n metadata: {\n title: claimName + ' thumbnail',\n description: 'a thumbnail for ' + claimName,\n author: details.title,\n language: 'en',\n license: license,\n nsfw: nsfw\n },\n claim_address: publishing.primaryClaimAddress,\n channel_name: publishing.thumbnailChannel,\n channel_id: publishing.thumbnailChannelId\n };\n },\n deleteTemporaryFile: function deleteTemporaryFile(filePath) {\n fs.unlink(filePath, function (err) {\n if (err) {\n logger.error('error deleting temporary file ' + filePath);\n throw err;\n }\n logger.debug('successfully deleted ' + filePath);\n });\n },\n addGetResultsToFileData: function addGetResultsToFileData(fileInfo, getResult) {\n fileInfo.fileName = getResult.file_name;\n fileInfo.filePath = getResult.download_path;\n return fileInfo;\n },\n createFileData: function createFileData(_ref3) {\n var name = _ref3.name,\n claimId = _ref3.claimId,\n outpoint = _ref3.outpoint,\n height = _ref3.height,\n address = _ref3.address,\n nsfw = _ref3.nsfw,\n contentType = _ref3.contentType;\n\n return {\n name: name,\n claimId: claimId,\n outpoint: outpoint,\n height: height,\n address: address,\n fileName: '',\n filePath: '',\n fileType: contentType,\n nsfw: nsfw\n };\n }\n};\n\n/***/ }),\n/* 10 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar logger = __webpack_require__(0);\n\nfunction mysql() {\n var _this = this;\n\n this.database = 'default';\n this.username = 'default';\n this.password = 'default';\n this.update = function (config) {\n if (!config) {\n return logger.warn('No MySQL config received.');\n }\n // configure credentials\n logger.info('configuring mysql...');\n var database = config.database,\n username = config.username,\n password = config.password;\n\n _this.database = database;\n _this.username = username;\n _this.password = password;\n };\n};\n\nmodule.exports = new mysql();\n\n/***/ }),\n/* 11 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"passport-local\");\n\n/***/ }),\n/* 12 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nmodule.exports = {\n returnShortId: function returnShortId(claimsArray, longId) {\n var claimIndex = void 0;\n var shortId = longId.substring(0, 1); // default short id is the first letter\n var shortIdLength = 0;\n // find the index of this claim id\n claimIndex = claimsArray.findIndex(function (element) {\n return element.claimId === longId;\n });\n if (claimIndex < 0) {\n throw new Error('claim id not found in claims list');\n }\n // get an array of all claims with lower height\n var possibleMatches = claimsArray.slice(0, claimIndex);\n // remove certificates with the same prefixes until none are left.\n while (possibleMatches.length > 0) {\n shortIdLength += 1;\n shortId = longId.substring(0, shortIdLength);\n possibleMatches = possibleMatches.filter(function (element) {\n return element.claimId && element.claimId.substring(0, shortIdLength) === shortId;\n });\n }\n return shortId;\n }\n};\n\n/***/ }),\n/* 13 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"sequelize\");\n\n/***/ }),\n/* 14 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _react = __webpack_require__(15);\n\nvar _react2 = _interopRequireDefault(_react);\n\nvar _server = __webpack_require__(16);\n\nvar _redux = __webpack_require__(17);\n\nvar _reactRedux = __webpack_require__(18);\n\nvar _reactRouterDom = __webpack_require__(19);\n\nvar _spee = __webpack_require__(20);\n\nvar _renderFullPage = __webpack_require__(21);\n\nvar _renderFullPage2 = _interopRequireDefault(_renderFullPage);\n\nvar _reactHelmet = __webpack_require__(22);\n\nvar _reactHelmet2 = _interopRequireDefault(_reactHelmet);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\nmodule.exports = function (req, res) {\n var context = {};\n\n // create a new Redux store instance\n var store = (0, _redux.createStore)(_spee.Reducers);\n\n // render component to a string\n var html = (0, _server.renderToString)(_react2.default.createElement(\n _reactRedux.Provider,\n { store: store },\n _react2.default.createElement(\n _reactRouterDom.StaticRouter,\n { location: req.url, context: context },\n _react2.default.createElement(\n _spee.GAListener,\n null,\n _react2.default.createElement(_spee.App, null)\n )\n )\n ));\n\n // get head tags from helmet\n var helmet = _reactHelmet2.default.renderStatic();\n\n // check for a redirect\n if (context.url) {\n // Somewhere a `` was rendered\n return res.redirect(301, context.url);\n } else {}\n // we're good, send the response\n\n\n // get the initial state from our Redux store\n var preloadedState = store.getState();\n\n // send the rendered page back to the client\n res.send((0, _renderFullPage2.default)(helmet, html, preloadedState));\n};\n\n/***/ }),\n/* 15 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"react\");\n\n/***/ }),\n/* 16 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"react-dom/server\");\n\n/***/ }),\n/* 17 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"redux\");\n\n/***/ }),\n/* 18 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"react-redux\");\n\n/***/ }),\n/* 19 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"react-router-dom\");\n\n/***/ }),\n/* 20 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"spee.ch-components\");\n\n/***/ }),\n/* 21 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nmodule.exports = function (helmet, html, preloadedState) {\n // take the html and preloadedState and return the full page\n return '\\n \\n \\n \\n \\n \\n \\n \\n ' + helmet.title.toString() + '\\n ' + helmet.meta.toString() + '\\n ' + helmet.link.toString() + '\\n \\n \\n \\n \\n \\n \\n \\n \\n
\\n
' + html + '
\\n
\\n \\n \\n \\n \\n ';\n};\n\n/***/ }),\n/* 22 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"react-helmet\");\n\n/***/ }),\n/* 23 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar logger = __webpack_require__(0);\n\nvar _require = __webpack_require__(6),\n getClaimId = _require.getClaimId,\n getLocalFileRecord = _require.getLocalFileRecord;\n\nvar _require2 = __webpack_require__(1),\n handleErrorResponse = _require2.handleErrorResponse;\n\nvar SERVE = 'SERVE';\nvar SHOW = 'SHOW';\nvar NO_FILE = 'NO_FILE';\nvar NO_CHANNEL = 'NO_CHANNEL';\nvar NO_CLAIM = 'NO_CLAIM';\n\nfunction clientAcceptsHtml(_ref) {\n var accept = _ref.accept;\n\n return accept && accept.match(/text\\/html/);\n};\n\nfunction requestIsFromBrowser(headers) {\n return headers['user-agent'] && headers['user-agent'].match(/Mozilla/);\n};\n\nfunction clientWantsAsset(_ref2) {\n var accept = _ref2.accept,\n range = _ref2.range;\n\n var imageIsWanted = accept && accept.match(/image\\/.*/) && !accept.match(/text\\/html/) && !accept.match(/text\\/\\*/);\n var videoIsWanted = accept && range;\n return imageIsWanted || videoIsWanted;\n};\n\nfunction isValidClaimId(claimId) {\n return claimId.length === 40 && !/[^A-Za-z0-9]/g.test(claimId);\n};\n\nfunction isValidShortId(claimId) {\n return claimId.length === 1; // it should really evaluate the short url itself\n};\n\nfunction isValidShortIdOrClaimId(input) {\n return isValidClaimId(input) || isValidShortId(input);\n};\n\nfunction serveAssetToClient(claimId, name, res) {\n return getLocalFileRecord(claimId, name).then(function (fileRecord) {\n // check that a local record was found\n if (fileRecord === NO_FILE) {\n return res.status(307).redirect('/api/claim/get/' + name + '/' + claimId);\n }\n // serve the file\n var filePath = fileRecord.filePath,\n fileType = fileRecord.fileType;\n\n logger.verbose('serving file: ' + filePath);\n var sendFileOptions = {\n headers: {\n 'X-Content-Type-Options': 'nosniff',\n 'Content-Type': fileType || 'image/jpeg'\n }\n };\n res.status(200).sendFile(filePath, sendFileOptions);\n }).catch(function (error) {\n throw error;\n });\n};\n\nmodule.exports = {\n getClaimIdAndServeAsset: function getClaimIdAndServeAsset(channelName, channelClaimId, claimName, claimId, originalUrl, ip, res) {\n // get the claim Id and then serve the asset\n getClaimId(channelName, channelClaimId, claimName, claimId).then(function (fullClaimId) {\n if (fullClaimId === NO_CLAIM) {\n return res.status(404).json({ success: false, message: 'no claim id could be found' });\n } else if (fullClaimId === NO_CHANNEL) {\n return res.status(404).json({ success: false, message: 'no channel id could be found' });\n }\n serveAssetToClient(fullClaimId, claimName, res);\n // postToStats(responseType, originalUrl, ip, claimName, fullClaimId, 'success');\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n // postToStats(responseType, originalUrl, ip, claimName, fullClaimId, 'fail');\n });\n },\n determineResponseType: function determineResponseType(hasFileExtension, headers) {\n var responseType = void 0;\n if (hasFileExtension) {\n responseType = SERVE; // assume a serve request if file extension is present\n if (clientAcceptsHtml(headers)) {\n // if the request comes from a browser, change it to a show request\n responseType = SHOW;\n }\n } else {\n responseType = SHOW;\n if (clientWantsAsset(headers) && requestIsFromBrowser(headers)) {\n // this is in case someone embeds a show url\n logger.debug('Show request came from browser but wants an image/video. Changing response to serve...');\n responseType = SERVE;\n }\n }\n return responseType;\n },\n flipClaimNameAndIdForBackwardsCompatibility: function flipClaimNameAndIdForBackwardsCompatibility(identifier, name) {\n // this is a patch for backwards compatability with '/name/claim_id' url format\n if (isValidShortIdOrClaimId(name) && !isValidShortIdOrClaimId(identifier)) {\n var tempName = name;\n name = identifier;\n identifier = tempName;\n }\n return [identifier, name];\n },\n logRequestData: function logRequestData(responseType, claimName, channelName, claimId) {\n logger.debug('responseType ===', responseType);\n logger.debug('claim name === ', claimName);\n logger.debug('channel name ===', channelName);\n logger.debug('claim id ===', claimId);\n }\n};\n\n/***/ }),\n/* 24 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i[\"return\"]) _i[\"return\"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError(\"Invalid attempt to destructure non-iterable instance\"); } }; }();\n\nvar logger = __webpack_require__(0);\n\nmodule.exports = {\n REGEXP_INVALID_CLAIM: /[^A-Za-z0-9-]/g,\n REGEXP_INVALID_CHANNEL: /[^A-Za-z0-9-@]/g,\n REGEXP_ADDRESS: /^b(?=[^0OIl]{32,33})[0-9A-Za-z]{32,33}$/,\n CHANNEL_CHAR: '@',\n parseIdentifier: function parseIdentifier(identifier) {\n logger.debug('parsing identifier:', identifier);\n var componentsRegex = new RegExp('([^:$#/]*)' + // value (stops at the first separator or end)\n '([:$#]?)([^/]*)' // modifier separator, modifier (stops at the first path separator or end)\n );\n\n var _componentsRegex$exec = componentsRegex.exec(identifier).map(function (match) {\n return match || null;\n }),\n _componentsRegex$exec2 = _slicedToArray(_componentsRegex$exec, 4),\n proto = _componentsRegex$exec2[0],\n value = _componentsRegex$exec2[1],\n modifierSeperator = _componentsRegex$exec2[2],\n modifier = _componentsRegex$exec2[3];\n\n logger.debug(proto + ', ' + value + ', ' + modifierSeperator + ', ' + modifier);\n\n // Validate and process name\n if (!value) {\n throw new Error('Check your url. No channel name provided before \"' + modifierSeperator + '\"');\n }\n var isChannel = value.startsWith(module.exports.CHANNEL_CHAR);\n var channelName = isChannel ? value : null;\n var claimId = void 0;\n if (isChannel) {\n if (!channelName) {\n throw new Error('No channel name after @.');\n }\n var nameBadChars = channelName.match(module.exports.REGEXP_INVALID_CHANNEL);\n if (nameBadChars) {\n throw new Error('Invalid characters in channel name: ' + nameBadChars.join(', ') + '.');\n }\n } else {\n claimId = value;\n }\n\n // Validate and process modifier\n var channelClaimId = void 0;\n if (modifierSeperator) {\n if (!modifier) {\n throw new Error('No modifier provided after separator \"' + modifierSeperator + '\"');\n }\n\n if (modifierSeperator === ':') {\n channelClaimId = modifier;\n } else {\n throw new Error('The \"' + modifierSeperator + '\" modifier is not currently supported');\n }\n }\n return {\n isChannel: isChannel,\n channelName: channelName,\n channelClaimId: channelClaimId,\n claimId: claimId\n };\n },\n parseClaim: function parseClaim(claim) {\n logger.debug('parsing name:', claim);\n var componentsRegex = new RegExp('([^:$#/.]*)' + // name (stops at the first modifier)\n '([:$#.]?)([^/]*)' // modifier separator, modifier (stops at the first path separator or end)\n );\n\n var _componentsRegex$exec3 = componentsRegex.exec(claim).map(function (match) {\n return match || null;\n }),\n _componentsRegex$exec4 = _slicedToArray(_componentsRegex$exec3, 4),\n proto = _componentsRegex$exec4[0],\n claimName = _componentsRegex$exec4[1],\n modifierSeperator = _componentsRegex$exec4[2],\n modifier = _componentsRegex$exec4[3];\n\n logger.debug(proto + ', ' + claimName + ', ' + modifierSeperator + ', ' + modifier);\n\n // Validate and process name\n if (!claimName) {\n throw new Error('No claim name provided before .');\n }\n var nameBadChars = claimName.match(module.exports.REGEXP_INVALID_CLAIM);\n if (nameBadChars) {\n throw new Error('Invalid characters in claim name: ' + nameBadChars.join(', ') + '.');\n }\n // Validate and process modifier\n if (modifierSeperator) {\n if (!modifier) {\n throw new Error('No file extension provided after separator ' + modifierSeperator + '.');\n }\n if (modifierSeperator !== '.') {\n throw new Error('The ' + modifierSeperator + ' modifier is not supported in the claim name');\n }\n }\n // return results\n return {\n claimName: claimName\n };\n },\n parseModifier: function parseModifier(claim) {\n logger.debug('parsing modifier:', claim);\n var componentsRegex = new RegExp('([^:$#/.]*)' + // name (stops at the first modifier)\n '([:$#.]?)([^/]*)' // modifier separator, modifier (stops at the first path separator or end)\n );\n\n var _componentsRegex$exec5 = componentsRegex.exec(claim).map(function (match) {\n return match || null;\n }),\n _componentsRegex$exec6 = _slicedToArray(_componentsRegex$exec5, 4),\n proto = _componentsRegex$exec6[0],\n claimName = _componentsRegex$exec6[1],\n modifierSeperator = _componentsRegex$exec6[2],\n modifier = _componentsRegex$exec6[3];\n\n logger.debug(proto + ', ' + claimName + ', ' + modifierSeperator + ', ' + modifier);\n // Validate and process modifier\n var hasFileExtension = false;\n if (modifierSeperator) {\n hasFileExtension = true;\n }\n return {\n hasFileExtension: hasFileExtension\n };\n }\n};\n\n/***/ }),\n/* 25 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _react = __webpack_require__(15);\n\nvar _react2 = _interopRequireDefault(_react);\n\nvar _server = __webpack_require__(16);\n\nvar _redux = __webpack_require__(17);\n\nvar _reactRedux = __webpack_require__(18);\n\nvar _reactRouterDom = __webpack_require__(19);\n\nvar _renderFullPage = __webpack_require__(21);\n\nvar _renderFullPage2 = _interopRequireDefault(_renderFullPage);\n\nvar _reduxSaga = __webpack_require__(87);\n\nvar _reduxSaga2 = _interopRequireDefault(_reduxSaga);\n\nvar _effects = __webpack_require__(88);\n\nvar _spee = __webpack_require__(20);\n\nvar _reactHelmet = __webpack_require__(22);\n\nvar _reactHelmet2 = _interopRequireDefault(_reactHelmet);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\nvar returnSagaWithParams = function returnSagaWithParams(saga, params) {\n return (/*#__PURE__*/regeneratorRuntime.mark(function _callee() {\n return regeneratorRuntime.wrap(function _callee$(_context) {\n while (1) {\n switch (_context.prev = _context.next) {\n case 0:\n _context.next = 2;\n return (0, _effects.call)(saga, params);\n\n case 2:\n case 'end':\n return _context.stop();\n }\n }\n }, _callee, this);\n })\n );\n};\n\nmodule.exports = function (req, res) {\n var context = {};\n\n // create and apply middleware\n var sagaMiddleware = (0, _reduxSaga2.default)();\n var middleware = (0, _redux.applyMiddleware)(sagaMiddleware);\n\n // create a new Redux store instance\n var store = (0, _redux.createStore)(_spee.Reducers, middleware);\n\n // create saga\n var action = _spee.Actions.onHandleShowPageUri(req.params);\n var saga = returnSagaWithParams(_spee.Sagas.handleShowPageUri, action);\n\n // run the saga middleware\n sagaMiddleware.run(saga).done.then(function () {\n // render component to a string\n var html = (0, _server.renderToString)(_react2.default.createElement(\n _reactRedux.Provider,\n { store: store },\n _react2.default.createElement(\n _reactRouterDom.StaticRouter,\n { location: req.url, context: context },\n _react2.default.createElement(\n _spee.GAListener,\n null,\n _react2.default.createElement(_spee.App, null)\n )\n )\n ));\n\n // get head tags from helmet\n var helmet = _reactHelmet2.default.renderStatic();\n\n // check for a redirect\n if (context.url) {\n return res.redirect(301, context.url);\n }\n\n // get the initial state from our Redux store\n var preloadedState = store.getState();\n\n // send the rendered page back to the client\n res.send((0, _renderFullPage2.default)(helmet, html, preloadedState));\n });\n};\n\n/***/ }),\n/* 26 */\n/***/ (function(module, exports, __webpack_require__) {\n\n__webpack_require__(27);\n__webpack_require__(28);\nmodule.exports = __webpack_require__(29);\n\n\n/***/ }),\n/* 27 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"babel-polyfill\");\n\n/***/ }),\n/* 28 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"whatwg-fetch\");\n\n/***/ }),\n/* 29 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar Server = __webpack_require__(30);\n\nvar _exports = {\n Server: Server\n};\n\nmodule.exports = _exports;\n\n/***/ }),\n/* 30 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\n// app dependencies\nvar express = __webpack_require__(31);\nvar bodyParser = __webpack_require__(32);\nvar expressHandlebars = __webpack_require__(33);\nvar Handlebars = __webpack_require__(34);\nvar helmet = __webpack_require__(35);\nvar cookieSession = __webpack_require__(36);\nvar http = __webpack_require__(37);\nvar logger = __webpack_require__(0);\nvar requestLogger = __webpack_require__(38);\nvar Path = __webpack_require__(39);\nvar loggerConfig = __webpack_require__(40);\nvar mysqlConfig = __webpack_require__(10);\nvar siteConfig = __webpack_require__(3);\nvar slackConfig = __webpack_require__(41);\n\nfunction Server() {\n var _this = this;\n\n this.configureLogger = function (userConfig) {\n loggerConfig.update(userConfig);\n };\n this.configureMysql = function (userConfig) {\n mysqlConfig.update(userConfig);\n };\n this.configureSiteDetails = function (userConfig) {\n siteConfig.update(userConfig);\n };\n this.configureSlack = function (userConfig) {\n slackConfig.update(userConfig);\n };\n this.configureClientBundle = function () {\n logger.debug('configure the client here by passing in the bundle and configuring it, or better yet: taking in the components to use dynamically from here.');\n };\n this.configureModels = function () {\n logger.debug('here is where you could add/overwrite the default models');\n };\n this.configureRoutes = function () {\n logger.debug('here is where you could add/overwrite the default routes');\n };\n this.createApp = function () {\n // create an Express application\n var app = express();\n\n // trust the proxy to get ip address for us\n app.enable('trust proxy');\n\n /* add middleware */\n // set HTTP headers to protect against well-known web vulnerabilties\n app.use(helmet());\n // 'express.static' to serve static files from public directory\n if (siteConfig.routes.publicFolder) {\n // take in a different public folder, so it can serve it's own bundle if needed\n var publicFolder = Path.resolve(process.cwd(), siteConfig.routes.publicFolder);\n app.use('/static', express.static(publicFolder));\n logger.info('serving static files from custom path:', publicFolder);\n } else {\n var publicPath = Path.resolve(__dirname, 'public');\n app.use('/static', express.static(publicPath));\n logger.info('serving static files from default path:', publicPath);\n };\n // 'body parser' for parsing application/json\n app.use(bodyParser.json());\n // 'body parser' for parsing application/x-www-form-urlencoded\n app.use(bodyParser.urlencoded({ extended: true }));\n\n // add custom middleware (note: build out to accept dynamically use what is in server/middleware/\n app.use(requestLogger);\n\n // configure passport\n var speechPassport = __webpack_require__(7);\n // initialize passport\n var sessionKey = siteConfig.auth.sessionKey;\n app.use(cookieSession({\n name: 'session',\n keys: [sessionKey],\n maxAge: 24 * 60 * 60 * 1000 // i.e. 24 hours\n }));\n app.use(speechPassport.initialize());\n app.use(speechPassport.session());\n\n // configure handlebars & register it with express app\n var hbs = expressHandlebars.create({\n defaultLayout: 'embed',\n handlebars: Handlebars\n });\n app.engine('handlebars', hbs.engine);\n app.set('view engine', 'handlebars');\n\n // set the routes on the app\n __webpack_require__(57)(app);\n __webpack_require__(62)(app);\n __webpack_require__(81)(app);\n __webpack_require__(85)(app);\n __webpack_require__(90)(app);\n\n _this.app = app;\n };\n this.initialize = function () {\n _this.createApp();\n _this.server = http.Server(_this.app);\n };\n this.start = function () {\n var db = __webpack_require__(2);\n var PORT = siteConfig.details.port;\n // sync sequelize\n db.sequelize.sync()\n // start the server\n .then(function () {\n _this.server.listen(PORT, function () {\n logger.info('Server is listening on PORT ' + PORT);\n });\n }).catch(function (error) {\n logger.error('Startup Error:', error);\n });\n };\n};\n\nmodule.exports = Server;\n\n/***/ }),\n/* 31 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"express\");\n\n/***/ }),\n/* 32 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"body-parser\");\n\n/***/ }),\n/* 33 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"express-handlebars\");\n\n/***/ }),\n/* 34 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"handlebars\");\n\n/***/ }),\n/* 35 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"helmet\");\n\n/***/ }),\n/* 36 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"cookie-session\");\n\n/***/ }),\n/* 37 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"http\");\n\n/***/ }),\n/* 38 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar logger = __webpack_require__(0);\n\nvar requestLogger = function requestLogger(req, res, next) {\n // custom logging middleware to log all incoming http requests\n logger.verbose('Request on ' + req.originalUrl + ' from ' + req.ip);\n next();\n};\n\nmodule.exports = requestLogger;\n\n/***/ }),\n/* 39 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"path\");\n\n/***/ }),\n/* 40 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar logger = __webpack_require__(0);\n\nfunction LoggerConfig() {\n var _this = this;\n\n this.logLevel = 'debug';\n this.update = function (config) {\n if (!config) {\n return logger.warn('No logger config received.');\n }\n logger.info('configuring winston logger...');\n // update values with local config params\n var logLevel = config.logLevel;\n\n _this.logLevel = logLevel;\n // configure the winston logger\n logger.configure({\n transports: [new logger.transports.Console({\n level: _this.logLevel,\n timestamp: false,\n colorize: true,\n prettyPrint: true,\n handleExceptions: true,\n humanReadableUnhandledException: true\n })]\n });\n // test all the log levels\n logger.info('testing winston log levels...');\n logger.error('Level 0');\n logger.warn('Level 1');\n logger.info('Level 2');\n logger.verbose('Level 3');\n logger.debug('Level 4');\n logger.silly('Level 5');\n };\n};\n\nmodule.exports = new LoggerConfig();\n\n/***/ }),\n/* 41 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar winstonSlackWebHook = __webpack_require__(42).SlackWebHook;\nvar winston = __webpack_require__(0);\n\nfunction SlackConfig() {\n var _this = this;\n\n this.slackWebHook = 'default';\n this.slackErrorChannel = 'default';\n this.slackInfoChannel = 'default';\n this.update = function (config) {\n if (!config) {\n return winston.warn('No slack config received');\n }\n // update variables\n winston.info('configuring slack logger...');\n var slackWebHook = config.slackWebHook,\n slackErrorChannel = config.slackErrorChannel,\n slackInfoChannel = config.slackInfoChannel;\n\n _this.slackWebHook = slackWebHook;\n _this.slackErrorChannel = slackErrorChannel;\n _this.slackInfoChannel = slackInfoChannel;\n // update slack webhook settings\n if (_this.slackWebHook) {\n // add a transport for errors to slack\n if (_this.slackErrorChannel) {\n winston.add(winstonSlackWebHook, {\n name: 'slack-errors-transport',\n level: 'warn',\n webhookUrl: _this.slackWebHook,\n channel: _this.slackErrorChannel,\n username: 'spee.ch',\n iconEmoji: ':face_with_head_bandage:'\n });\n };\n if (slackInfoChannel) {\n winston.add(winstonSlackWebHook, {\n name: 'slack-info-transport',\n level: 'info',\n webhookUrl: _this.slackWebHook,\n channel: _this.slackInfoChannel,\n username: 'spee.ch',\n iconEmoji: ':nerd_face:'\n });\n };\n // send test messages\n winston.info('testing slack logger...');\n winston.error('Slack \"error\" logging is online.');\n winston.info('Slack \"info\" logging is online.');\n } else {\n winston.warn('Slack logging is not enabled because no slackWebHook config var provided.');\n }\n };\n};\n\nmodule.exports = new SlackConfig();\n\n/***/ }),\n/* 42 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"winston-slack-webhook\");\n\n/***/ }),\n/* 43 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"passport\");\n\n/***/ }),\n/* 44 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar PassportLocalStrategy = __webpack_require__(11).Strategy;\nvar logger = __webpack_require__(0);\nvar db = __webpack_require__(2);\n\nvar returnUserAndChannelInfo = function returnUserAndChannelInfo(userInstance) {\n return new Promise(function (resolve, reject) {\n var userInfo = {};\n userInfo['id'] = userInstance.id;\n userInfo['userName'] = userInstance.userName;\n userInstance.getChannel().then(function (_ref) {\n var channelName = _ref.channelName,\n channelClaimId = _ref.channelClaimId;\n\n userInfo['channelName'] = channelName;\n userInfo['channelClaimId'] = channelClaimId;\n return db.Certificate.getShortChannelIdFromLongChannelId(channelClaimId, channelName);\n }).then(function (shortChannelId) {\n userInfo['shortChannelId'] = shortChannelId;\n resolve(userInfo);\n }).catch(function (error) {\n reject(error);\n });\n });\n};\n\nmodule.exports = new PassportLocalStrategy({\n usernameField: 'username',\n passwordField: 'password'\n}, function (username, password, done) {\n return db.User.findOne({\n where: { userName: username }\n }).then(function (user) {\n if (!user) {\n logger.debug('no user found');\n return done(null, false, { message: 'Incorrect username or password' });\n }\n return user.comparePassword(password).then(function (isMatch) {\n if (!isMatch) {\n logger.debug('incorrect password');\n return done(null, false, { message: 'Incorrect username or password' });\n }\n logger.debug('Password was a match, returning User');\n return returnUserAndChannelInfo(user).then(function (userInfo) {\n return done(null, userInfo);\n }).catch(function (error) {\n return error;\n });\n }).catch(function (error) {\n return error;\n });\n }).catch(function (error) {\n return done(error);\n });\n});\n\n/***/ }),\n/* 45 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar logger = __webpack_require__(0);\n\nvar _require = __webpack_require__(12),\n returnShortId = _require.returnShortId;\n\nmodule.exports = function (sequelize, _ref) {\n var STRING = _ref.STRING,\n BOOLEAN = _ref.BOOLEAN,\n INTEGER = _ref.INTEGER,\n TEXT = _ref.TEXT,\n DECIMAL = _ref.DECIMAL;\n\n var Certificate = sequelize.define('Certificate', {\n address: {\n type: STRING,\n default: null\n },\n amount: {\n type: DECIMAL(19, 8),\n default: null\n },\n claimId: {\n type: STRING,\n default: null\n },\n claimSequence: {\n type: INTEGER,\n default: null\n },\n decodedClaim: {\n type: BOOLEAN,\n default: null\n },\n depth: {\n type: INTEGER,\n default: null\n },\n effectiveAmount: {\n type: DECIMAL(19, 8),\n default: null\n },\n hasSignature: {\n type: BOOLEAN,\n default: null\n },\n height: {\n type: INTEGER,\n default: null\n },\n hex: {\n type: TEXT('long'),\n default: null\n },\n name: {\n type: STRING,\n default: null\n },\n nout: {\n type: INTEGER,\n default: null\n },\n txid: {\n type: STRING,\n default: null\n },\n validAtHeight: {\n type: INTEGER,\n default: null\n },\n outpoint: {\n type: STRING,\n default: null\n },\n valueVersion: {\n type: STRING,\n default: null\n },\n claimType: {\n type: STRING,\n default: null\n },\n certificateVersion: {\n type: STRING,\n default: null\n },\n keyType: {\n type: STRING,\n default: null\n },\n publicKey: {\n type: TEXT('long'),\n default: null\n }\n }, {\n freezeTableName: true\n });\n\n Certificate.associate = function (db) {\n Certificate.belongsTo(db.Channel, {\n foreignKey: {\n allowNull: true\n }\n });\n };\n\n Certificate.getShortChannelIdFromLongChannelId = function (longChannelId, channelName) {\n var _this = this;\n\n logger.debug('getShortChannelIdFromLongChannelId ' + channelName + ':' + longChannelId);\n return new Promise(function (resolve, reject) {\n _this.findAll({\n where: { name: channelName },\n order: [['height', 'ASC']]\n }).then(function (result) {\n switch (result.length) {\n case 0:\n throw new Error('No channel(s) found with that channel name');\n default:\n return resolve(returnShortId(result, longChannelId));\n }\n }).catch(function (error) {\n reject(error);\n });\n });\n };\n\n Certificate.getLongChannelIdFromShortChannelId = function (channelName, channelClaimId) {\n var _this2 = this;\n\n logger.debug('getLongChannelIdFromShortChannelId(' + channelName + ', ' + channelClaimId + ')');\n return new Promise(function (resolve, reject) {\n _this2.findAll({\n where: {\n name: channelName,\n claimId: {\n $like: channelClaimId + '%'\n }\n },\n order: [['height', 'ASC']]\n }).then(function (result) {\n switch (result.length) {\n case 0:\n return resolve(null);\n default:\n // note results must be sorted\n return resolve(result[0].claimId);\n }\n }).catch(function (error) {\n reject(error);\n });\n });\n };\n\n Certificate.getLongChannelIdFromChannelName = function (channelName) {\n var _this3 = this;\n\n logger.debug('getLongChannelIdFromChannelName(' + channelName + ')');\n return new Promise(function (resolve, reject) {\n _this3.findAll({\n where: { name: channelName },\n order: [['effectiveAmount', 'DESC'], ['height', 'ASC']]\n }).then(function (result) {\n switch (result.length) {\n case 0:\n return resolve(null);\n default:\n return resolve(result[0].claimId);\n }\n }).catch(function (error) {\n reject(error);\n });\n });\n };\n\n Certificate.validateLongChannelId = function (name, claimId) {\n var _this4 = this;\n\n logger.debug('validateLongChannelId(' + name + ', ' + claimId + ')');\n return new Promise(function (resolve, reject) {\n _this4.findOne({\n where: { name: name, claimId: claimId }\n }).then(function (result) {\n if (!result) {\n return resolve(null);\n };\n resolve(claimId);\n }).catch(function (error) {\n reject(error);\n });\n });\n };\n\n Certificate.getLongChannelId = function (channelName, channelClaimId) {\n logger.debug('getLongChannelId(' + channelName + ', ' + channelClaimId + ')');\n if (channelClaimId && channelClaimId.length === 40) {\n // if a full channel id is provided\n return this.validateLongChannelId(channelName, channelClaimId);\n } else if (channelClaimId && channelClaimId.length < 40) {\n // if a short channel id is provided\n return this.getLongChannelIdFromShortChannelId(channelName, channelClaimId);\n } else {\n return this.getLongChannelIdFromChannelName(channelName); // if no channel id provided\n }\n };\n\n return Certificate;\n};\n\n/***/ }),\n/* 46 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nmodule.exports = function (sequelize, _ref) {\n var STRING = _ref.STRING;\n\n var Channel = sequelize.define('Channel', {\n channelName: {\n type: STRING,\n allowNull: false\n },\n channelClaimId: {\n type: STRING,\n allowNull: false\n }\n }, {\n freezeTableName: true\n });\n\n Channel.associate = function (db) {\n Channel.belongsTo(db.User);\n Channel.hasOne(db.Certificate);\n };\n\n return Channel;\n};\n\n/***/ }),\n/* 47 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar logger = __webpack_require__(0);\n\nvar _require = __webpack_require__(12),\n returnShortId = _require.returnShortId;\n\nvar _require2 = __webpack_require__(3),\n defaultThumbnail = _require2.assetDefaults.thumbnail,\n host = _require2.details.host;\n\nfunction determineFileExtensionFromContentType(contentType) {\n switch (contentType) {\n case 'image/jpeg':\n case 'image/jpg':\n return 'jpeg';\n case 'image/png':\n return 'png';\n case 'image/gif':\n return 'gif';\n case 'video/mp4':\n return 'mp4';\n default:\n logger.debug('setting unknown file type as file extension jpeg');\n return 'jpeg';\n }\n};\n\nfunction determineThumbnail(storedThumbnail, defaultThumbnail) {\n if (storedThumbnail === '') {\n return defaultThumbnail;\n }\n return storedThumbnail;\n};\n\nfunction prepareClaimData(claim) {\n // logger.debug('preparing claim data based on resolved data:', claim);\n claim['thumbnail'] = determineThumbnail(claim.thumbnail, defaultThumbnail);\n claim['fileExt'] = determineFileExtensionFromContentType(claim.contentType);\n claim['host'] = host;\n return claim;\n};\n\nmodule.exports = function (sequelize, _ref) {\n var STRING = _ref.STRING,\n BOOLEAN = _ref.BOOLEAN,\n INTEGER = _ref.INTEGER,\n TEXT = _ref.TEXT,\n DECIMAL = _ref.DECIMAL;\n\n var Claim = sequelize.define('Claim', {\n address: {\n type: STRING,\n default: null\n },\n amount: {\n type: DECIMAL(19, 8),\n default: null\n },\n claimId: {\n type: STRING,\n default: null\n },\n claimSequence: {\n type: INTEGER,\n default: null\n },\n decodedClaim: {\n type: BOOLEAN,\n default: null\n },\n depth: {\n type: INTEGER,\n default: null\n },\n effectiveAmount: {\n type: DECIMAL(19, 8),\n default: null\n },\n hasSignature: {\n type: BOOLEAN,\n default: null\n },\n height: {\n type: INTEGER,\n default: null\n },\n hex: {\n type: TEXT('long'),\n default: null\n },\n name: {\n type: STRING,\n default: null\n },\n nout: {\n type: INTEGER,\n default: null\n },\n txid: {\n type: STRING,\n default: null\n },\n validAtHeight: {\n type: INTEGER,\n default: null\n },\n outpoint: {\n type: STRING,\n default: null\n },\n claimType: {\n type: STRING,\n default: null\n },\n certificateId: {\n type: STRING,\n default: null\n },\n author: {\n type: STRING,\n default: null\n },\n description: {\n type: TEXT('long'),\n default: null\n },\n language: {\n type: STRING,\n default: null\n },\n license: {\n type: STRING,\n default: null\n },\n licenseUrl: {\n type: STRING,\n default: null\n },\n nsfw: {\n type: BOOLEAN,\n default: null\n },\n preview: {\n type: STRING,\n default: null\n },\n thumbnail: {\n type: STRING,\n default: null\n },\n title: {\n type: STRING,\n default: null\n },\n metadataVersion: {\n type: STRING,\n default: null\n },\n contentType: {\n type: STRING,\n default: null\n },\n source: {\n type: STRING,\n default: null\n },\n sourceType: {\n type: STRING,\n default: null\n },\n sourceVersion: {\n type: STRING,\n default: null\n },\n streamVersion: {\n type: STRING,\n default: null\n },\n valueVersion: {\n type: STRING,\n default: null\n },\n channelName: {\n type: STRING,\n allowNull: true,\n default: null\n }\n }, {\n freezeTableName: true\n });\n\n Claim.associate = function (db) {\n Claim.belongsTo(db.File, {\n foreignKey: {\n allowNull: true\n }\n });\n };\n\n Claim.getShortClaimIdFromLongClaimId = function (claimId, claimName) {\n var _this = this;\n\n logger.debug('Claim.getShortClaimIdFromLongClaimId for ' + claimName + '#' + claimId);\n return new Promise(function (resolve, reject) {\n _this.findAll({\n where: { name: claimName },\n order: [['height', 'ASC']]\n }).then(function (result) {\n switch (result.length) {\n case 0:\n throw new Error('No claim(s) found with that claim name');\n default:\n resolve(returnShortId(result, claimId));\n }\n }).catch(function (error) {\n reject(error);\n });\n });\n };\n\n Claim.getAllChannelClaims = function (channelClaimId) {\n var _this2 = this;\n\n logger.debug('Claim.getAllChannelClaims for ' + channelClaimId);\n return new Promise(function (resolve, reject) {\n _this2.findAll({\n where: { certificateId: channelClaimId },\n order: [['height', 'ASC']],\n raw: true // returns an array of only data, not an array of instances\n }).then(function (channelClaimsArray) {\n // logger.debug('channelclaimsarray length:', channelClaimsArray.length);\n switch (channelClaimsArray.length) {\n case 0:\n return resolve(null);\n default:\n channelClaimsArray.forEach(function (claim) {\n claim['fileExt'] = determineFileExtensionFromContentType(claim.contentType);\n claim['thumbnail'] = determineThumbnail(claim.thumbnail, defaultThumbnail);\n return claim;\n });\n return resolve(channelClaimsArray);\n }\n }).catch(function (error) {\n reject(error);\n });\n });\n };\n\n Claim.getClaimIdByLongChannelId = function (channelClaimId, claimName) {\n var _this3 = this;\n\n logger.debug('finding claim id for claim ' + claimName + ' from channel ' + channelClaimId);\n return new Promise(function (resolve, reject) {\n _this3.findAll({\n where: { name: claimName, certificateId: channelClaimId },\n order: [['id', 'ASC']]\n }).then(function (result) {\n switch (result.length) {\n case 0:\n return resolve(null);\n case 1:\n return resolve(result[0].claimId);\n default:\n logger.error(result.length + ' records found for \"' + claimName + '\" in channel \"' + channelClaimId + '\"');\n return resolve(result[0].claimId);\n }\n }).catch(function (error) {\n reject(error);\n });\n });\n };\n\n Claim.getLongClaimIdFromShortClaimId = function (name, shortId) {\n var _this4 = this;\n\n return new Promise(function (resolve, reject) {\n _this4.findAll({\n where: {\n name: name,\n claimId: {\n $like: shortId + '%'\n } },\n order: [['height', 'ASC']]\n }).then(function (result) {\n switch (result.length) {\n case 0:\n return resolve(null);\n default:\n // note results must be sorted\n return resolve(result[0].claimId);\n }\n }).catch(function (error) {\n reject(error);\n });\n });\n };\n\n Claim.getTopFreeClaimIdByClaimName = function (name) {\n var _this5 = this;\n\n return new Promise(function (resolve, reject) {\n _this5.findAll({\n where: { name: name },\n order: [['effectiveAmount', 'DESC'], ['height', 'ASC']] // note: maybe height and effective amount need to switch?\n }).then(function (result) {\n logger.debug('length of result', result.length);\n switch (result.length) {\n case 0:\n return resolve(null);\n default:\n return resolve(result[0].dataValues.claimId);\n }\n }).catch(function (error) {\n reject(error);\n });\n });\n };\n\n Claim.validateLongClaimId = function (name, claimId) {\n var _this6 = this;\n\n return new Promise(function (resolve, reject) {\n _this6.findOne({\n where: { name: name, claimId: claimId }\n }).then(function (result) {\n if (!result) {\n return resolve(null);\n };\n resolve(claimId);\n }).catch(function (error) {\n reject(error);\n });\n });\n };\n\n Claim.getLongClaimId = function (claimName, claimId) {\n logger.debug('getLongClaimId(' + claimName + ', ' + claimId + ')');\n if (claimId && claimId.length === 40) {\n // if a full claim id is provided\n return this.validateLongClaimId(claimName, claimId);\n } else if (claimId && claimId.length < 40) {\n return this.getLongClaimIdFromShortClaimId(claimName, claimId); // if a short claim id is provided\n } else {\n return this.getTopFreeClaimIdByClaimName(claimName); // if no claim id is provided\n }\n };\n\n Claim.resolveClaim = function (name, claimId) {\n var _this7 = this;\n\n logger.debug('Claim.resolveClaim: ' + name + ' ' + claimId);\n return new Promise(function (resolve, reject) {\n _this7.findAll({\n where: { name: name, claimId: claimId }\n }).then(function (claimArray) {\n switch (claimArray.length) {\n case 0:\n return resolve(null);\n case 1:\n return resolve(prepareClaimData(claimArray[0].dataValues));\n default:\n logger.error('more than one record matches ' + name + '#' + claimId + ' in db.Claim');\n return resolve(prepareClaimData(claimArray[0].dataValues));\n }\n }).catch(function (error) {\n reject(error);\n });\n });\n };\n\n return Claim;\n};\n\n/***/ }),\n/* 48 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nmodule.exports = function (sequelize, _ref) {\n var STRING = _ref.STRING,\n BOOLEAN = _ref.BOOLEAN,\n INTEGER = _ref.INTEGER;\n\n var File = sequelize.define('File', {\n name: {\n type: STRING,\n allowNull: false\n },\n claimId: {\n type: STRING,\n allowNull: false\n },\n address: {\n type: STRING,\n allowNull: false\n },\n outpoint: {\n type: STRING,\n allowNull: false\n },\n height: {\n type: INTEGER,\n allowNull: false,\n default: 0\n },\n fileName: {\n type: STRING,\n allowNull: false\n },\n filePath: {\n type: STRING,\n allowNull: false\n },\n fileType: {\n type: STRING\n },\n nsfw: {\n type: BOOLEAN,\n allowNull: false,\n defaultValue: false\n },\n trendingEligible: {\n type: BOOLEAN,\n allowNull: false,\n defaultValue: true\n }\n }, {\n freezeTableName: true\n });\n\n File.associate = function (db) {\n File.hasMany(db.Request);\n File.hasOne(db.Claim);\n };\n\n File.getRecentClaims = function () {\n return this.findAll({\n where: { nsfw: false, trendingEligible: true },\n order: [['createdAt', 'DESC']],\n limit: 25\n });\n };\n\n return File;\n};\n\n/***/ }),\n/* 49 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nmodule.exports = function (sequelize, _ref) {\n var STRING = _ref.STRING,\n BOOLEAN = _ref.BOOLEAN,\n TEXT = _ref.TEXT;\n\n var Request = sequelize.define('Request', {\n action: {\n type: STRING,\n allowNull: false\n },\n url: {\n type: STRING,\n allowNull: false\n },\n ipAddress: {\n type: STRING,\n allowNull: true\n },\n result: {\n type: TEXT('long'),\n allowNull: true,\n default: null\n }\n }, {\n freezeTableName: true\n });\n\n Request.associate = function (db) {\n Request.belongsTo(db.File, {\n foreignKey: {\n allowNull: true\n }\n });\n };\n\n return Request;\n};\n\n/***/ }),\n/* 50 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar bcrypt = __webpack_require__(51);\nvar logger = __webpack_require__(0);\n\nmodule.exports = function (sequelize, _ref) {\n var STRING = _ref.STRING;\n\n var User = sequelize.define('User', {\n userName: {\n type: STRING,\n allowNull: false\n },\n password: {\n type: STRING,\n allowNull: false\n }\n }, {\n freezeTableName: true\n });\n\n User.associate = function (db) {\n User.hasOne(db.Channel);\n };\n\n User.prototype.comparePassword = function (password) {\n return bcrypt.compare(password, this.password);\n };\n\n User.prototype.changePassword = function (newPassword) {\n var _this = this;\n\n return new Promise(function (resolve, reject) {\n // generate a salt string to use for hashing\n bcrypt.genSalt(function (saltError, salt) {\n if (saltError) {\n logger.error('salt error', saltError);\n reject(saltError);\n return;\n }\n // generate a hashed version of the user's password\n bcrypt.hash(newPassword, salt, function (hashError, hash) {\n // if there is an error with the hash generation return the error\n if (hashError) {\n logger.error('hash error', hashError);\n reject(hashError);\n return;\n }\n // replace the current password with the new hash\n _this.update({ password: hash }).then(function () {\n resolve();\n }).catch(function (error) {\n reject(error);\n });\n });\n });\n });\n };\n\n // pre-save hook method to hash the user's password before the user's info is saved to the db.\n User.hook('beforeCreate', function (user, options) {\n logger.debug('User.beforeCreate hook...');\n return new Promise(function (resolve, reject) {\n // generate a salt string to use for hashing\n bcrypt.genSalt(function (saltError, salt) {\n if (saltError) {\n logger.error('salt error', saltError);\n reject(saltError);\n return;\n }\n // generate a hashed version of the user's password\n bcrypt.hash(user.password, salt, function (hashError, hash) {\n // if there is an error with the hash generation return the error\n if (hashError) {\n logger.error('hash error', hashError);\n reject(hashError);\n return;\n }\n // replace the password string with the hash password value\n user.password = hash;\n resolve();\n });\n });\n });\n });\n\n return User;\n};\n\n/***/ }),\n/* 51 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"bcrypt\");\n\n/***/ }),\n/* 52 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i[\"return\"]) _i[\"return\"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError(\"Invalid attempt to destructure non-iterable instance\"); } }; }();\n\nvar PassportLocalStrategy = __webpack_require__(11).Strategy;\nvar lbryApi = __webpack_require__(5);\nvar logger = __webpack_require__(0);\nvar db = __webpack_require__(2);\n\nmodule.exports = new PassportLocalStrategy({\n usernameField: 'username',\n passwordField: 'password'\n}, function (username, password, done) {\n logger.verbose('new channel signup request. user: ' + username + ' pass: ' + password + ' .');\n var userInfo = {};\n // server-side validaton of inputs (username, password)\n\n // create the channel and retrieve the metadata\n return lbryApi.createChannel('@' + username).then(function (tx) {\n // create user record\n var userData = {\n userName: username,\n password: password\n };\n logger.verbose('userData >', userData);\n // create user record\n var channelData = {\n channelName: '@' + username,\n channelClaimId: tx.claim_id\n };\n logger.verbose('channelData >', channelData);\n // create certificate record\n var certificateData = {\n claimId: tx.claim_id,\n name: '@' + username\n // address,\n };\n logger.verbose('certificateData >', certificateData);\n // save user and certificate to db\n return Promise.all([db.User.create(userData), db.Channel.create(channelData), db.Certificate.create(certificateData)]);\n }).then(function (_ref) {\n var _ref2 = _slicedToArray(_ref, 3),\n newUser = _ref2[0],\n newChannel = _ref2[1],\n newCertificate = _ref2[2];\n\n logger.verbose('user and certificate successfully created');\n // store the relevant newUser info to be passed back for req.User\n userInfo['id'] = newUser.id;\n userInfo['userName'] = newUser.userName;\n userInfo['channelName'] = newChannel.channelName;\n userInfo['channelClaimId'] = newChannel.channelClaimId;\n // associate the instances\n return Promise.all([newCertificate.setChannel(newChannel), newChannel.setUser(newUser)]);\n }).then(function () {\n logger.verbose('user and certificate successfully associated');\n return db.Certificate.getShortChannelIdFromLongChannelId(userInfo.channelClaimId, userInfo.channelName);\n }).then(function (shortChannelId) {\n userInfo['shortChannelId'] = shortChannelId;\n return done(null, userInfo);\n }).catch(function (error) {\n logger.error('signup error', error);\n return done(error);\n });\n});\n\n/***/ }),\n/* 53 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"axios\");\n\n/***/ }),\n/* 54 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar lbryConfig = {\n api: {\n apiHost: 'localhost',\n apiPort: '5279'\n }\n};\n\nmodule.exports = lbryConfig;\n\n/***/ }),\n/* 55 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"universal-analytics\");\n\n/***/ }),\n/* 56 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nmodule.exports = {\n serializeSpeechUser: function serializeSpeechUser(user, done) {\n // returns user data to be serialized into session\n console.log('serializing user');\n done(null, user);\n },\n deserializeSpeechUser: function deserializeSpeechUser(user, done) {\n // deserializes session and populates additional info to req.user\n console.log('deserializing user');\n done(null, user);\n }\n};\n\n/***/ }),\n/* 57 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar speechPassport = __webpack_require__(7);\nvar handleSignupRequest = __webpack_require__(58);\nvar handleLoginRequest = __webpack_require__(59);\nvar handleLogoutRequest = __webpack_require__(60);\nvar handleUserRequest = __webpack_require__(61);\n\nmodule.exports = function (app) {\n app.post('/signup', speechPassport.authenticate('local-signup'), handleSignupRequest);\n app.post('/login', handleLoginRequest);\n app.get('/logout', handleLogoutRequest);\n app.get('/user', handleUserRequest);\n};\n\n/***/ }),\n/* 58 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar signup = function signup(req, res) {\n res.status(200).json({\n success: true,\n channelName: req.user.channelName,\n channelClaimId: req.user.channelClaimId,\n shortChannelId: req.user.shortChannelId\n });\n};\n\nmodule.exports = signup;\n\n/***/ }),\n/* 59 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar speechPassport = __webpack_require__(7);\n\nvar login = function login(req, res, next) {\n speechPassport.authenticate('local-login', function (err, user, info) {\n if (err) {\n return next(err);\n }\n if (!user) {\n return res.status(400).json({\n success: false,\n message: info.message\n });\n }\n req.logIn(user, function (err) {\n if (err) {\n return next(err);\n }\n return res.status(200).json({\n success: true,\n channelName: req.user.channelName,\n channelClaimId: req.user.channelClaimId,\n shortChannelId: req.user.shortChannelId\n });\n });\n })(req, res, next);\n};\n\nmodule.exports = login;\n\n/***/ }),\n/* 60 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar logout = function logout(req, res) {\n req.logout();\n res.status(200).json({ success: true, message: 'you successfully logged out' });\n};\n\nmodule.exports = logout;\n\n/***/ }),\n/* 61 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar user = function user(req, res) {\n if (req.user) {\n res.status(200).json({ success: true, data: req.user });\n } else {\n res.status(401).json({ success: false, message: 'user is not logged in' });\n }\n};\n\nmodule.exports = user;\n\n/***/ }),\n/* 62 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar channelAvailability = __webpack_require__(63);\nvar channelClaims = __webpack_require__(65);\nvar channelData = __webpack_require__(67);\nvar channelShortId = __webpack_require__(68);\nvar claimAvailability = __webpack_require__(69);\nvar claimData = __webpack_require__(70);\nvar claimGet = __webpack_require__(71);\nvar claimLongId = __webpack_require__(72);\nvar claimPublish = __webpack_require__(73);\nvar claimResolve = __webpack_require__(75);\nvar claimShortId = __webpack_require__(76);\nvar claimList = __webpack_require__(77);\nvar fileAvailability = __webpack_require__(78);\n\nvar multipartMiddleware = __webpack_require__(79);\n\nmodule.exports = function (app) {\n // channel routes\n app.get('/api/channel/availability/:name', channelAvailability);\n app.get('/api/channel/short-id/:longId/:name', channelShortId);\n app.get('/api/channel/data/:channelName/:channelClaimId', channelData);\n app.get('/api/channel/claims/:channelName/:channelClaimId/:page', channelClaims);\n // claim routes\n app.get('/api/claim/list/:name', claimList);\n app.get('/api/claim/get/:name/:claimId', claimGet);\n app.get('/api/claim/availability/:name', claimAvailability);\n app.get('/api/claim/resolve/:name/:claimId', claimResolve);\n app.post('/api/claim/publish', multipartMiddleware, claimPublish);\n app.get('/api/claim/short-id/:longId/:name', claimShortId);\n app.post('/api/claim/long-id', claimLongId);\n app.get('/api/claim/data/:claimName/:claimId', claimData);\n // file routes\n app.get('/api/file/availability/:name/:claimId', fileAvailability);\n};\n\n/***/ }),\n/* 63 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(8),\n checkChannelAvailability = _require.checkChannelAvailability;\n\nvar _require2 = __webpack_require__(4),\n sendGATimingEvent = _require2.sendGATimingEvent;\n\nvar _require3 = __webpack_require__(1),\n handleErrorResponse = _require3.handleErrorResponse;\n\n/*\n\n route to check whether site has published to a channel\n\n*/\n\nvar channelAvailability = function channelAvailability(_ref, res) {\n var ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n name = _ref.params.name;\n\n var gaStartTime = Date.now();\n checkChannelAvailability(name).then(function (availableName) {\n res.status(200).json(availableName);\n sendGATimingEvent('end-to-end', 'claim name availability', name, gaStartTime, Date.now());\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = channelAvailability;\n\n/***/ }),\n/* 64 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"fs\");\n\n/***/ }),\n/* 65 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(6),\n getChannelClaims = _require.getChannelClaims;\n\nvar _require2 = __webpack_require__(1),\n handleErrorResponse = _require2.handleErrorResponse;\n\nvar NO_CHANNEL = 'NO_CHANNEL';\n\n/*\n\n route to get all claims for channel\n\n*/\n\nvar channelClaims = function channelClaims(_ref, res) {\n var ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n body = _ref.body,\n params = _ref.params;\n\n var channelName = params.channelName;\n var channelClaimId = params.channelClaimId;\n if (channelClaimId === 'none') channelClaimId = null;\n var page = params.page;\n getChannelClaims(channelName, channelClaimId, page).then(function (data) {\n if (data === NO_CHANNEL) {\n return res.status(404).json({ success: false, message: 'No matching channel was found' });\n }\n res.status(200).json({ success: true, data: data });\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = channelClaims;\n\n/***/ }),\n/* 66 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar CLAIMS_PER_PAGE = 12;\n\nmodule.exports = {\n returnPaginatedChannelClaims: function returnPaginatedChannelClaims(channelName, longChannelClaimId, claims, page) {\n var totalPages = module.exports.determineTotalPages(claims);\n var paginationPage = module.exports.getPageFromQuery(page);\n var viewData = {\n channelName: channelName,\n longChannelClaimId: longChannelClaimId,\n claims: module.exports.extractPageFromClaims(claims, paginationPage),\n previousPage: module.exports.determinePreviousPage(paginationPage),\n currentPage: paginationPage,\n nextPage: module.exports.determineNextPage(totalPages, paginationPage),\n totalPages: totalPages,\n totalResults: module.exports.determineTotalClaims(claims)\n };\n return viewData;\n },\n getPageFromQuery: function getPageFromQuery(page) {\n if (page) {\n return parseInt(page);\n }\n return 1;\n },\n extractPageFromClaims: function extractPageFromClaims(claims, pageNumber) {\n if (!claims) {\n return []; // if no claims, return this default\n }\n // logger.debug('claims is array?', Array.isArray(claims));\n // logger.debug(`pageNumber ${pageNumber} is number?`, Number.isInteger(pageNumber));\n var claimStartIndex = (pageNumber - 1) * CLAIMS_PER_PAGE;\n var claimEndIndex = claimStartIndex + CLAIMS_PER_PAGE;\n var pageOfClaims = claims.slice(claimStartIndex, claimEndIndex);\n return pageOfClaims;\n },\n determineTotalPages: function determineTotalPages(claims) {\n if (!claims) {\n return 0;\n } else {\n var totalClaims = claims.length;\n if (totalClaims < CLAIMS_PER_PAGE) {\n return 1;\n }\n var fullPages = Math.floor(totalClaims / CLAIMS_PER_PAGE);\n var remainder = totalClaims % CLAIMS_PER_PAGE;\n if (remainder === 0) {\n return fullPages;\n }\n return fullPages + 1;\n }\n },\n determinePreviousPage: function determinePreviousPage(currentPage) {\n if (currentPage === 1) {\n return null;\n }\n return currentPage - 1;\n },\n determineNextPage: function determineNextPage(totalPages, currentPage) {\n if (currentPage === totalPages) {\n return null;\n }\n return currentPage + 1;\n },\n determineTotalClaims: function determineTotalClaims(claims) {\n if (!claims) {\n return 0;\n }\n return claims.length;\n }\n};\n\n/***/ }),\n/* 67 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(6),\n getChannelData = _require.getChannelData;\n\nvar _require2 = __webpack_require__(1),\n handleErrorResponse = _require2.handleErrorResponse;\n\nvar NO_CHANNEL = 'NO_CHANNEL';\n\n/*\n\n route to get data for a channel\n\n*/\n\nvar channelData = function channelData(_ref, res) {\n var ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n body = _ref.body,\n params = _ref.params;\n\n var channelName = params.channelName;\n var channelClaimId = params.channelClaimId;\n if (channelClaimId === 'none') channelClaimId = null;\n getChannelData(channelName, channelClaimId, 0).then(function (data) {\n if (data === NO_CHANNEL) {\n return res.status(404).json({ success: false, message: 'No matching channel was found' });\n }\n res.status(200).json({ success: true, data: data });\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = channelData;\n\n/***/ }),\n/* 68 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(1),\n handleErrorResponse = _require.handleErrorResponse;\n\nvar db = __webpack_require__(2);\n\n/*\n\nroute to get a short channel id from long channel Id\n\n*/\n\nvar channelShortIdRoute = function channelShortIdRoute(_ref, res) {\n var ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n params = _ref.params;\n\n db.Certificate.getShortChannelIdFromLongChannelId(params.longId, params.name).then(function (shortId) {\n res.status(200).json(shortId);\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = channelShortIdRoute;\n\n/***/ }),\n/* 69 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(8),\n claimNameIsAvailable = _require.claimNameIsAvailable;\n\nvar _require2 = __webpack_require__(4),\n sendGATimingEvent = _require2.sendGATimingEvent;\n\nvar _require3 = __webpack_require__(1),\n handleErrorResponse = _require3.handleErrorResponse;\n\n/*\n\n route to check whether this site published to a claim\n\n*/\n\nvar claimAvailability = function claimAvailability(_ref, res) {\n var ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n name = _ref.params.name;\n\n var gaStartTime = Date.now();\n claimNameIsAvailable(name).then(function (result) {\n res.status(200).json(result);\n sendGATimingEvent('end-to-end', 'claim name availability', name, gaStartTime, Date.now());\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimAvailability;\n\n/***/ }),\n/* 70 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(1),\n handleErrorResponse = _require.handleErrorResponse;\n\nvar db = __webpack_require__(2);\n\n/*\n\n route to return data for a claim\n\n*/\n\nvar claimData = function claimData(_ref, res) {\n var ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n body = _ref.body,\n params = _ref.params;\n\n var claimName = params.claimName;\n var claimId = params.claimId;\n if (claimId === 'none') claimId = null;\n db.Claim.resolveClaim(claimName, claimId).then(function (claimInfo) {\n if (!claimInfo) {\n return res.status(404).json({ success: false, message: 'No claim could be found' });\n }\n res.status(200).json({ success: true, data: claimInfo });\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimData;\n\n/***/ }),\n/* 71 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i[\"return\"]) _i[\"return\"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError(\"Invalid attempt to destructure non-iterable instance\"); } }; }();\n\nvar _require = __webpack_require__(5),\n getClaim = _require.getClaim;\n\nvar _require2 = __webpack_require__(9),\n addGetResultsToFileData = _require2.addGetResultsToFileData,\n createFileData = _require2.createFileData;\n\nvar _require3 = __webpack_require__(1),\n handleErrorResponse = _require3.handleErrorResponse;\n\nvar db = __webpack_require__(2);\n\n/*\n\n route to get a claim\n\n*/\n\nvar claimGet = function claimGet(_ref, res) {\n var ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n params = _ref.params;\n\n var name = params.name;\n var claimId = params.claimId;\n // resolve the claim\n db.Claim.resolveClaim(name, claimId).then(function (resolveResult) {\n // make sure a claim actually exists at that uri\n if (!resolveResult) {\n throw new Error('No matching uri found in Claim table');\n }\n var fileData = createFileData(resolveResult);\n // get the claim\n return Promise.all([fileData, getClaim(name + '#' + claimId)]);\n }).then(function (_ref2) {\n var _ref3 = _slicedToArray(_ref2, 2),\n fileData = _ref3[0],\n getResult = _ref3[1];\n\n fileData = addGetResultsToFileData(fileData, getResult);\n return Promise.all([db.upsert(db.File, fileData, { name: name, claimId: claimId }, 'File'), getResult]);\n }).then(function (_ref4) {\n var _ref5 = _slicedToArray(_ref4, 2),\n fileRecord = _ref5[0],\n _ref5$ = _ref5[1],\n message = _ref5$.message,\n completed = _ref5$.completed;\n\n res.status(200).json({ success: true, message: message, completed: completed });\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimGet;\n\n/***/ }),\n/* 72 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(6),\n getClaimId = _require.getClaimId;\n\nvar _require2 = __webpack_require__(1),\n handleErrorResponse = _require2.handleErrorResponse;\n\nvar NO_CHANNEL = 'NO_CHANNEL';\nvar NO_CLAIM = 'NO_CLAIM';\n\n/*\n\n route to get a long claim id\n\n*/\n\nvar claimLongId = function claimLongId(_ref, res) {\n var ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n body = _ref.body,\n params = _ref.params;\n\n var channelName = body.channelName;\n var channelClaimId = body.channelClaimId;\n var claimName = body.claimName;\n var claimId = body.claimId;\n getClaimId(channelName, channelClaimId, claimName, claimId).then(function (result) {\n if (result === NO_CHANNEL) {\n return res.status(404).json({ success: false, message: 'No matching channel could be found' });\n }\n if (result === NO_CLAIM) {\n return res.status(404).json({ success: false, message: 'No matching claim id could be found' });\n }\n res.status(200).json({ success: true, data: result });\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimLongId;\n\n/***/ }),\n/* 73 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i[\"return\"]) _i[\"return\"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError(\"Invalid attempt to destructure non-iterable instance\"); } }; }();\n\nvar _require = __webpack_require__(9),\n createBasicPublishParams = _require.createBasicPublishParams,\n createThumbnailPublishParams = _require.createThumbnailPublishParams,\n parsePublishApiRequestBody = _require.parsePublishApiRequestBody,\n parsePublishApiRequestFiles = _require.parsePublishApiRequestFiles;\n\nvar _require2 = __webpack_require__(8),\n claimNameIsAvailable = _require2.claimNameIsAvailable,\n publish = _require2.publish;\n\nvar _require3 = __webpack_require__(74),\n authenticateUser = _require3.authenticateUser;\n\nvar _require4 = __webpack_require__(4),\n sendGATimingEvent = _require4.sendGATimingEvent;\n\nvar _require5 = __webpack_require__(1),\n handleErrorResponse = _require5.handleErrorResponse;\n\nvar _require6 = __webpack_require__(3),\n host = _require6.details.host;\n\n/*\n\n route to publish a claim through the daemon\n\n*/\n\nvar claimPublish = function claimPublish(_ref, res) {\n var body = _ref.body,\n files = _ref.files,\n headers = _ref.headers,\n ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n user = _ref.user;\n\n // define variables\n var channelName = void 0,\n channelId = void 0,\n channelPassword = void 0,\n description = void 0,\n fileName = void 0,\n filePath = void 0,\n fileType = void 0,\n gaStartTime = void 0,\n license = void 0,\n name = void 0,\n nsfw = void 0,\n thumbnail = void 0,\n thumbnailFileName = void 0,\n thumbnailFilePath = void 0,\n thumbnailFileType = void 0,\n title = void 0;\n // record the start time of the request\n gaStartTime = Date.now();\n // validate the body and files of the request\n try {\n var _parsePublishApiReque = parsePublishApiRequestBody(body);\n // validateApiPublishRequest(body, files);\n\n\n name = _parsePublishApiReque.name;\n nsfw = _parsePublishApiReque.nsfw;\n license = _parsePublishApiReque.license;\n title = _parsePublishApiReque.title;\n description = _parsePublishApiReque.description;\n thumbnail = _parsePublishApiReque.thumbnail;\n\n var _parsePublishApiReque2 = parsePublishApiRequestFiles(files);\n\n fileName = _parsePublishApiReque2.fileName;\n filePath = _parsePublishApiReque2.filePath;\n fileType = _parsePublishApiReque2.fileType;\n thumbnailFileName = _parsePublishApiReque2.thumbnailFileName;\n thumbnailFilePath = _parsePublishApiReque2.thumbnailFilePath;\n thumbnailFileType = _parsePublishApiReque2.thumbnailFileType;\n channelName = body.channelName;\n channelId = body.channelId;\n channelPassword = body.channelPassword;\n } catch (error) {\n return res.status(400).json({ success: false, message: error.message });\n }\n // check channel authorization\n Promise.all([authenticateUser(channelName, channelId, channelPassword, user), claimNameIsAvailable(name), createBasicPublishParams(filePath, name, title, description, license, nsfw, thumbnail), createThumbnailPublishParams(thumbnailFilePath, name, license, nsfw)]).then(function (_ref2) {\n var _ref3 = _slicedToArray(_ref2, 4),\n _ref3$ = _ref3[0],\n channelName = _ref3$.channelName,\n channelClaimId = _ref3$.channelClaimId,\n validatedClaimName = _ref3[1],\n publishParams = _ref3[2],\n thumbnailPublishParams = _ref3[3];\n\n // add channel details to the publish params\n if (channelName && channelClaimId) {\n publishParams['channel_name'] = channelName;\n publishParams['channel_id'] = channelClaimId;\n }\n // publish the thumbnail\n if (thumbnailPublishParams) {\n publish(thumbnailPublishParams, thumbnailFileName, thumbnailFileType);\n }\n // publish the asset\n return publish(publishParams, fileName, fileType);\n }).then(function (result) {\n res.status(200).json({\n success: true,\n message: 'publish completed successfully',\n data: {\n name: name,\n claimId: result.claim_id,\n url: host + '/' + result.claim_id + '/' + name,\n lbryTx: result\n }\n });\n // record the publish end time and send to google analytics\n sendGATimingEvent('end-to-end', 'publish', fileType, gaStartTime, Date.now());\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimPublish;\n\n/***/ }),\n/* 74 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar db = __webpack_require__(2);\nvar logger = __webpack_require__(0);\n\nmodule.exports = {\n authenticateUser: function authenticateUser(channelName, channelId, channelPassword, user) {\n // case: no channelName or channel Id are provided (anonymous), regardless of whether user token is provided\n if (!channelName && !channelId) {\n return {\n channelName: null,\n channelClaimId: null\n };\n }\n // case: channelName or channel Id are provided with user token\n if (user) {\n if (channelName && channelName !== user.channelName) {\n throw new Error('the provided channel name does not match user credentials');\n }\n if (channelId && channelId !== user.channelClaimId) {\n throw new Error('the provided channel id does not match user credentials');\n }\n return {\n channelName: user.channelName,\n channelClaimId: user.channelClaimId\n };\n }\n // case: channelName or channel Id are provided with password instead of user token\n if (!channelPassword) throw new Error('no channel password provided');\n return module.exports.authenticateChannelCredentials(channelName, channelId, channelPassword);\n },\n authenticateChannelCredentials: function authenticateChannelCredentials(channelName, channelId, userPassword) {\n return new Promise(function (resolve, reject) {\n // hoisted variables\n var channelData = void 0;\n // build the params for finding the channel\n var channelFindParams = {};\n if (channelName) channelFindParams['channelName'] = channelName;\n if (channelId) channelFindParams['channelClaimId'] = channelId;\n // find the channel\n db.Channel.findOne({\n where: channelFindParams\n }).then(function (channel) {\n if (!channel) {\n logger.debug('no channel found');\n throw new Error('Authentication failed, you do not have access to that channel');\n }\n channelData = channel.get();\n logger.debug('channel data:', channelData);\n return db.User.findOne({\n where: { userName: channelData.channelName.substring(1) }\n });\n }).then(function (user) {\n if (!user) {\n logger.debug('no user found');\n throw new Error('Authentication failed, you do not have access to that channel');\n }\n return user.comparePassword(userPassword);\n }).then(function (isMatch) {\n if (!isMatch) {\n logger.debug('incorrect password');\n throw new Error('Authentication failed, you do not have access to that channel');\n }\n logger.debug('...password was a match...');\n resolve(channelData);\n }).catch(function (error) {\n reject(error);\n });\n });\n }\n};\n\n/***/ }),\n/* 75 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(5),\n resolveUri = _require.resolveUri;\n\nvar _require2 = __webpack_require__(1),\n handleErrorResponse = _require2.handleErrorResponse;\n\n/*\n\n route to run a resolve request on the daemon\n\n*/\n\nvar claimResolve = function claimResolve(_ref, res) {\n var headers = _ref.headers,\n ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n params = _ref.params;\n\n resolveUri(params.name + '#' + params.claimId).then(function (resolvedUri) {\n res.status(200).json(resolvedUri);\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimResolve;\n\n/***/ }),\n/* 76 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(1),\n handleErrorResponse = _require.handleErrorResponse;\n\nvar db = __webpack_require__(2);\n\n/*\n\n route to get a short claim id from long claim Id\n\n*/\n\nvar claimShortId = function claimShortId(_ref, res) {\n var ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n body = _ref.body,\n params = _ref.params;\n\n db.Claim.getShortClaimIdFromLongClaimId(params.longId, params.name).then(function (shortId) {\n res.status(200).json({ success: true, data: shortId });\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimShortId;\n\n/***/ }),\n/* 77 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(5),\n getClaimList = _require.getClaimList;\n\nvar _require2 = __webpack_require__(1),\n handleErrorResponse = _require2.handleErrorResponse;\n\n/*\n\n route to get list of claims\n\n*/\n\nvar claimList = function claimList(_ref, res) {\n var ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n params = _ref.params;\n\n getClaimList(params.name).then(function (claimsList) {\n res.status(200).json(claimsList);\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimList;\n\n/***/ }),\n/* 78 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(1),\n handleErrorResponse = _require.handleErrorResponse;\n\nvar db = __webpack_require__(2);\n\n/*\n\n route to see if asset is available locally\n\n*/\n\nvar fileAvailability = function fileAvailability(_ref, res) {\n var ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n params = _ref.params;\n\n var name = params.name;\n var claimId = params.claimId;\n db.File.findOne({\n where: {\n name: name,\n claimId: claimId\n }\n }).then(function (result) {\n if (result) {\n return res.status(200).json({ success: true, data: true });\n }\n res.status(200).json({ success: true, data: false });\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = fileAvailability;\n\n/***/ }),\n/* 79 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar multipart = __webpack_require__(80);\n\nvar _require = __webpack_require__(3),\n uploadDirectory = _require.publishing.uploadDirectory;\n\nvar multipartMiddleware = multipart({ uploadDir: uploadDirectory });\n\nmodule.exports = multipartMiddleware;\n\n/***/ }),\n/* 80 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"connect-multiparty\");\n\n/***/ }),\n/* 81 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar handlePageRequest = __webpack_require__(82);\nvar handleEmbedRequest = __webpack_require__(83);\nvar redirect = __webpack_require__(84);\n\nmodule.exports = function (app) {\n app.get('/', handlePageRequest);\n app.get('/login', handlePageRequest);\n app.get('/about', handlePageRequest);\n app.get('/trending', redirect('/popular'));\n app.get('/popular', handlePageRequest);\n app.get('/new', handlePageRequest);\n app.get('/embed/:claimId/:name', handleEmbedRequest); // route to send embedable video player (for twitter)\n};\n\n/***/ }),\n/* 82 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar handlePageRender = __webpack_require__(14);\n\nvar sendReactApp = function sendReactApp(req, res) {\n handlePageRender(req, res);\n};\n\nmodule.exports = sendReactApp;\n\n/***/ }),\n/* 83 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(3),\n host = _require.details.host;\n\nvar sendEmbedPage = function sendEmbedPage(_ref, res) {\n var params = _ref.params;\n\n var claimId = params.claimId;\n var name = params.name;\n // get and render the content\n res.status(200).render('embed', { layout: 'embed', host: host, claimId: claimId, name: name });\n};\n\nmodule.exports = sendEmbedPage;\n\n/***/ }),\n/* 84 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar redirect = function redirect(route) {\n return function (req, res) {\n res.status(301).redirect(route);\n };\n};\n\nmodule.exports = redirect;\n\n/***/ }),\n/* 85 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar serveAssetByClaim = __webpack_require__(86);\nvar serveAssetByIdentifierAndClaim = __webpack_require__(89);\n\nmodule.exports = function (app, db) {\n app.get('/:identifier/:claim', serveAssetByIdentifierAndClaim);\n app.get('/:claim', serveAssetByClaim);\n};\n\n/***/ }),\n/* 86 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(4),\n sendGAServeEvent = _require.sendGAServeEvent;\n\nvar _require2 = __webpack_require__(23),\n determineResponseType = _require2.determineResponseType,\n logRequestData = _require2.logRequestData,\n getClaimIdAndServeAsset = _require2.getClaimIdAndServeAsset;\n\nvar lbryUri = __webpack_require__(24);\nvar handleShowRender = __webpack_require__(25);\nvar SERVE = 'SERVE';\n\n/*\n\n route to serve an asset or the react app via the claim name only\n\n*/\n\nvar serverAssetByClaim = function serverAssetByClaim(req, res) {\n var headers = req.headers,\n ip = req.ip,\n originalUrl = req.originalUrl,\n params = req.params;\n // decide if this is a show request\n\n var hasFileExtension = void 0;\n try {\n var _lbryUri$parseModifie = lbryUri.parseModifier(params.claim);\n\n hasFileExtension = _lbryUri$parseModifie.hasFileExtension;\n } catch (error) {\n return res.status(400).json({ success: false, message: error.message });\n }\n var responseType = determineResponseType(hasFileExtension, headers);\n if (responseType !== SERVE) {\n return handleShowRender(req, res);\n }\n // handle serve request\n // send google analytics\n sendGAServeEvent(headers, ip, originalUrl);\n // parse the claim\n var claimName = void 0;\n try {\n var _lbryUri$parseClaim = lbryUri.parseClaim(params.claim);\n\n claimName = _lbryUri$parseClaim.claimName;\n } catch (error) {\n return res.status(400).json({ success: false, message: error.message });\n }\n // log the request data for debugging\n logRequestData(responseType, claimName, null, null);\n // get the claim Id and then serve the asset\n getClaimIdAndServeAsset(null, null, claimName, null, originalUrl, ip, res);\n};\n\nmodule.exports = serverAssetByClaim;\n\n/***/ }),\n/* 87 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"redux-saga\");\n\n/***/ }),\n/* 88 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"redux-saga/effects\");\n\n/***/ }),\n/* 89 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i[\"return\"]) _i[\"return\"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError(\"Invalid attempt to destructure non-iterable instance\"); } }; }();\n\nvar _require = __webpack_require__(4),\n sendGAServeEvent = _require.sendGAServeEvent;\n\nvar _require2 = __webpack_require__(23),\n determineResponseType = _require2.determineResponseType,\n flipClaimNameAndIdForBackwardsCompatibility = _require2.flipClaimNameAndIdForBackwardsCompatibility,\n logRequestData = _require2.logRequestData,\n getClaimIdAndServeAsset = _require2.getClaimIdAndServeAsset;\n\nvar lbryUri = __webpack_require__(24);\nvar handleShowRender = __webpack_require__(25);\n\nvar SERVE = 'SERVE';\n\n/*\n\n route to serve an asset or the react app via the claim name and an identifier\n\n*/\n\nvar serverAssetByIdentifierAndClaim = function serverAssetByIdentifierAndClaim(req, res) {\n var headers = req.headers,\n ip = req.ip,\n originalUrl = req.originalUrl,\n params = req.params;\n // decide if this is a show request\n\n var hasFileExtension = void 0;\n try {\n var _lbryUri$parseModifie = lbryUri.parseModifier(params.claim);\n\n hasFileExtension = _lbryUri$parseModifie.hasFileExtension;\n } catch (error) {\n return res.status(400).json({ success: false, message: error.message });\n }\n var responseType = determineResponseType(hasFileExtension, headers);\n if (responseType !== SERVE) {\n return handleShowRender(req, res);\n }\n // handle serve request\n // send google analytics\n sendGAServeEvent(headers, ip, originalUrl);\n // parse the claim\n var claimName = void 0;\n try {\n var _lbryUri$parseClaim = lbryUri.parseClaim(params.claim);\n\n claimName = _lbryUri$parseClaim.claimName;\n } catch (error) {\n return res.status(400).json({ success: false, message: error.message });\n }\n // parse the identifier\n var isChannel = void 0,\n channelName = void 0,\n channelClaimId = void 0,\n claimId = void 0;\n try {\n var _lbryUri$parseIdentif = lbryUri.parseIdentifier(params.identifier);\n\n isChannel = _lbryUri$parseIdentif.isChannel;\n channelName = _lbryUri$parseIdentif.channelName;\n channelClaimId = _lbryUri$parseIdentif.channelClaimId;\n claimId = _lbryUri$parseIdentif.claimId;\n } catch (error) {\n return res.status(400).json({ success: false, message: error.message });\n }\n if (!isChannel) {\n var _flipClaimNameAndIdFo = flipClaimNameAndIdForBackwardsCompatibility(claimId, claimName);\n\n var _flipClaimNameAndIdFo2 = _slicedToArray(_flipClaimNameAndIdFo, 2);\n\n claimId = _flipClaimNameAndIdFo2[0];\n claimName = _flipClaimNameAndIdFo2[1];\n }\n // log the request data for debugging\n logRequestData(responseType, claimName, channelName, claimId);\n // get the claim Id and then serve the asset\n getClaimIdAndServeAsset(channelName, channelClaimId, claimName, claimId, originalUrl, ip, res);\n};\n\nmodule.exports = serverAssetByIdentifierAndClaim;\n\n/***/ }),\n/* 90 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar handlePageRequest = __webpack_require__(91);\n\nmodule.exports = function (app) {\n app.get('*', handlePageRequest);\n};\n\n/***/ }),\n/* 91 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar handlePageRender = __webpack_require__(14);\n\nvar sendReactApp = function sendReactApp(req, res) {\n handlePageRender(req, res);\n};\n\nmodule.exports = sendReactApp;\n\n/***/ })\n/******/ ]);\n\n\n// WEBPACK FOOTER //\n// index.js"," \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId]) {\n \t\t\treturn installedModules[moduleId].exports;\n \t\t}\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\ti: moduleId,\n \t\t\tl: false,\n \t\t\texports: {}\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.l = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// define getter function for harmony exports\n \t__webpack_require__.d = function(exports, name, getter) {\n \t\tif(!__webpack_require__.o(exports, name)) {\n \t\t\tObject.defineProperty(exports, name, {\n \t\t\t\tconfigurable: false,\n \t\t\t\tenumerable: true,\n \t\t\t\tget: getter\n \t\t\t});\n \t\t}\n \t};\n\n \t// getDefaultExport function for compatibility with non-harmony modules\n \t__webpack_require__.n = function(module) {\n \t\tvar getter = module && module.__esModule ?\n \t\t\tfunction getDefault() { return module['default']; } :\n \t\t\tfunction getModuleExports() { return module; };\n \t\t__webpack_require__.d(getter, 'a', getter);\n \t\treturn getter;\n \t};\n\n \t// Object.prototype.hasOwnProperty.call\n \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"/\";\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(__webpack_require__.s = 26);\n\n\n\n// WEBPACK FOOTER //\n// webpack/bootstrap 997e9b26fb4c0319bb2a","module.exports = require(\"winston\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"winston\"\n// module id = 0\n// module chunks = 0","const logger = require('winston');\n\nmodule.exports = {\n handleErrorResponse: function (originalUrl, ip, error, res) {\n logger.error(`Error on ${originalUrl}`, module.exports.useObjectPropertiesIfNoKeys(error));\n const [status, message] = module.exports.returnErrorMessageAndStatus(error);\n res\n .status(status)\n .json(module.exports.createErrorResponsePayload(status, message));\n },\n returnErrorMessageAndStatus: function (error) {\n let status, message;\n // check for daemon being turned off\n if (error.code === 'ECONNREFUSED') {\n status = 503;\n message = 'Connection refused. The daemon may not be running.';\n // fallback for everything else\n } else {\n status = 400;\n if (error.message) {\n message = error.message;\n } else {\n message = error;\n };\n };\n return [status, message];\n },\n useObjectPropertiesIfNoKeys: function (err) {\n if (Object.keys(err).length === 0) {\n let newErrorObject = {};\n Object.getOwnPropertyNames(err).forEach((key) => {\n newErrorObject[key] = err[key];\n });\n return newErrorObject;\n }\n return err;\n },\n createErrorResponsePayload (status, message) {\n return {\n status,\n success: false,\n message,\n };\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/errorHandlers.js","const Certificate = require('models/certificate.js');\nconst Channel = require('models/channel.js');\nconst Claim = require('models/claim.js');\nconst File = require('models/file.js');\nconst Request = require('models/request.js');\nconst User = require('models/user.js');\n\nconst Sequelize = require('sequelize');\nconst logger = require('winston');\n\nconst {database, username, password} = require('mysqlConfig.js');\n\n// set sequelize options\nconst sequelize = new Sequelize(database, username, password, {\n host : 'localhost',\n dialect : 'mysql',\n dialectOptions: {decimalNumbers: true},\n logging : false,\n pool : {\n max : 5,\n min : 0,\n idle : 10000,\n acquire: 10000,\n },\n});\n\n// establish mysql connection\nsequelize\n .authenticate()\n .then(() => {\n logger.info('Sequelize has established mysql connection successfully.');\n })\n .catch(err => {\n logger.error('Sequelize was unable to connect to the database:', err);\n });\n\n// manually add each model to the db object (note: make this dynamic)\nconst db = {};\ndb['Certificate'] = sequelize.import('Certificate', Certificate);\ndb['Channel'] = sequelize.import('Channel', Channel);\ndb['Claim'] = sequelize.import('Claim', Claim);\ndb['File'] = sequelize.import('File', File);\ndb['Request'] = sequelize.import('Request', Request);\ndb['User'] = sequelize.import('User', User);\n\n// run model.association for each model in the db object that has an association\nlogger.info('associating db models...');\nObject.keys(db).forEach(modelName => {\n if (db[modelName].associate) {\n logger.info('Associating model:', modelName);\n db[modelName].associate(db);\n }\n});\n\n// add sequelize/Sequelize to db\ndb.sequelize = sequelize;\ndb.Sequelize = Sequelize;\n// add an 'upsert' method to the db object\ndb.upsert = (Model, values, condition, tableName) => {\n return Model\n .findOne({\n where: condition,\n })\n .then(obj => {\n if (obj) { // update\n logger.debug(`updating record in db.${tableName}`);\n return obj.update(values);\n } else { // insert\n logger.debug(`creating record in db.${tableName}`);\n return Model.create(values);\n }\n })\n .catch(function (error) {\n logger.error(`${tableName}.upsert error`, error);\n throw error;\n });\n};\n\nmodule.exports = db;\n\n\n\n// WEBPACK FOOTER //\n// ./server/models/index.js","function SiteConfig () {\n this.analytics = {\n googleId: 'default',\n };\n this.assetDefaults = {\n description: 'An asset published on Spee.ch',\n thumbnail : 'https://spee.ch/assets/img/video_thumb_default.png',\n title : 'Spee.ch',\n };\n this.auth = {\n sessionKey: 'default',\n };\n this.customComponents = {\n components: {},\n containers: {},\n pages : {},\n };\n this.details = {\n description: 'Open-source, decentralized image and video sharing.',\n host : 'default',\n port : 3000,\n title : 'Spee.ch',\n twitter : '@spee_ch',\n };\n this.publishing = {\n additionalClaimAddresses: [],\n disabled : false,\n disabledMessage : 'Please check back soon.',\n primaryClaimAddress : 'default',\n thumbnailChannel : 'default',\n thumbnailChannelId : 'default',\n uploadDirectory : '/home/lbry/Uploads',\n };\n this.routes = {};\n this.update = (config) => {\n if (!config) {\n return console.log('No site config received.');\n }\n const { analytics, assetDefaults, auth, customComponents, details, publishing, routes } = config;\n console.log('Configuring site details...');\n this.analytics = analytics;\n this.assetDefaults = assetDefaults;\n this.auth = auth;\n this.details = details;\n this.publishing = publishing;\n this.customComponents = customComponents;\n this.routes = routes;\n };\n};\n\nmodule.exports = new SiteConfig();\n\n\n\n// WEBPACK FOOTER //\n// ./config/siteConfig.js","const logger = require('winston');\nconst ua = require('universal-analytics');\nconst { analytics : { googleId }, details: { title } } = require('../../config/siteConfig.js');\n\nfunction createServeEventParams (headers, ip, originalUrl) {\n return {\n eventCategory : 'client requests',\n eventAction : 'serve request',\n eventLabel : originalUrl,\n ipOverride : ip,\n userAgentOverride: headers['user-agent'],\n };\n};\n\nfunction createPublishTimingEventParams (category, variable, label, startTime, endTime) {\n const duration = endTime - startTime;\n return {\n userTimingCategory : category,\n userTimingVariableName: variable,\n userTimingTime : duration,\n userTimingLabel : label,\n };\n};\n\nfunction sendGoogleAnalyticsEvent (ip, params) {\n const visitorId = ip.replace(/\\./g, '-');\n const visitor = ua(googleId, visitorId, { strictCidFormat: false, https: true });\n visitor.event(params, (err) => {\n if (err) {\n logger.error('Google Analytics Event Error >>', err);\n }\n });\n};\n\nfunction sendGoogleAnalyticsTiming (visitorId, params) {\n const visitor = ua(googleId, visitorId, { strictCidFormat: false, https: true });\n visitor.timing(params, (err) => {\n if (err) {\n logger.error('Google Analytics Event Error >>', err);\n }\n logger.debug(`Timing event successfully sent to google analytics`);\n });\n};\n\nmodule.exports = {\n sendGAServeEvent (headers, ip, originalUrl) {\n const params = createServeEventParams(headers, ip, originalUrl);\n sendGoogleAnalyticsEvent(ip, params);\n },\n sendGATimingEvent (category, variable, label, startTime, endTime) {\n const params = createPublishTimingEventParams(category, variable, label, startTime, endTime);\n sendGoogleAnalyticsTiming(title, params);\n },\n chooseGaLbrynetPublishLabel ({ channel_name: channelName, channel_id: channelId }) {\n return (channelName || channelId ? 'PUBLISH_IN_CHANNEL_CLAIM' : 'PUBLISH_ANONYMOUS_CLAIM');\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/googleAnalytics.js","const axios = require('axios');\nconst logger = require('winston');\nconst { api: { apiHost, apiPort } } = require('../../config/lbryConfig.js');\nconst lbryApiUri = 'http://' + apiHost + ':' + apiPort;\nconst { chooseGaLbrynetPublishLabel, sendGATimingEvent } = require('./googleAnalytics.js');\n\nconst handleLbrynetResponse = ({ data }, resolve, reject) => {\n logger.debug('lbry api data:', data);\n if (data.result) {\n // check for an error\n if (data.result.error) {\n logger.debug('Lbrynet api error:', data.result.error);\n reject(new Error(data.result.error));\n return;\n };\n resolve(data.result);\n return;\n }\n // fallback in case it just timed out\n reject(JSON.stringify(data));\n};\n\nmodule.exports = {\n publishClaim (publishParams) {\n logger.debug(`lbryApi >> Publishing claim to \"${publishParams.name}\"`);\n const gaStartTime = Date.now();\n return new Promise((resolve, reject) => {\n axios\n .post(lbryApiUri, {\n method: 'publish',\n params: publishParams,\n })\n .then(response => {\n sendGATimingEvent('lbrynet', 'publish', chooseGaLbrynetPublishLabel(publishParams), gaStartTime, Date.now());\n handleLbrynetResponse(response, resolve, reject);\n })\n .catch(error => {\n reject(error);\n });\n });\n },\n getClaim (uri) {\n logger.debug(`lbryApi >> Getting Claim for \"${uri}\"`);\n const gaStartTime = Date.now();\n return new Promise((resolve, reject) => {\n axios\n .post(lbryApiUri, {\n method: 'get',\n params: { uri, timeout: 20 },\n })\n .then(response => {\n sendGATimingEvent('lbrynet', 'getClaim', 'GET', gaStartTime, Date.now());\n handleLbrynetResponse(response, resolve, reject);\n })\n .catch(error => {\n reject(error);\n });\n });\n },\n getClaimList (claimName) {\n logger.debug(`lbryApi >> Getting claim_list for \"${claimName}\"`);\n const gaStartTime = Date.now();\n return new Promise((resolve, reject) => {\n axios\n .post(lbryApiUri, {\n method: 'claim_list',\n params: { name: claimName },\n })\n .then(response => {\n sendGATimingEvent('lbrynet', 'getClaimList', 'CLAIM_LIST', gaStartTime, Date.now());\n handleLbrynetResponse(response, resolve, reject);\n })\n .catch(error => {\n reject(error);\n });\n });\n },\n resolveUri (uri) {\n logger.debug(`lbryApi >> Resolving URI for \"${uri}\"`);\n const gaStartTime = Date.now();\n return new Promise((resolve, reject) => {\n axios\n .post(lbryApiUri, {\n method: 'resolve',\n params: { uri },\n })\n .then(({ data }) => {\n sendGATimingEvent('lbrynet', 'resolveUri', 'RESOLVE', gaStartTime, Date.now());\n if (data.result[uri].error) { // check for errors\n reject(data.result[uri].error);\n } else { // if no errors, resolve\n resolve(data.result[uri]);\n }\n })\n .catch(error => {\n reject(error);\n });\n });\n },\n getDownloadDirectory () {\n logger.debug('lbryApi >> Retrieving the download directory path from lbry daemon...');\n const gaStartTime = Date.now();\n return new Promise((resolve, reject) => {\n axios\n .post(lbryApiUri, {\n method: 'settings_get',\n })\n .then(({ data }) => {\n sendGATimingEvent('lbrynet', 'getDownloadDirectory', 'SETTINGS_GET', gaStartTime, Date.now());\n if (data.result) {\n resolve(data.result.download_directory);\n } else {\n return new Error('Successfully connected to lbry daemon, but unable to retrieve the download directory.');\n }\n })\n .catch(error => {\n logger.error('Lbrynet Error:', error);\n resolve('/home/lbry/Downloads/');\n });\n });\n },\n createChannel (name) {\n logger.debug(`lbryApi >> Creating channel for ${name}...`);\n const gaStartTime = Date.now();\n return new Promise((resolve, reject) => {\n axios\n .post(lbryApiUri, {\n method: 'channel_new',\n params: {\n channel_name: name,\n amount : 0.1,\n },\n })\n .then(response => {\n sendGATimingEvent('lbrynet', 'createChannel', 'CHANNEL_NEW', gaStartTime, Date.now());\n handleLbrynetResponse(response, resolve, reject);\n })\n .catch(error => {\n reject(error);\n });\n });\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/lbryApi.js","const db = require('models');\nconst logger = require('winston');\nconst { returnPaginatedChannelClaims } = require('helpers/channelPagination.js');\n\nconst NO_CHANNEL = 'NO_CHANNEL';\nconst NO_CLAIM = 'NO_CLAIM';\nconst NO_FILE = 'NO_FILE';\n\nmodule.exports = {\n getClaimId (channelName, channelClaimId, name, claimId) {\n if (channelName) {\n return module.exports.getClaimIdByChannel(channelName, channelClaimId, name);\n } else {\n return module.exports.getClaimIdByClaim(name, claimId);\n }\n },\n getClaimIdByClaim (claimName, claimId) {\n logger.debug(`getClaimIdByClaim(${claimName}, ${claimId})`);\n return new Promise((resolve, reject) => {\n db.Claim.getLongClaimId(claimName, claimId)\n .then(longClaimId => {\n if (!longClaimId) {\n resolve(NO_CLAIM);\n }\n resolve(longClaimId);\n })\n .catch(error => {\n reject(error);\n });\n });\n },\n getClaimIdByChannel (channelName, channelClaimId, claimName) {\n logger.debug(`getClaimIdByChannel(${channelName}, ${channelClaimId}, ${claimName})`);\n return new Promise((resolve, reject) => {\n db.Certificate.getLongChannelId(channelName, channelClaimId) // 1. get the long channel id\n .then(longChannelId => {\n if (!longChannelId) {\n return [null, null];\n }\n return Promise.all([longChannelId, db.Claim.getClaimIdByLongChannelId(longChannelId, claimName)]); // 2. get the long claim id\n })\n .then(([longChannelId, longClaimId]) => {\n if (!longChannelId) {\n return resolve(NO_CHANNEL);\n }\n if (!longClaimId) {\n return resolve(NO_CLAIM);\n }\n resolve(longClaimId);\n })\n .catch(error => {\n reject(error);\n });\n });\n },\n getChannelData (channelName, channelClaimId, page) {\n return new Promise((resolve, reject) => {\n // 1. get the long channel Id (make sure channel exists)\n db.Certificate.getLongChannelId(channelName, channelClaimId)\n .then(longChannelClaimId => {\n if (!longChannelClaimId) {\n return [null, null, null];\n }\n // 2. get the short ID and all claims for that channel\n return Promise.all([longChannelClaimId, db.Certificate.getShortChannelIdFromLongChannelId(longChannelClaimId, channelName)]);\n })\n .then(([longChannelClaimId, shortChannelClaimId]) => {\n if (!longChannelClaimId) {\n return resolve(NO_CHANNEL);\n }\n // 3. return all the channel information\n resolve({\n channelName,\n longChannelClaimId,\n shortChannelClaimId,\n });\n })\n .catch(error => {\n reject(error);\n });\n });\n },\n getChannelClaims (channelName, channelClaimId, page) {\n return new Promise((resolve, reject) => {\n // 1. get the long channel Id (make sure channel exists)\n db.Certificate.getLongChannelId(channelName, channelClaimId)\n .then(longChannelClaimId => {\n if (!longChannelClaimId) {\n return [null, null, null];\n }\n // 2. get the short ID and all claims for that channel\n return Promise.all([longChannelClaimId, db.Claim.getAllChannelClaims(longChannelClaimId)]);\n })\n .then(([longChannelClaimId, channelClaimsArray]) => {\n if (!longChannelClaimId) {\n return resolve(NO_CHANNEL);\n }\n // 3. format the data for the view, including pagination\n let paginatedChannelViewData = returnPaginatedChannelClaims(channelName, longChannelClaimId, channelClaimsArray, page);\n // 4. return all the channel information and contents\n resolve(paginatedChannelViewData);\n })\n .catch(error => {\n reject(error);\n });\n });\n },\n getLocalFileRecord (claimId, name) {\n return db.File.findOne({where: {claimId, name}})\n .then(file => {\n if (!file) {\n return NO_FILE;\n }\n return file.dataValues;\n });\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/controllers/serveController.js","const passport = require('passport');\nconst localLoginStrategy = require('./local-login.js');\nconst localSignupStrategy = require('./local-signup.js');\nconst { serializeSpeechUser, deserializeSpeechUser } = require('helpers/authHelpers.js');\n\npassport.deserializeUser(deserializeSpeechUser);\npassport.serializeUser(serializeSpeechUser);\npassport.use('local-login', localLoginStrategy);\npassport.use('local-signup', localSignupStrategy);\n\nmodule.exports = passport;\n\n\n\n// WEBPACK FOOTER //\n// ./server/speechPassport/index.js","const logger = require('winston');\nconst db = require('models');\nconst lbryApi = require('helpers/lbryApi.js');\nconst publishHelpers = require('helpers/publishHelpers.js');\nconst { publishing: { primaryClaimAddress, additionalClaimAddresses } } = require('siteConfig.js');\nconst Sequelize = require('sequelize');\nconst Op = Sequelize.Op;\n\nmodule.exports = {\n publish (publishParams, fileName, fileType) {\n return new Promise((resolve, reject) => {\n let publishResults, certificateId, channelName;\n // publish the file\n return lbryApi.publishClaim(publishParams)\n .then(tx => {\n logger.info(`Successfully published ${publishParams.name} ${fileName}`, tx);\n publishResults = tx;\n // get the channel information\n if (publishParams.channel_name) {\n logger.debug(`this claim was published in channel: ${publishParams.channel_name}`);\n return db.Channel.findOne({\n where: {\n channelName: publishParams.channel_name,\n },\n });\n } else {\n logger.debug('this claim was not published in a channel');\n return null;\n }\n })\n .then(channel => {\n // set channel information\n certificateId = null;\n channelName = null;\n if (channel) {\n certificateId = channel.channelClaimId;\n channelName = channel.channelName;\n }\n logger.debug(`certificateId: ${certificateId}`);\n })\n .then(() => {\n // create the File record\n const fileRecord = {\n name : publishParams.name,\n claimId : publishResults.claim_id,\n title : publishParams.metadata.title,\n description: publishParams.metadata.description,\n address : publishParams.claim_address,\n outpoint : `${publishResults.txid}:${publishResults.nout}`,\n height : 0,\n fileName,\n filePath : publishParams.file_path,\n fileType,\n nsfw : publishParams.metadata.nsfw,\n };\n // create the Claim record\n const claimRecord = {\n name : publishParams.name,\n claimId : publishResults.claim_id,\n title : publishParams.metadata.title,\n description: publishParams.metadata.description,\n address : publishParams.claim_address,\n thumbnail : publishParams.metadata.thumbnail,\n outpoint : `${publishResults.txid}:${publishResults.nout}`,\n height : 0,\n contentType: fileType,\n nsfw : publishParams.metadata.nsfw,\n amount : publishParams.bid,\n certificateId,\n channelName,\n };\n // upsert criteria\n const upsertCriteria = {\n name : publishParams.name,\n claimId: publishResults.claim_id,\n };\n // upsert the records\n return Promise.all([db.upsert(db.File, fileRecord, upsertCriteria, 'File'), db.upsert(db.Claim, claimRecord, upsertCriteria, 'Claim')]);\n })\n .then(([file, claim]) => {\n logger.debug('File and Claim records successfully created');\n return Promise.all([file.setClaim(claim), claim.setFile(file)]);\n })\n .then(() => {\n logger.debug('File and Claim records successfully associated');\n resolve(publishResults); // resolve the promise with the result from lbryApi.publishClaim;\n })\n .catch(error => {\n logger.error('PUBLISH ERROR', error);\n publishHelpers.deleteTemporaryFile(publishParams.file_path); // delete the local file\n reject(error);\n });\n });\n },\n claimNameIsAvailable (name) {\n const claimAddresses = additionalClaimAddresses || [];\n claimAddresses.push(primaryClaimAddress);\n // find any records where the name is used\n return db.Claim\n .findAll({\n attributes: ['address'],\n where : {\n name,\n address: {\n [Op.or]: claimAddresses,\n },\n },\n })\n .then(result => {\n if (result.length >= 1) {\n throw new Error('That claim is already in use');\n };\n return name;\n })\n .catch(error => {\n throw error;\n });\n },\n checkChannelAvailability (name) {\n return db.Channel\n .findAll({\n where: { channelName: name },\n })\n .then(result => {\n if (result.length >= 1) {\n throw new Error('That channel has already been claimed');\n }\n return name;\n })\n .catch(error => {\n throw error;\n });\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/controllers/publishController.js","const logger = require('winston');\nconst fs = require('fs');\n\nconst { details, publishing } = require('../../config/siteConfig.js');\n\nmodule.exports = {\n parsePublishApiRequestBody ({name, nsfw, license, title, description, thumbnail}) {\n // validate name\n if (!name) {\n throw new Error('no name field found in request');\n }\n const invalidNameCharacters = /[^A-Za-z0-9,-]/.exec(name);\n if (invalidNameCharacters) {\n throw new Error('The claim name you provided is not allowed. Only the following characters are allowed: A-Z, a-z, 0-9, and \"-\"');\n }\n // optional parameters\n nsfw = (nsfw === 'true');\n license = license || null;\n title = title || null;\n description = description || null;\n thumbnail = thumbnail || null;\n // return results\n return {\n name,\n nsfw,\n license,\n title,\n description,\n thumbnail,\n };\n },\n parsePublishApiRequestFiles ({file, thumbnail}) {\n // make sure a file was provided\n if (!file) {\n throw new Error('no file with key of [file] found in request');\n }\n if (!file.path) {\n throw new Error('no file path found');\n }\n if (!file.type) {\n throw new Error('no file type found');\n }\n if (!file.size) {\n throw new Error('no file type found');\n }\n // validate the file name\n if (/'/.test(file.name)) {\n throw new Error('apostrophes are not allowed in the file name');\n }\n // validate the file\n module.exports.validateFileTypeAndSize(file);\n // return results\n return {\n fileName : file.name,\n filePath : file.path,\n fileType : file.type,\n thumbnailFileName: (thumbnail ? thumbnail.name : null),\n thumbnailFilePath: (thumbnail ? thumbnail.path : null),\n thumbnailFileType: (thumbnail ? thumbnail.type : null),\n };\n },\n validateFileTypeAndSize (file) {\n // check file type and size\n switch (file.type) {\n case 'image/jpeg':\n case 'image/jpg':\n case 'image/png':\n if (file.size > 10000000) {\n logger.debug('publish > file validation > .jpeg/.jpg/.png was too big');\n throw new Error('Sorry, images are limited to 10 megabytes.');\n }\n break;\n case 'image/gif':\n if (file.size > 50000000) {\n logger.debug('publish > file validation > .gif was too big');\n throw new Error('Sorry, .gifs are limited to 50 megabytes.');\n }\n break;\n case 'video/mp4':\n if (file.size > 50000000) {\n logger.debug('publish > file validation > .mp4 was too big');\n throw new Error('Sorry, videos are limited to 50 megabytes.');\n }\n break;\n default:\n logger.debug('publish > file validation > unrecognized file type');\n throw new Error('The ' + file.type + ' content type is not supported. Only, .jpeg, .png, .gif, and .mp4 files are currently supported.');\n }\n return file;\n },\n createBasicPublishParams (filePath, name, title, description, license, nsfw, thumbnail) {\n logger.debug(`Creating Publish Parameters`);\n // provide defaults for title\n if (title === null || title.trim() === '') {\n title = name;\n }\n // provide default for description\n if (description === null || description.trim() === '') {\n description = '';\n }\n // provide default for license\n if (license === null || license.trim() === '') {\n license = ' '; // default to empty string\n }\n // create the publish params\n const publishParams = {\n name,\n file_path: filePath,\n bid : 0.01,\n metadata : {\n description,\n title,\n author : details.title,\n language: 'en',\n license,\n nsfw,\n },\n claim_address: publishing.primaryClaimAddress,\n };\n // add thumbnail to channel if video\n if (thumbnail) {\n publishParams['metadata']['thumbnail'] = thumbnail;\n }\n return publishParams;\n },\n createThumbnailPublishParams (thumbnailFilePath, claimName, license, nsfw) {\n if (!thumbnailFilePath) {\n return;\n }\n logger.debug(`Creating Thumbnail Publish Parameters`);\n // create the publish params\n return {\n name : `${claimName}-thumb`,\n file_path: thumbnailFilePath,\n bid : 0.01,\n metadata : {\n title : `${claimName} thumbnail`,\n description: `a thumbnail for ${claimName}`,\n author : details.title,\n language : 'en',\n license,\n nsfw,\n },\n claim_address: publishing.primaryClaimAddress,\n channel_name : publishing.thumbnailChannel,\n channel_id : publishing.thumbnailChannelId,\n };\n },\n deleteTemporaryFile (filePath) {\n fs.unlink(filePath, err => {\n if (err) {\n logger.error(`error deleting temporary file ${filePath}`);\n throw err;\n }\n logger.debug(`successfully deleted ${filePath}`);\n });\n },\n addGetResultsToFileData (fileInfo, getResult) {\n fileInfo.fileName = getResult.file_name;\n fileInfo.filePath = getResult.download_path;\n return fileInfo;\n },\n createFileData ({ name, claimId, outpoint, height, address, nsfw, contentType }) {\n return {\n name,\n claimId,\n outpoint,\n height,\n address,\n fileName: '',\n filePath: '',\n fileType: contentType,\n nsfw,\n };\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/publishHelpers.js","const logger = require('winston');\n\nfunction mysql () {\n this.database = 'default';\n this.username = 'default';\n this.password = 'default';\n this.update = (config) => {\n if (!config) {\n return logger.warn('No MySQL config received.');\n }\n // configure credentials\n logger.info('configuring mysql...');\n const { database, username, password } = config;\n this.database = database;\n this.username = username;\n this.password = password;\n };\n};\n\nmodule.exports = new mysql();\n\n\n\n// WEBPACK FOOTER //\n// ./config/mysqlConfig.js","module.exports = require(\"passport-local\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"passport-local\"\n// module id = 11\n// module chunks = 0","module.exports = {\n returnShortId: function (claimsArray, longId) {\n let claimIndex;\n let shortId = longId.substring(0, 1); // default short id is the first letter\n let shortIdLength = 0;\n // find the index of this claim id\n claimIndex = claimsArray.findIndex(element => {\n return element.claimId === longId;\n });\n if (claimIndex < 0) {\n throw new Error('claim id not found in claims list');\n }\n // get an array of all claims with lower height\n let possibleMatches = claimsArray.slice(0, claimIndex);\n // remove certificates with the same prefixes until none are left.\n while (possibleMatches.length > 0) {\n shortIdLength += 1;\n shortId = longId.substring(0, shortIdLength);\n possibleMatches = possibleMatches.filter(element => {\n return (element.claimId && (element.claimId.substring(0, shortIdLength) === shortId));\n });\n }\n return shortId;\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/sequelizeHelpers.js","module.exports = require(\"sequelize\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"sequelize\"\n// module id = 13\n// module chunks = 0","import React from 'react';\nimport { renderToString } from 'react-dom/server';\nimport { createStore } from 'redux';\nimport { Provider } from 'react-redux';\nimport { StaticRouter } from 'react-router-dom';\nimport { Reducers, GAListener, App } from 'spee.ch-components';\nimport renderFullPage from './renderFullPage.js';\nimport Helmet from 'react-helmet';\n\nmodule.exports = (req, res) => {\n let context = {};\n\n // create a new Redux store instance\n const store = createStore(Reducers);\n\n // render component to a string\n const html = renderToString(\n \n \n \n \n \n \n \n );\n\n // get head tags from helmet\n const helmet = Helmet.renderStatic();\n\n // check for a redirect\n if (context.url) {\n // Somewhere a `` was rendered\n return res.redirect(301, context.url);\n } else {\n // we're good, send the response\n }\n\n // get the initial state from our Redux store\n const preloadedState = store.getState();\n\n // send the rendered page back to the client\n res.send(renderFullPage(helmet, html, preloadedState));\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/handlePageRender.jsx","module.exports = require(\"react\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"react\"\n// module id = 15\n// module chunks = 0","module.exports = require(\"react-dom/server\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"react-dom/server\"\n// module id = 16\n// module chunks = 0","module.exports = require(\"redux\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"redux\"\n// module id = 17\n// module chunks = 0","module.exports = require(\"react-redux\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"react-redux\"\n// module id = 18\n// module chunks = 0","module.exports = require(\"react-router-dom\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"react-router-dom\"\n// module id = 19\n// module chunks = 0","module.exports = require(\"spee.ch-components\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"spee.ch-components\"\n// module id = 20\n// module chunks = 0","module.exports = (helmet, html, preloadedState) => {\n // take the html and preloadedState and return the full page\n return `\n \n \n \n \n \n \n \n ${helmet.title.toString()}\n ${helmet.meta.toString()}\n ${helmet.link.toString()}\n \n \n \n \n \n \n \n \n
\n
${html}
\n
\n \n \n \n \n `;\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/renderFullPage.js","module.exports = require(\"react-helmet\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"react-helmet\"\n// module id = 22\n// module chunks = 0","const logger = require('winston');\nconst { getClaimId, getLocalFileRecord } = require('../controllers/serveController.js');\nconst { handleErrorResponse } = require('./errorHandlers.js');\n\nconst SERVE = 'SERVE';\nconst SHOW = 'SHOW';\nconst NO_FILE = 'NO_FILE';\nconst NO_CHANNEL = 'NO_CHANNEL';\nconst NO_CLAIM = 'NO_CLAIM';\n\nfunction clientAcceptsHtml ({accept}) {\n return accept && accept.match(/text\\/html/);\n};\n\nfunction requestIsFromBrowser (headers) {\n return headers['user-agent'] && headers['user-agent'].match(/Mozilla/);\n};\n\nfunction clientWantsAsset ({accept, range}) {\n const imageIsWanted = accept && accept.match(/image\\/.*/) && !accept.match(/text\\/html/) && !accept.match(/text\\/\\*/);\n const videoIsWanted = accept && range;\n return imageIsWanted || videoIsWanted;\n};\n\nfunction isValidClaimId (claimId) {\n return ((claimId.length === 40) && !/[^A-Za-z0-9]/g.test(claimId));\n};\n\nfunction isValidShortId (claimId) {\n return claimId.length === 1; // it should really evaluate the short url itself\n};\n\nfunction isValidShortIdOrClaimId (input) {\n return (isValidClaimId(input) || isValidShortId(input));\n};\n\nfunction serveAssetToClient (claimId, name, res) {\n return getLocalFileRecord(claimId, name)\n .then(fileRecord => {\n // check that a local record was found\n if (fileRecord === NO_FILE) {\n return res.status(307).redirect(`/api/claim/get/${name}/${claimId}`);\n }\n // serve the file\n const {filePath, fileType} = fileRecord;\n logger.verbose(`serving file: ${filePath}`);\n const sendFileOptions = {\n headers: {\n 'X-Content-Type-Options': 'nosniff',\n 'Content-Type' : fileType || 'image/jpeg',\n },\n };\n res.status(200).sendFile(filePath, sendFileOptions);\n })\n .catch(error => {\n throw error;\n });\n};\n\nmodule.exports = {\n getClaimIdAndServeAsset (channelName, channelClaimId, claimName, claimId, originalUrl, ip, res) {\n // get the claim Id and then serve the asset\n getClaimId(channelName, channelClaimId, claimName, claimId)\n .then(fullClaimId => {\n if (fullClaimId === NO_CLAIM) {\n return res.status(404).json({success: false, message: 'no claim id could be found'});\n } else if (fullClaimId === NO_CHANNEL) {\n return res.status(404).json({success: false, message: 'no channel id could be found'});\n }\n serveAssetToClient(fullClaimId, claimName, res);\n // postToStats(responseType, originalUrl, ip, claimName, fullClaimId, 'success');\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n // postToStats(responseType, originalUrl, ip, claimName, fullClaimId, 'fail');\n });\n },\n determineResponseType (hasFileExtension, headers) {\n let responseType;\n if (hasFileExtension) {\n responseType = SERVE; // assume a serve request if file extension is present\n if (clientAcceptsHtml(headers)) { // if the request comes from a browser, change it to a show request\n responseType = SHOW;\n }\n } else {\n responseType = SHOW;\n if (clientWantsAsset(headers) && requestIsFromBrowser(headers)) { // this is in case someone embeds a show url\n logger.debug('Show request came from browser but wants an image/video. Changing response to serve...');\n responseType = SERVE;\n }\n }\n return responseType;\n },\n flipClaimNameAndIdForBackwardsCompatibility (identifier, name) {\n // this is a patch for backwards compatability with '/name/claim_id' url format\n if (isValidShortIdOrClaimId(name) && !isValidShortIdOrClaimId(identifier)) {\n const tempName = name;\n name = identifier;\n identifier = tempName;\n }\n return [identifier, name];\n },\n logRequestData (responseType, claimName, channelName, claimId) {\n logger.debug('responseType ===', responseType);\n logger.debug('claim name === ', claimName);\n logger.debug('channel name ===', channelName);\n logger.debug('claim id ===', claimId);\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/serveHelpers.js","const logger = require('winston');\n\nmodule.exports = {\n REGEXP_INVALID_CLAIM : /[^A-Za-z0-9-]/g,\n REGEXP_INVALID_CHANNEL: /[^A-Za-z0-9-@]/g,\n REGEXP_ADDRESS : /^b(?=[^0OIl]{32,33})[0-9A-Za-z]{32,33}$/,\n CHANNEL_CHAR : '@',\n parseIdentifier : function (identifier) {\n logger.debug('parsing identifier:', identifier);\n const componentsRegex = new RegExp(\n '([^:$#/]*)' + // value (stops at the first separator or end)\n '([:$#]?)([^/]*)' // modifier separator, modifier (stops at the first path separator or end)\n );\n const [proto, value, modifierSeperator, modifier] = componentsRegex\n .exec(identifier)\n .map(match => match || null);\n logger.debug(`${proto}, ${value}, ${modifierSeperator}, ${modifier}`);\n\n // Validate and process name\n if (!value) {\n throw new Error(`Check your url. No channel name provided before \"${modifierSeperator}\"`);\n }\n const isChannel = value.startsWith(module.exports.CHANNEL_CHAR);\n const channelName = isChannel ? value : null;\n let claimId;\n if (isChannel) {\n if (!channelName) {\n throw new Error('No channel name after @.');\n }\n const nameBadChars = (channelName).match(module.exports.REGEXP_INVALID_CHANNEL);\n if (nameBadChars) {\n throw new Error(`Invalid characters in channel name: ${nameBadChars.join(', ')}.`);\n }\n } else {\n claimId = value;\n }\n\n // Validate and process modifier\n let channelClaimId;\n if (modifierSeperator) {\n if (!modifier) {\n throw new Error(`No modifier provided after separator \"${modifierSeperator}\"`);\n }\n\n if (modifierSeperator === ':') {\n channelClaimId = modifier;\n } else {\n throw new Error(`The \"${modifierSeperator}\" modifier is not currently supported`);\n }\n }\n return {\n isChannel,\n channelName,\n channelClaimId,\n claimId,\n };\n },\n parseClaim: function (claim) {\n logger.debug('parsing name:', claim);\n const componentsRegex = new RegExp(\n '([^:$#/.]*)' + // name (stops at the first modifier)\n '([:$#.]?)([^/]*)' // modifier separator, modifier (stops at the first path separator or end)\n );\n const [proto, claimName, modifierSeperator, modifier] = componentsRegex\n .exec(claim)\n .map(match => match || null);\n logger.debug(`${proto}, ${claimName}, ${modifierSeperator}, ${modifier}`);\n\n // Validate and process name\n if (!claimName) {\n throw new Error('No claim name provided before .');\n }\n const nameBadChars = (claimName).match(module.exports.REGEXP_INVALID_CLAIM);\n if (nameBadChars) {\n throw new Error(`Invalid characters in claim name: ${nameBadChars.join(', ')}.`);\n }\n // Validate and process modifier\n if (modifierSeperator) {\n if (!modifier) {\n throw new Error(`No file extension provided after separator ${modifierSeperator}.`);\n }\n if (modifierSeperator !== '.') {\n throw new Error(`The ${modifierSeperator} modifier is not supported in the claim name`);\n }\n }\n // return results\n return {\n claimName,\n };\n },\n parseModifier: function (claim) {\n logger.debug('parsing modifier:', claim);\n const componentsRegex = new RegExp(\n '([^:$#/.]*)' + // name (stops at the first modifier)\n '([:$#.]?)([^/]*)' // modifier separator, modifier (stops at the first path separator or end)\n );\n const [proto, claimName, modifierSeperator, modifier] = componentsRegex\n .exec(claim)\n .map(match => match || null);\n logger.debug(`${proto}, ${claimName}, ${modifierSeperator}, ${modifier}`);\n // Validate and process modifier\n let hasFileExtension = false;\n if (modifierSeperator) {\n hasFileExtension = true;\n }\n return {\n hasFileExtension,\n };\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/lbryUri.js","import React from 'react';\nimport { renderToString } from 'react-dom/server';\nimport { createStore, applyMiddleware } from 'redux';\nimport { Provider } from 'react-redux';\nimport { StaticRouter } from 'react-router-dom';\nimport renderFullPage from './renderFullPage';\nimport createSagaMiddleware from 'redux-saga';\nimport { call } from 'redux-saga/effects';\nimport { Reducers, GAListener, App, Sagas, Actions } from 'spee.ch-components';\n\nimport Helmet from 'react-helmet';\n\nconst returnSagaWithParams = (saga, params) => {\n return function * () {\n yield call(saga, params);\n };\n};\n\nmodule.exports = (req, res) => {\n let context = {};\n\n // create and apply middleware\n const sagaMiddleware = createSagaMiddleware();\n const middleware = applyMiddleware(sagaMiddleware);\n\n // create a new Redux store instance\n const store = createStore(Reducers, middleware);\n\n // create saga\n const action = Actions.onHandleShowPageUri(req.params);\n const saga = returnSagaWithParams(Sagas.handleShowPageUri, action);\n\n // run the saga middleware\n sagaMiddleware\n .run(saga)\n .done\n .then(() => {\n // render component to a string\n const html = renderToString(\n \n \n \n \n \n \n \n );\n\n // get head tags from helmet\n const helmet = Helmet.renderStatic();\n\n // check for a redirect\n if (context.url) {\n return res.redirect(301, context.url);\n }\n\n // get the initial state from our Redux store\n const preloadedState = store.getState();\n\n // send the rendered page back to the client\n res.send(renderFullPage(helmet, html, preloadedState));\n });\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/handleShowRender.jsx","module.exports = require(\"babel-polyfill\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"babel-polyfill\"\n// module id = 27\n// module chunks = 0","module.exports = require(\"whatwg-fetch\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"whatwg-fetch\"\n// module id = 28\n// module chunks = 0","const Server = require('./server');\n\nconst exports = {\n Server,\n};\n\nmodule.exports = exports;\n\n\n\n// WEBPACK FOOTER //\n// ./speech.js","// app dependencies\nconst express = require('express');\nconst bodyParser = require('body-parser');\nconst expressHandlebars = require('express-handlebars');\nconst Handlebars = require('handlebars');\nconst helmet = require('helmet');\nconst cookieSession = require('cookie-session');\nconst http = require('http');\nconst logger = require('winston');\nconst requestLogger = require('middleware/requestLogger.js');\nconst Path = require('path');\nconst loggerConfig = require('loggerConfig.js');\nconst mysqlConfig = require('mysqlConfig.js');\nconst siteConfig = require('siteConfig.js');\nconst slackConfig = require('slackConfig.js');\n\nfunction Server () {\n this.configureLogger = (userConfig) => {\n loggerConfig.update(userConfig);\n };\n this.configureMysql = (userConfig) => {\n mysqlConfig.update(userConfig);\n };\n this.configureSiteDetails = (userConfig) => {\n siteConfig.update(userConfig);\n };\n this.configureSlack = (userConfig) => {\n slackConfig.update(userConfig);\n };\n this.configureClientBundle = () => {\n logger.debug('configure the client here by passing in the bundle and configuring it, or better yet: taking in the components to use dynamically from here.');\n };\n this.configureModels = () => {\n logger.debug('here is where you could add/overwrite the default models')\n };\n this.configureRoutes = () => {\n logger.debug('here is where you could add/overwrite the default routes')\n };\n this.createApp = () => {\n // create an Express application\n const app = express();\n\n // trust the proxy to get ip address for us\n app.enable('trust proxy');\n\n /* add middleware */\n // set HTTP headers to protect against well-known web vulnerabilties\n app.use(helmet());\n // 'express.static' to serve static files from public directory\n if (siteConfig.routes.publicFolder) {\n // take in a different public folder, so it can serve it's own bundle if needed\n const publicFolder = Path.resolve(process.cwd(), siteConfig.routes.publicFolder);\n app.use('/static', express.static(publicFolder));\n logger.info('serving static files from custom path:', publicFolder);\n } else {\n const publicPath = Path.resolve(__dirname, 'public');\n app.use('/static', express.static(publicPath));\n logger.info('serving static files from default path:', publicPath);\n };\n // 'body parser' for parsing application/json\n app.use(bodyParser.json());\n // 'body parser' for parsing application/x-www-form-urlencoded\n app.use(bodyParser.urlencoded({ extended: true }));\n\n // add custom middleware (note: build out to accept dynamically use what is in server/middleware/\n app.use(requestLogger);\n\n // configure passport\n const speechPassport = require('speechPassport');\n // initialize passport\n const sessionKey = siteConfig.auth.sessionKey;\n app.use(cookieSession({\n name : 'session',\n keys : [sessionKey],\n maxAge: 24 * 60 * 60 * 1000, // i.e. 24 hours\n }));\n app.use(speechPassport.initialize());\n app.use(speechPassport.session());\n\n // configure handlebars & register it with express app\n const hbs = expressHandlebars.create({\n defaultLayout: 'embed',\n handlebars : Handlebars,\n });\n app.engine('handlebars', hbs.engine);\n app.set('view engine', 'handlebars');\n\n // set the routes on the app\n require('./routes/auth/')(app);\n require('./routes/api/')(app);\n require('./routes/pages/')(app);\n require('./routes/assets/')(app);\n require('./routes/fallback/')(app);\n\n this.app = app;\n };\n this.initialize = () => {\n this.createApp();\n this.server = http.Server(this.app);\n };\n this.start = () => {\n const db = require('models');\n const PORT = siteConfig.details.port;\n // sync sequelize\n db.sequelize.sync()\n // start the server\n .then(() => {\n this.server.listen(PORT, () => {\n logger.info(`Server is listening on PORT ${PORT}`);\n });\n })\n .catch((error) => {\n logger.error(`Startup Error:`, error);\n });\n };\n};\n\nmodule.exports = Server;\n\n\n\n// WEBPACK FOOTER //\n// ./server/index.js","module.exports = require(\"express\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"express\"\n// module id = 31\n// module chunks = 0","module.exports = require(\"body-parser\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"body-parser\"\n// module id = 32\n// module chunks = 0","module.exports = require(\"express-handlebars\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"express-handlebars\"\n// module id = 33\n// module chunks = 0","module.exports = require(\"handlebars\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"handlebars\"\n// module id = 34\n// module chunks = 0","module.exports = require(\"helmet\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"helmet\"\n// module id = 35\n// module chunks = 0","module.exports = require(\"cookie-session\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"cookie-session\"\n// module id = 36\n// module chunks = 0","module.exports = require(\"http\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"http\"\n// module id = 37\n// module chunks = 0","const logger = require('winston');\n\nconst requestLogger = (req, res, next) => { // custom logging middleware to log all incoming http requests\n logger.verbose(`Request on ${req.originalUrl} from ${req.ip}`);\n next();\n};\n\nmodule.exports = requestLogger;\n\n\n\n// WEBPACK FOOTER //\n// ./server/middleware/requestLogger.js","module.exports = require(\"path\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"path\"\n// module id = 39\n// module chunks = 0","const logger = require('winston');\n\nfunction LoggerConfig () {\n this.logLevel = 'debug';\n this.update = (config) => {\n if (!config) {\n return logger.warn('No logger config received.');\n }\n logger.info('configuring winston logger...');\n // update values with local config params\n const {logLevel} = config;\n this.logLevel = logLevel;\n // configure the winston logger\n logger.configure({\n transports: [\n new (logger.transports.Console)({\n level : this.logLevel,\n timestamp : false,\n colorize : true,\n prettyPrint : true,\n handleExceptions : true,\n humanReadableUnhandledException: true,\n }),\n ],\n });\n // test all the log levels\n logger.info('testing winston log levels...');\n logger.error('Level 0');\n logger.warn('Level 1');\n logger.info('Level 2');\n logger.verbose('Level 3');\n logger.debug('Level 4');\n logger.silly('Level 5');\n };\n};\n\nmodule.exports = new LoggerConfig();\n\n\n\n// WEBPACK FOOTER //\n// ./config/loggerConfig.js","const winstonSlackWebHook = require('winston-slack-webhook').SlackWebHook;\nconst winston = require('winston');\n\nfunction SlackConfig () {\n this.slackWebHook = 'default';\n this.slackErrorChannel = 'default';\n this.slackInfoChannel = 'default';\n this.update = (config) => {\n if (!config) {\n return winston.warn('No slack config received');\n }\n // update variables\n winston.info('configuring slack logger...');\n const {slackWebHook, slackErrorChannel, slackInfoChannel} = config;\n this.slackWebHook = slackWebHook;\n this.slackErrorChannel = slackErrorChannel;\n this.slackInfoChannel = slackInfoChannel;\n // update slack webhook settings\n if (this.slackWebHook) {\n // add a transport for errors to slack\n if (this.slackErrorChannel) {\n winston.add(winstonSlackWebHook, {\n name : 'slack-errors-transport',\n level : 'warn',\n webhookUrl: this.slackWebHook,\n channel : this.slackErrorChannel,\n username : 'spee.ch',\n iconEmoji : ':face_with_head_bandage:',\n });\n };\n if (slackInfoChannel) {\n winston.add(winstonSlackWebHook, {\n name : 'slack-info-transport',\n level : 'info',\n webhookUrl: this.slackWebHook,\n channel : this.slackInfoChannel,\n username : 'spee.ch',\n iconEmoji : ':nerd_face:',\n });\n };\n // send test messages\n winston.info('testing slack logger...');\n winston.error('Slack \"error\" logging is online.');\n winston.info('Slack \"info\" logging is online.');\n } else {\n winston.warn('Slack logging is not enabled because no slackWebHook config var provided.');\n }\n };\n};\n\nmodule.exports = new SlackConfig();\n\n\n\n// WEBPACK FOOTER //\n// ./config/slackConfig.js","module.exports = require(\"winston-slack-webhook\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"winston-slack-webhook\"\n// module id = 42\n// module chunks = 0","module.exports = require(\"passport\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"passport\"\n// module id = 43\n// module chunks = 0","const PassportLocalStrategy = require('passport-local').Strategy;\nconst logger = require('winston');\nconst db = require('models');\n\nconst returnUserAndChannelInfo = (userInstance) => {\n return new Promise((resolve, reject) => {\n let userInfo = {};\n userInfo['id'] = userInstance.id;\n userInfo['userName'] = userInstance.userName;\n userInstance\n .getChannel()\n .then(({channelName, channelClaimId}) => {\n userInfo['channelName'] = channelName;\n userInfo['channelClaimId'] = channelClaimId;\n return db.Certificate.getShortChannelIdFromLongChannelId(channelClaimId, channelName);\n })\n .then(shortChannelId => {\n userInfo['shortChannelId'] = shortChannelId;\n resolve(userInfo);\n })\n .catch(error => {\n reject(error);\n });\n });\n};\n\nmodule.exports = new PassportLocalStrategy(\n {\n usernameField: 'username',\n passwordField: 'password',\n },\n (username, password, done) => {\n return db.User\n .findOne({\n where: {userName: username},\n })\n .then(user => {\n if (!user) {\n logger.debug('no user found');\n return done(null, false, {message: 'Incorrect username or password'});\n }\n return user.comparePassword(password)\n .then(isMatch => {\n if (!isMatch) {\n logger.debug('incorrect password');\n return done(null, false, {message: 'Incorrect username or password'});\n }\n logger.debug('Password was a match, returning User');\n return returnUserAndChannelInfo(user)\n .then(userInfo => {\n return done(null, userInfo);\n })\n .catch(error => {\n return error;\n });\n })\n .catch(error => {\n return error;\n });\n })\n .catch(error => {\n return done(error);\n });\n },\n);\n\n\n\n// WEBPACK FOOTER //\n// ./server/speechPassport/local-login.js","const logger = require('winston');\nconst { returnShortId } = require('../helpers/sequelizeHelpers.js');\n\nmodule.exports = (sequelize, { STRING, BOOLEAN, INTEGER, TEXT, DECIMAL }) => {\n const Certificate = sequelize.define(\n 'Certificate',\n {\n address: {\n type : STRING,\n default: null,\n },\n amount: {\n type : DECIMAL(19, 8),\n default: null,\n },\n claimId: {\n type : STRING,\n default: null,\n },\n claimSequence: {\n type : INTEGER,\n default: null,\n },\n decodedClaim: {\n type : BOOLEAN,\n default: null,\n },\n depth: {\n type : INTEGER,\n default: null,\n },\n effectiveAmount: {\n type : DECIMAL(19, 8),\n default: null,\n },\n hasSignature: {\n type : BOOLEAN,\n default: null,\n },\n height: {\n type : INTEGER,\n default: null,\n },\n hex: {\n type : TEXT('long'),\n default: null,\n },\n name: {\n type : STRING,\n default: null,\n },\n nout: {\n type : INTEGER,\n default: null,\n },\n txid: {\n type : STRING,\n default: null,\n },\n validAtHeight: {\n type : INTEGER,\n default: null,\n },\n outpoint: {\n type : STRING,\n default: null,\n },\n valueVersion: {\n type : STRING,\n default: null,\n },\n claimType: {\n type : STRING,\n default: null,\n },\n certificateVersion: {\n type : STRING,\n default: null,\n },\n keyType: {\n type : STRING,\n default: null,\n },\n publicKey: {\n type : TEXT('long'),\n default: null,\n },\n },\n {\n freezeTableName: true,\n }\n );\n\n Certificate.associate = db => {\n Certificate.belongsTo(db.Channel, {\n foreignKey: {\n allowNull: true,\n },\n });\n };\n\n Certificate.getShortChannelIdFromLongChannelId = function (longChannelId, channelName) {\n logger.debug(`getShortChannelIdFromLongChannelId ${channelName}:${longChannelId}`);\n return new Promise((resolve, reject) => {\n this\n .findAll({\n where: {name: channelName},\n order: [['height', 'ASC']],\n })\n .then(result => {\n switch (result.length) {\n case 0:\n throw new Error('No channel(s) found with that channel name');\n default:\n return resolve(returnShortId(result, longChannelId));\n }\n })\n .catch(error => {\n reject(error);\n });\n });\n };\n\n Certificate.getLongChannelIdFromShortChannelId = function (channelName, channelClaimId) {\n logger.debug(`getLongChannelIdFromShortChannelId(${channelName}, ${channelClaimId})`);\n return new Promise((resolve, reject) => {\n this\n .findAll({\n where: {\n name : channelName,\n claimId: {\n $like: `${channelClaimId}%`,\n },\n },\n order: [['height', 'ASC']],\n })\n .then(result => {\n switch (result.length) {\n case 0:\n return resolve(null);\n default: // note results must be sorted\n return resolve(result[0].claimId);\n }\n })\n .catch(error => {\n reject(error);\n });\n });\n };\n\n Certificate.getLongChannelIdFromChannelName = function (channelName) {\n logger.debug(`getLongChannelIdFromChannelName(${channelName})`);\n return new Promise((resolve, reject) => {\n this\n .findAll({\n where: { name: channelName },\n order: [['effectiveAmount', 'DESC'], ['height', 'ASC']],\n })\n .then(result => {\n switch (result.length) {\n case 0:\n return resolve(null);\n default:\n return resolve(result[0].claimId);\n }\n })\n .catch(error => {\n reject(error);\n });\n });\n };\n\n Certificate.validateLongChannelId = function (name, claimId) {\n logger.debug(`validateLongChannelId(${name}, ${claimId})`);\n return new Promise((resolve, reject) => {\n this.findOne({\n where: {name, claimId},\n })\n .then(result => {\n if (!result) {\n return resolve(null);\n };\n resolve(claimId);\n })\n .catch(error => {\n reject(error);\n });\n });\n };\n\n Certificate.getLongChannelId = function (channelName, channelClaimId) {\n logger.debug(`getLongChannelId(${channelName}, ${channelClaimId})`);\n if (channelClaimId && (channelClaimId.length === 40)) { // if a full channel id is provided\n return this.validateLongChannelId(channelName, channelClaimId);\n } else if (channelClaimId && channelClaimId.length < 40) { // if a short channel id is provided\n return this.getLongChannelIdFromShortChannelId(channelName, channelClaimId);\n } else {\n return this.getLongChannelIdFromChannelName(channelName); // if no channel id provided\n }\n };\n\n return Certificate;\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/models/certificate.js","module.exports = (sequelize, { STRING }) => {\n const Channel = sequelize.define(\n 'Channel',\n {\n channelName: {\n type : STRING,\n allowNull: false,\n },\n channelClaimId: {\n type : STRING,\n allowNull: false,\n },\n },\n {\n freezeTableName: true,\n }\n );\n\n Channel.associate = db => {\n Channel.belongsTo(db.User);\n Channel.hasOne(db.Certificate);\n };\n\n return Channel;\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/models/channel.js","const logger = require('winston');\nconst { returnShortId } = require('../helpers/sequelizeHelpers.js');\nconst { assetDefaults: { thumbnail: defaultThumbnail }, details: { host } } = require('../../config/siteConfig.js');\n\nfunction determineFileExtensionFromContentType (contentType) {\n switch (contentType) {\n case 'image/jpeg':\n case 'image/jpg':\n return 'jpeg';\n case 'image/png':\n return 'png';\n case 'image/gif':\n return 'gif';\n case 'video/mp4':\n return 'mp4';\n default:\n logger.debug('setting unknown file type as file extension jpeg');\n return 'jpeg';\n }\n};\n\nfunction determineThumbnail (storedThumbnail, defaultThumbnail) {\n if (storedThumbnail === '') {\n return defaultThumbnail;\n }\n return storedThumbnail;\n};\n\nfunction prepareClaimData (claim) {\n // logger.debug('preparing claim data based on resolved data:', claim);\n claim['thumbnail'] = determineThumbnail(claim.thumbnail, defaultThumbnail);\n claim['fileExt'] = determineFileExtensionFromContentType(claim.contentType);\n claim['host'] = host;\n return claim;\n};\n\nmodule.exports = (sequelize, { STRING, BOOLEAN, INTEGER, TEXT, DECIMAL }) => {\n const Claim = sequelize.define(\n 'Claim',\n {\n address: {\n type : STRING,\n default: null,\n },\n amount: {\n type : DECIMAL(19, 8),\n default: null,\n },\n claimId: {\n type : STRING,\n default: null,\n },\n claimSequence: {\n type : INTEGER,\n default: null,\n },\n decodedClaim: {\n type : BOOLEAN,\n default: null,\n },\n depth: {\n type : INTEGER,\n default: null,\n },\n effectiveAmount: {\n type : DECIMAL(19, 8),\n default: null,\n },\n hasSignature: {\n type : BOOLEAN,\n default: null,\n },\n height: {\n type : INTEGER,\n default: null,\n },\n hex: {\n type : TEXT('long'),\n default: null,\n },\n name: {\n type : STRING,\n default: null,\n },\n nout: {\n type : INTEGER,\n default: null,\n },\n txid: {\n type : STRING,\n default: null,\n },\n validAtHeight: {\n type : INTEGER,\n default: null,\n },\n outpoint: {\n type : STRING,\n default: null,\n },\n claimType: {\n type : STRING,\n default: null,\n },\n certificateId: {\n type : STRING,\n default: null,\n },\n author: {\n type : STRING,\n default: null,\n },\n description: {\n type : TEXT('long'),\n default: null,\n },\n language: {\n type : STRING,\n default: null,\n },\n license: {\n type : STRING,\n default: null,\n },\n licenseUrl: {\n type : STRING,\n default: null,\n },\n nsfw: {\n type : BOOLEAN,\n default: null,\n },\n preview: {\n type : STRING,\n default: null,\n },\n thumbnail: {\n type : STRING,\n default: null,\n },\n title: {\n type : STRING,\n default: null,\n },\n metadataVersion: {\n type : STRING,\n default: null,\n },\n contentType: {\n type : STRING,\n default: null,\n },\n source: {\n type : STRING,\n default: null,\n },\n sourceType: {\n type : STRING,\n default: null,\n },\n sourceVersion: {\n type : STRING,\n default: null,\n },\n streamVersion: {\n type : STRING,\n default: null,\n },\n valueVersion: {\n type : STRING,\n default: null,\n },\n channelName: {\n type : STRING,\n allowNull: true,\n default : null,\n },\n },\n {\n freezeTableName: true,\n }\n );\n\n Claim.associate = db => {\n Claim.belongsTo(db.File, {\n foreignKey: {\n allowNull: true,\n },\n });\n };\n\n Claim.getShortClaimIdFromLongClaimId = function (claimId, claimName) {\n logger.debug(`Claim.getShortClaimIdFromLongClaimId for ${claimName}#${claimId}`);\n return new Promise((resolve, reject) => {\n this\n .findAll({\n where: { name: claimName },\n order: [['height', 'ASC']],\n })\n .then(result => {\n switch (result.length) {\n case 0:\n throw new Error('No claim(s) found with that claim name');\n default:\n resolve(returnShortId(result, claimId));\n }\n })\n .catch(error => {\n reject(error);\n });\n });\n };\n\n Claim.getAllChannelClaims = function (channelClaimId) {\n logger.debug(`Claim.getAllChannelClaims for ${channelClaimId}`);\n return new Promise((resolve, reject) => {\n this\n .findAll({\n where: { certificateId: channelClaimId },\n order: [['height', 'ASC']],\n raw : true, // returns an array of only data, not an array of instances\n })\n .then(channelClaimsArray => {\n // logger.debug('channelclaimsarray length:', channelClaimsArray.length);\n switch (channelClaimsArray.length) {\n case 0:\n return resolve(null);\n default:\n channelClaimsArray.forEach(claim => {\n claim['fileExt'] = determineFileExtensionFromContentType(claim.contentType);\n claim['thumbnail'] = determineThumbnail(claim.thumbnail, defaultThumbnail);\n return claim;\n });\n return resolve(channelClaimsArray);\n }\n })\n .catch(error => {\n reject(error);\n });\n });\n };\n\n Claim.getClaimIdByLongChannelId = function (channelClaimId, claimName) {\n logger.debug(`finding claim id for claim ${claimName} from channel ${channelClaimId}`);\n return new Promise((resolve, reject) => {\n this\n .findAll({\n where: { name: claimName, certificateId: channelClaimId },\n order: [['id', 'ASC']],\n })\n .then(result => {\n switch (result.length) {\n case 0:\n return resolve(null);\n case 1:\n return resolve(result[0].claimId);\n default:\n logger.error(`${result.length} records found for \"${claimName}\" in channel \"${channelClaimId}\"`);\n return resolve(result[0].claimId);\n }\n })\n .catch(error => {\n reject(error);\n });\n });\n };\n\n Claim.getLongClaimIdFromShortClaimId = function (name, shortId) {\n return new Promise((resolve, reject) => {\n this\n .findAll({\n where: {\n name,\n claimId: {\n $like: `${shortId}%`,\n }},\n order: [['height', 'ASC']],\n })\n .then(result => {\n switch (result.length) {\n case 0:\n return resolve(null);\n default: // note results must be sorted\n return resolve(result[0].claimId);\n }\n })\n .catch(error => {\n reject(error);\n });\n });\n };\n\n Claim.getTopFreeClaimIdByClaimName = function (name) {\n return new Promise((resolve, reject) => {\n this\n .findAll({\n where: { name },\n order: [['effectiveAmount', 'DESC'], ['height', 'ASC']], // note: maybe height and effective amount need to switch?\n })\n .then(result => {\n logger.debug('length of result', result.length);\n switch (result.length) {\n case 0:\n return resolve(null);\n default:\n return resolve(result[0].dataValues.claimId);\n }\n })\n .catch(error => {\n reject(error);\n });\n });\n };\n\n Claim.validateLongClaimId = function (name, claimId) {\n return new Promise((resolve, reject) => {\n this.findOne({\n where: {name, claimId},\n })\n .then(result => {\n if (!result) {\n return resolve(null);\n };\n resolve(claimId);\n })\n .catch(error => {\n reject(error);\n });\n });\n };\n\n Claim.getLongClaimId = function (claimName, claimId) {\n logger.debug(`getLongClaimId(${claimName}, ${claimId})`);\n if (claimId && (claimId.length === 40)) { // if a full claim id is provided\n return this.validateLongClaimId(claimName, claimId);\n } else if (claimId && claimId.length < 40) {\n return this.getLongClaimIdFromShortClaimId(claimName, claimId); // if a short claim id is provided\n } else {\n return this.getTopFreeClaimIdByClaimName(claimName); // if no claim id is provided\n }\n };\n\n Claim.resolveClaim = function (name, claimId) {\n logger.debug(`Claim.resolveClaim: ${name} ${claimId}`);\n return new Promise((resolve, reject) => {\n this\n .findAll({\n where: { name, claimId },\n })\n .then(claimArray => {\n switch (claimArray.length) {\n case 0:\n return resolve(null);\n case 1:\n return resolve(prepareClaimData(claimArray[0].dataValues));\n default:\n logger.error(`more than one record matches ${name}#${claimId} in db.Claim`);\n return resolve(prepareClaimData(claimArray[0].dataValues));\n }\n })\n .catch(error => {\n reject(error);\n });\n });\n };\n\n return Claim;\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/models/claim.js","module.exports = (sequelize, { STRING, BOOLEAN, INTEGER }) => {\n const File = sequelize.define(\n 'File',\n {\n name: {\n type : STRING,\n allowNull: false,\n },\n claimId: {\n type : STRING,\n allowNull: false,\n },\n address: {\n type : STRING,\n allowNull: false,\n },\n outpoint: {\n type : STRING,\n allowNull: false,\n },\n height: {\n type : INTEGER,\n allowNull: false,\n default : 0,\n },\n fileName: {\n type : STRING,\n allowNull: false,\n },\n filePath: {\n type : STRING,\n allowNull: false,\n },\n fileType: {\n type: STRING,\n },\n nsfw: {\n type : BOOLEAN,\n allowNull : false,\n defaultValue: false,\n },\n trendingEligible: {\n type : BOOLEAN,\n allowNull : false,\n defaultValue: true,\n },\n },\n {\n freezeTableName: true,\n }\n );\n\n File.associate = db => {\n File.hasMany(db.Request);\n File.hasOne(db.Claim);\n };\n\n File.getRecentClaims = function () {\n return this.findAll({\n where: { nsfw: false, trendingEligible: true },\n order: [['createdAt', 'DESC']],\n limit: 25,\n });\n };\n\n return File;\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/models/file.js","module.exports = (sequelize, { STRING, BOOLEAN, TEXT }) => {\n const Request = sequelize.define(\n 'Request',\n {\n action: {\n type : STRING,\n allowNull: false,\n },\n url: {\n type : STRING,\n allowNull: false,\n },\n ipAddress: {\n type : STRING,\n allowNull: true,\n },\n result: {\n type : TEXT('long'),\n allowNull: true,\n default : null,\n },\n },\n {\n freezeTableName: true,\n }\n );\n\n Request.associate = db => {\n Request.belongsTo(db.File, {\n foreignKey: {\n allowNull: true,\n },\n });\n };\n\n return Request;\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/models/request.js","'use strict';\nconst bcrypt = require('bcrypt');\nconst logger = require('winston');\n\nmodule.exports = (sequelize, { STRING }) => {\n const User = sequelize.define(\n 'User',\n {\n userName: {\n type : STRING,\n allowNull: false,\n },\n password: {\n type : STRING,\n allowNull: false,\n },\n },\n {\n freezeTableName: true,\n }\n );\n\n User.associate = db => {\n User.hasOne(db.Channel);\n };\n\n User.prototype.comparePassword = function (password) {\n return bcrypt.compare(password, this.password);\n };\n\n User.prototype.changePassword = function (newPassword) {\n return new Promise((resolve, reject) => {\n // generate a salt string to use for hashing\n bcrypt.genSalt((saltError, salt) => {\n if (saltError) {\n logger.error('salt error', saltError);\n reject(saltError);\n return;\n }\n // generate a hashed version of the user's password\n bcrypt.hash(newPassword, salt, (hashError, hash) => {\n // if there is an error with the hash generation return the error\n if (hashError) {\n logger.error('hash error', hashError);\n reject(hashError);\n return;\n }\n // replace the current password with the new hash\n this\n .update({password: hash})\n .then(() => {\n resolve();\n })\n .catch(error => {\n reject(error);\n });\n });\n });\n });\n };\n\n // pre-save hook method to hash the user's password before the user's info is saved to the db.\n User.hook('beforeCreate', (user, options) => {\n logger.debug('User.beforeCreate hook...');\n return new Promise((resolve, reject) => {\n // generate a salt string to use for hashing\n bcrypt.genSalt((saltError, salt) => {\n if (saltError) {\n logger.error('salt error', saltError);\n reject(saltError);\n return;\n }\n // generate a hashed version of the user's password\n bcrypt.hash(user.password, salt, (hashError, hash) => {\n // if there is an error with the hash generation return the error\n if (hashError) {\n logger.error('hash error', hashError);\n reject(hashError);\n return;\n }\n // replace the password string with the hash password value\n user.password = hash;\n resolve();\n });\n });\n });\n });\n\n return User;\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/models/user.js","module.exports = require(\"bcrypt\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"bcrypt\"\n// module id = 51\n// module chunks = 0","const PassportLocalStrategy = require('passport-local').Strategy;\nconst lbryApi = require('../helpers/lbryApi.js');\nconst logger = require('winston');\nconst db = require('models');\n\nmodule.exports = new PassportLocalStrategy(\n {\n usernameField: 'username',\n passwordField: 'password',\n },\n (username, password, done) => {\n logger.verbose(`new channel signup request. user: ${username} pass: ${password} .`);\n let userInfo = {};\n // server-side validaton of inputs (username, password)\n\n // create the channel and retrieve the metadata\n return lbryApi.createChannel(`@${username}`)\n .then(tx => {\n // create user record\n const userData = {\n userName: username,\n password: password,\n };\n logger.verbose('userData >', userData);\n // create user record\n const channelData = {\n channelName : `@${username}`,\n channelClaimId: tx.claim_id,\n };\n logger.verbose('channelData >', channelData);\n // create certificate record\n const certificateData = {\n claimId: tx.claim_id,\n name : `@${username}`,\n // address,\n };\n logger.verbose('certificateData >', certificateData);\n // save user and certificate to db\n return Promise.all([db.User.create(userData), db.Channel.create(channelData), db.Certificate.create(certificateData)]);\n })\n .then(([newUser, newChannel, newCertificate]) => {\n logger.verbose('user and certificate successfully created');\n // store the relevant newUser info to be passed back for req.User\n userInfo['id'] = newUser.id;\n userInfo['userName'] = newUser.userName;\n userInfo['channelName'] = newChannel.channelName;\n userInfo['channelClaimId'] = newChannel.channelClaimId;\n // associate the instances\n return Promise.all([newCertificate.setChannel(newChannel), newChannel.setUser(newUser)]);\n })\n .then(() => {\n logger.verbose('user and certificate successfully associated');\n return db.Certificate.getShortChannelIdFromLongChannelId(userInfo.channelClaimId, userInfo.channelName);\n })\n .then(shortChannelId => {\n userInfo['shortChannelId'] = shortChannelId;\n return done(null, userInfo);\n })\n .catch(error => {\n logger.error('signup error', error);\n return done(error);\n });\n }\n);\n\n\n\n// WEBPACK FOOTER //\n// ./server/speechPassport/local-signup.js","module.exports = require(\"axios\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"axios\"\n// module id = 53\n// module chunks = 0","const lbryConfig = {\n api: {\n apiHost: 'localhost',\n apiPort: '5279',\n },\n};\n\nmodule.exports = lbryConfig;\n\n\n\n// WEBPACK FOOTER //\n// ./config/lbryConfig.js","module.exports = require(\"universal-analytics\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"universal-analytics\"\n// module id = 55\n// module chunks = 0","module.exports = {\n serializeSpeechUser (user, done) { // returns user data to be serialized into session\n console.log('serializing user');\n done(null, user);\n },\n deserializeSpeechUser (user, done) { // deserializes session and populates additional info to req.user\n console.log('deserializing user');\n done(null, user);\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/authHelpers.js","const speechPassport = require('speechPassport');\nconst handleSignupRequest = require('./signup');\nconst handleLoginRequest = require('./login');\nconst handleLogoutRequest = require('./logout');\nconst handleUserRequest = require('./user');\n\nmodule.exports = (app) => {\n app.post('/signup', speechPassport.authenticate('local-signup'), handleSignupRequest);\n app.post('/login', handleLoginRequest);\n app.get('/logout', handleLogoutRequest);\n app.get('/user', handleUserRequest);\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/auth/index.js","const signup = (req, res) => {\n res.status(200).json({\n success : true,\n channelName : req.user.channelName,\n channelClaimId: req.user.channelClaimId,\n shortChannelId: req.user.shortChannelId,\n });\n};\n\nmodule.exports = signup;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/auth/signup.js","const speechPassport = require('speechPassport');\n\nconst login = (req, res, next) => {\n speechPassport.authenticate('local-login', (err, user, info) => {\n if (err) {\n return next(err);\n }\n if (!user) {\n return res.status(400).json({\n success: false,\n message: info.message,\n });\n }\n req.logIn(user, (err) => {\n if (err) {\n return next(err);\n }\n return res.status(200).json({\n success : true,\n channelName : req.user.channelName,\n channelClaimId: req.user.channelClaimId,\n shortChannelId: req.user.shortChannelId,\n });\n });\n })(req, res, next);\n};\n\nmodule.exports = login;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/auth/login.js","const logout = (req, res) => {\n req.logout();\n res.status(200).json({success: true, message: 'you successfully logged out'});\n};\n\nmodule.exports = logout;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/auth/logout.js","const user = (req, res) => {\n if (req.user) {\n res.status(200).json({success: true, data: req.user});\n } else {\n res.status(401).json({success: false, message: 'user is not logged in'});\n }\n};\n\nmodule.exports = user;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/auth/user.js","const channelAvailability = require('./channelAvailability');\nconst channelClaims = require('./channelClaims');\nconst channelData = require('./channelData');\nconst channelShortId = require('./channelShortId');\nconst claimAvailability = require('./claimAvailability');\nconst claimData = require('./claimData');\nconst claimGet = require('./claimGet');\nconst claimLongId = require('./claimLongId');\nconst claimPublish = require('./claimPublish');\nconst claimResolve = require('./claimResolve');\nconst claimShortId = require('./claimShortId');\nconst claimList = require('./claimList');\nconst fileAvailability = require('./fileAvailability');\n\nconst multipartMiddleware = require('helpers/multipartMiddleware');\n\nmodule.exports = (app) => {\n // channel routes\n app.get('/api/channel/availability/:name', channelAvailability);\n app.get('/api/channel/short-id/:longId/:name', channelShortId);\n app.get('/api/channel/data/:channelName/:channelClaimId', channelData);\n app.get('/api/channel/claims/:channelName/:channelClaimId/:page', channelClaims);\n // claim routes\n app.get('/api/claim/list/:name', claimList);\n app.get('/api/claim/get/:name/:claimId', claimGet);\n app.get('/api/claim/availability/:name', claimAvailability);\n app.get('/api/claim/resolve/:name/:claimId', claimResolve);\n app.post('/api/claim/publish', multipartMiddleware, claimPublish);\n app.get('/api/claim/short-id/:longId/:name', claimShortId);\n app.post('/api/claim/long-id', claimLongId);\n app.get('/api/claim/data/:claimName/:claimId', claimData);\n // file routes\n app.get('/api/file/availability/:name/:claimId', fileAvailability);\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/index.js","const { checkChannelAvailability } = require('controllers/publishController.js');\nconst { sendGATimingEvent } = require('helpers/googleAnalytics.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\n\n/*\n\n route to check whether site has published to a channel\n\n*/\n\nconst channelAvailability = ({ ip, originalUrl, params: { name } }, res) => {\n const gaStartTime = Date.now();\n checkChannelAvailability(name)\n .then(availableName => {\n res.status(200).json(availableName);\n sendGATimingEvent('end-to-end', 'claim name availability', name, gaStartTime, Date.now());\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = channelAvailability;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/channelAvailability.js","module.exports = require(\"fs\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"fs\"\n// module id = 64\n// module chunks = 0","const { getChannelClaims } = require('controllers/serveController.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\n\nconst NO_CHANNEL = 'NO_CHANNEL';\n\n/*\n\n route to get all claims for channel\n\n*/\n\nconst channelClaims = ({ ip, originalUrl, body, params }, res) => {\n const channelName = params.channelName;\n let channelClaimId = params.channelClaimId;\n if (channelClaimId === 'none') channelClaimId = null;\n const page = params.page;\n getChannelClaims(channelName, channelClaimId, page)\n .then(data => {\n if (data === NO_CHANNEL) {\n return res.status(404).json({success: false, message: 'No matching channel was found'});\n }\n res.status(200).json({success: true, data});\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = channelClaims;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/channelClaims.js","const CLAIMS_PER_PAGE = 12;\n\nmodule.exports = {\n returnPaginatedChannelClaims (channelName, longChannelClaimId, claims, page) {\n const totalPages = module.exports.determineTotalPages(claims);\n const paginationPage = module.exports.getPageFromQuery(page);\n const viewData = {\n channelName : channelName,\n longChannelClaimId: longChannelClaimId,\n claims : module.exports.extractPageFromClaims(claims, paginationPage),\n previousPage : module.exports.determinePreviousPage(paginationPage),\n currentPage : paginationPage,\n nextPage : module.exports.determineNextPage(totalPages, paginationPage),\n totalPages : totalPages,\n totalResults : module.exports.determineTotalClaims(claims),\n };\n return viewData;\n },\n getPageFromQuery (page) {\n if (page) {\n return parseInt(page);\n }\n return 1;\n },\n extractPageFromClaims (claims, pageNumber) {\n if (!claims) {\n return []; // if no claims, return this default\n }\n // logger.debug('claims is array?', Array.isArray(claims));\n // logger.debug(`pageNumber ${pageNumber} is number?`, Number.isInteger(pageNumber));\n const claimStartIndex = (pageNumber - 1) * CLAIMS_PER_PAGE;\n const claimEndIndex = claimStartIndex + CLAIMS_PER_PAGE;\n const pageOfClaims = claims.slice(claimStartIndex, claimEndIndex);\n return pageOfClaims;\n },\n determineTotalPages (claims) {\n if (!claims) {\n return 0;\n } else {\n const totalClaims = claims.length;\n if (totalClaims < CLAIMS_PER_PAGE) {\n return 1;\n }\n const fullPages = Math.floor(totalClaims / CLAIMS_PER_PAGE);\n const remainder = totalClaims % CLAIMS_PER_PAGE;\n if (remainder === 0) {\n return fullPages;\n }\n return fullPages + 1;\n }\n },\n determinePreviousPage (currentPage) {\n if (currentPage === 1) {\n return null;\n }\n return currentPage - 1;\n },\n determineNextPage (totalPages, currentPage) {\n if (currentPage === totalPages) {\n return null;\n }\n return currentPage + 1;\n },\n determineTotalClaims (claims) {\n if (!claims) {\n return 0;\n }\n return claims.length;\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/channelPagination.js","const { getChannelData } = require('controllers/serveController.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\n\nconst NO_CHANNEL = 'NO_CHANNEL';\n\n/*\n\n route to get data for a channel\n\n*/\n\nconst channelData = ({ ip, originalUrl, body, params }, res) => {\n const channelName = params.channelName;\n let channelClaimId = params.channelClaimId;\n if (channelClaimId === 'none') channelClaimId = null;\n getChannelData(channelName, channelClaimId, 0)\n .then(data => {\n if (data === NO_CHANNEL) {\n return res.status(404).json({success: false, message: 'No matching channel was found'});\n }\n res.status(200).json({success: true, data});\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = channelData;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/channelData.js","const { handleErrorResponse } = require('helpers/errorHandlers.js');\nconst db = require('models');\n\n/*\n\nroute to get a short channel id from long channel Id\n\n*/\n\nconst channelShortIdRoute = ({ ip, originalUrl, params }, res) => {\n db.Certificate.getShortChannelIdFromLongChannelId(params.longId, params.name)\n .then(shortId => {\n res.status(200).json(shortId);\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = channelShortIdRoute;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/channelShortId.js","const { claimNameIsAvailable } = require('controllers/publishController.js');\nconst { sendGATimingEvent } = require('helpers/googleAnalytics.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\n\n/*\n\n route to check whether this site published to a claim\n\n*/\n\nconst claimAvailability = ({ ip, originalUrl, params: { name } }, res) => {\n const gaStartTime = Date.now();\n claimNameIsAvailable(name)\n .then(result => {\n res.status(200).json(result);\n sendGATimingEvent('end-to-end', 'claim name availability', name, gaStartTime, Date.now());\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimAvailability;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimAvailability.js","const { handleErrorResponse } = require('helpers/errorHandlers.js');\nconst db = require('models');\n\n/*\n\n route to return data for a claim\n\n*/\n\nconst claimData = ({ ip, originalUrl, body, params }, res) => {\n const claimName = params.claimName;\n let claimId = params.claimId;\n if (claimId === 'none') claimId = null;\n db.Claim.resolveClaim(claimName, claimId)\n .then(claimInfo => {\n if (!claimInfo) {\n return res.status(404).json({success: false, message: 'No claim could be found'});\n }\n res.status(200).json({success: true, data: claimInfo});\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimData;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimData.js","const { getClaim } = require('helpers/lbryApi.js');\nconst { addGetResultsToFileData, createFileData } = require('../../helpers/publishHelpers.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\nconst db = require('models');\n\n/*\n\n route to get a claim\n\n*/\n\nconst claimGet = ({ ip, originalUrl, params }, res) => {\n const name = params.name;\n const claimId = params.claimId;\n // resolve the claim\n db.Claim.resolveClaim(name, claimId)\n .then(resolveResult => {\n // make sure a claim actually exists at that uri\n if (!resolveResult) {\n throw new Error('No matching uri found in Claim table');\n }\n let fileData = createFileData(resolveResult);\n // get the claim\n return Promise.all([fileData, getClaim(`${name}#${claimId}`)]);\n })\n .then(([ fileData, getResult ]) => {\n fileData = addGetResultsToFileData(fileData, getResult);\n return Promise.all([db.upsert(db.File, fileData, {name, claimId}, 'File'), getResult]);\n })\n .then(([ fileRecord, {message, completed} ]) => {\n res.status(200).json({ success: true, message, completed });\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimGet;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimGet.js","const { getClaimId } = require('controllers/serveController.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\n\nconst NO_CHANNEL = 'NO_CHANNEL';\nconst NO_CLAIM = 'NO_CLAIM';\n\n/*\n\n route to get a long claim id\n\n*/\n\nconst claimLongId = ({ ip, originalUrl, body, params }, res) => {\n const channelName = body.channelName;\n const channelClaimId = body.channelClaimId;\n const claimName = body.claimName;\n const claimId = body.claimId;\n getClaimId(channelName, channelClaimId, claimName, claimId)\n .then(result => {\n if (result === NO_CHANNEL) {\n return res.status(404).json({success: false, message: 'No matching channel could be found'});\n }\n if (result === NO_CLAIM) {\n return res.status(404).json({success: false, message: 'No matching claim id could be found'});\n }\n res.status(200).json({success: true, data: result});\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimLongId;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimLongId.js","const { createBasicPublishParams, createThumbnailPublishParams, parsePublishApiRequestBody, parsePublishApiRequestFiles } = require('helpers/publishHelpers.js');\nconst { claimNameIsAvailable, publish } = require('controllers/publishController.js');\nconst { authenticateUser } = require('auth/authentication.js');\nconst { sendGATimingEvent } = require('helpers/googleAnalytics.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\nconst { details: { host } } = require('siteConfig.js');\n\n/*\n\n route to publish a claim through the daemon\n\n*/\n\nconst claimPublish = ({ body, files, headers, ip, originalUrl, user }, res) => {\n // define variables\n let channelName, channelId, channelPassword, description, fileName, filePath, fileType, gaStartTime, license, name, nsfw, thumbnail, thumbnailFileName, thumbnailFilePath, thumbnailFileType, title;\n // record the start time of the request\n gaStartTime = Date.now();\n // validate the body and files of the request\n try {\n // validateApiPublishRequest(body, files);\n ({name, nsfw, license, title, description, thumbnail} = parsePublishApiRequestBody(body));\n ({fileName, filePath, fileType, thumbnailFileName, thumbnailFilePath, thumbnailFileType} = parsePublishApiRequestFiles(files));\n ({channelName, channelId, channelPassword} = body);\n } catch (error) {\n return res.status(400).json({success: false, message: error.message});\n }\n // check channel authorization\n Promise\n .all([\n authenticateUser(channelName, channelId, channelPassword, user),\n claimNameIsAvailable(name),\n createBasicPublishParams(filePath, name, title, description, license, nsfw, thumbnail),\n createThumbnailPublishParams(thumbnailFilePath, name, license, nsfw),\n ])\n .then(([{channelName, channelClaimId}, validatedClaimName, publishParams, thumbnailPublishParams]) => {\n // add channel details to the publish params\n if (channelName && channelClaimId) {\n publishParams['channel_name'] = channelName;\n publishParams['channel_id'] = channelClaimId;\n }\n // publish the thumbnail\n if (thumbnailPublishParams) {\n publish(thumbnailPublishParams, thumbnailFileName, thumbnailFileType);\n }\n // publish the asset\n return publish(publishParams, fileName, fileType);\n })\n .then(result => {\n res.status(200).json({\n success: true,\n message: 'publish completed successfully',\n data : {\n name,\n claimId: result.claim_id,\n url : `${host}/${result.claim_id}/${name}`,\n lbryTx : result,\n },\n });\n // record the publish end time and send to google analytics\n sendGATimingEvent('end-to-end', 'publish', fileType, gaStartTime, Date.now());\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimPublish;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimPublish.js","const db = require('models');\nconst logger = require('winston');\n\nmodule.exports = {\n authenticateUser (channelName, channelId, channelPassword, user) {\n // case: no channelName or channel Id are provided (anonymous), regardless of whether user token is provided\n if (!channelName && !channelId) {\n return {\n channelName : null,\n channelClaimId: null,\n };\n }\n // case: channelName or channel Id are provided with user token\n if (user) {\n if (channelName && channelName !== user.channelName) {\n throw new Error('the provided channel name does not match user credentials');\n }\n if (channelId && channelId !== user.channelClaimId) {\n throw new Error('the provided channel id does not match user credentials');\n }\n return {\n channelName : user.channelName,\n channelClaimId: user.channelClaimId,\n };\n }\n // case: channelName or channel Id are provided with password instead of user token\n if (!channelPassword) throw new Error('no channel password provided');\n return module.exports.authenticateChannelCredentials(channelName, channelId, channelPassword);\n },\n authenticateChannelCredentials (channelName, channelId, userPassword) {\n return new Promise((resolve, reject) => {\n // hoisted variables\n let channelData;\n // build the params for finding the channel\n let channelFindParams = {};\n if (channelName) channelFindParams['channelName'] = channelName;\n if (channelId) channelFindParams['channelClaimId'] = channelId;\n // find the channel\n db.Channel\n .findOne({\n where: channelFindParams,\n })\n .then(channel => {\n if (!channel) {\n logger.debug('no channel found');\n throw new Error('Authentication failed, you do not have access to that channel');\n }\n channelData = channel.get();\n logger.debug('channel data:', channelData);\n return db.User.findOne({\n where: { userName: channelData.channelName.substring(1) },\n });\n })\n .then(user => {\n if (!user) {\n logger.debug('no user found');\n throw new Error('Authentication failed, you do not have access to that channel');\n }\n return user.comparePassword(userPassword);\n })\n .then(isMatch => {\n if (!isMatch) {\n logger.debug('incorrect password');\n throw new Error('Authentication failed, you do not have access to that channel');\n }\n logger.debug('...password was a match...');\n resolve(channelData);\n })\n .catch(error => {\n reject(error);\n });\n });\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/auth/authentication.js","const { resolveUri } = require('helpers/lbryApi.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\n\n/*\n\n route to run a resolve request on the daemon\n\n*/\n\nconst claimResolve = ({ headers, ip, originalUrl, params }, res) => {\n resolveUri(`${params.name}#${params.claimId}`)\n .then(resolvedUri => {\n res.status(200).json(resolvedUri);\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimResolve;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimResolve.js","const { handleErrorResponse } = require('helpers/errorHandlers.js');\nconst db = require('models');\n\n/*\n\n route to get a short claim id from long claim Id\n\n*/\n\nconst claimShortId = ({ ip, originalUrl, body, params }, res) => {\n db.Claim.getShortClaimIdFromLongClaimId(params.longId, params.name)\n .then(shortId => {\n res.status(200).json({success: true, data: shortId});\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimShortId;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimShortId.js","const { getClaimList } = require('helpers/lbryApi.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\n\n/*\n\n route to get list of claims\n\n*/\n\nconst claimList = ({ ip, originalUrl, params }, res) => {\n getClaimList(params.name)\n .then(claimsList => {\n res.status(200).json(claimsList);\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimList;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimList.js","const { handleErrorResponse } = require('helpers/errorHandlers.js');\nconst db = require('models');\n\n/*\n\n route to see if asset is available locally\n\n*/\n\nconst fileAvailability = ({ ip, originalUrl, params }, res) => {\n const name = params.name;\n const claimId = params.claimId;\n db.File\n .findOne({\n where: {\n name,\n claimId,\n },\n })\n .then(result => {\n if (result) {\n return res.status(200).json({success: true, data: true});\n }\n res.status(200).json({success: true, data: false});\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = fileAvailability;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/fileAvailability.js","const multipart = require('connect-multiparty');\r\nconst { publishing: { uploadDirectory } } = require('siteConfig.js');\r\nconst multipartMiddleware = multipart({uploadDir: uploadDirectory});\r\n\r\nmodule.exports = multipartMiddleware;\r\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/multipartMiddleware.js","module.exports = require(\"connect-multiparty\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"connect-multiparty\"\n// module id = 80\n// module chunks = 0","const handlePageRequest = require('./sendReactApp');\nconst handleEmbedRequest = require('./sendEmbedPage');\nconst redirect = require('./redirect');\n\nmodule.exports = (app) => {\n app.get('/', handlePageRequest);\n app.get('/login', handlePageRequest);\n app.get('/about', handlePageRequest);\n app.get('/trending', redirect('/popular'));\n app.get('/popular', handlePageRequest);\n app.get('/new', handlePageRequest);\n app.get('/embed/:claimId/:name', handleEmbedRequest); // route to send embedable video player (for twitter)\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/pages/index.js","const handlePageRender = require('helpers/handlePageRender.jsx');\n\nconst sendReactApp = (req, res) => {\n handlePageRender(req, res);\n};\n\nmodule.exports = sendReactApp;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/pages/sendReactApp.js","const { details: { host } } = require('siteConfig.js');\n\nconst sendEmbedPage = ({ params }, res) => {\n const claimId = params.claimId;\n const name = params.name;\n // get and render the content\n res.status(200).render('embed', { layout: 'embed', host, claimId, name });\n};\n\nmodule.exports = sendEmbedPage;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/pages/sendEmbedPage.js","const redirect = (route) => {\n return (req, res) => {\n res.status(301).redirect(route);\n };\n};\n\nmodule.exports = redirect;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/pages/redirect.js","const serveAssetByClaim = require('./serveAssetByClaim');\nconst serveAssetByIdentifierAndClaim = require('./serveAssetByIdentifierAndClaim');\n\nmodule.exports = (app, db) => {\n app.get('/:identifier/:claim', serveAssetByIdentifierAndClaim);\n app.get('/:claim', serveAssetByClaim);\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/assets/index.js","const { sendGAServeEvent } = require('helpers/googleAnalytics');\nconst { determineResponseType, logRequestData, getClaimIdAndServeAsset } = require('helpers/serveHelpers.js');\nconst lbryUri = require('helpers/lbryUri.js');\nconst handleShowRender = require('helpers/handleShowRender.jsx');\nconst SERVE = 'SERVE';\n\n/*\n\n route to serve an asset or the react app via the claim name only\n\n*/\n\nconst serverAssetByClaim = (req, res) => {\n const { headers, ip, originalUrl, params } = req;\n // decide if this is a show request\n let hasFileExtension;\n try {\n ({ hasFileExtension } = lbryUri.parseModifier(params.claim));\n } catch (error) {\n return res.status(400).json({success: false, message: error.message});\n }\n let responseType = determineResponseType(hasFileExtension, headers);\n if (responseType !== SERVE) {\n return handleShowRender(req, res);\n }\n // handle serve request\n // send google analytics\n sendGAServeEvent(headers, ip, originalUrl);\n // parse the claim\n let claimName;\n try {\n ({claimName} = lbryUri.parseClaim(params.claim));\n } catch (error) {\n return res.status(400).json({success: false, message: error.message});\n }\n // log the request data for debugging\n logRequestData(responseType, claimName, null, null);\n // get the claim Id and then serve the asset\n getClaimIdAndServeAsset(null, null, claimName, null, originalUrl, ip, res);\n};\n\nmodule.exports = serverAssetByClaim;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/assets/serveAssetByClaim.js","module.exports = require(\"redux-saga\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"redux-saga\"\n// module id = 87\n// module chunks = 0","module.exports = require(\"redux-saga/effects\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"redux-saga/effects\"\n// module id = 88\n// module chunks = 0","const { sendGAServeEvent } = require('helpers/googleAnalytics');\nconst {\n determineResponseType,\n flipClaimNameAndIdForBackwardsCompatibility,\n logRequestData,\n getClaimIdAndServeAsset,\n} = require('helpers/serveHelpers.js');\nconst lbryUri = require('helpers/lbryUri.js');\nconst handleShowRender = require('helpers/handleShowRender.jsx');\n\nconst SERVE = 'SERVE';\n\n/*\n\n route to serve an asset or the react app via the claim name and an identifier\n\n*/\n\nconst serverAssetByIdentifierAndClaim = (req, res) => {\n const { headers, ip, originalUrl, params } = req;\n // decide if this is a show request\n let hasFileExtension;\n try {\n ({ hasFileExtension } = lbryUri.parseModifier(params.claim));\n } catch (error) {\n return res.status(400).json({success: false, message: error.message});\n }\n let responseType = determineResponseType(hasFileExtension, headers);\n if (responseType !== SERVE) {\n return handleShowRender(req, res);\n }\n // handle serve request\n // send google analytics\n sendGAServeEvent(headers, ip, originalUrl);\n // parse the claim\n let claimName;\n try {\n ({ claimName } = lbryUri.parseClaim(params.claim));\n } catch (error) {\n return res.status(400).json({success: false, message: error.message});\n }\n // parse the identifier\n let isChannel, channelName, channelClaimId, claimId;\n try {\n ({ isChannel, channelName, channelClaimId, claimId } = lbryUri.parseIdentifier(params.identifier));\n } catch (error) {\n return res.status(400).json({success: false, message: error.message});\n }\n if (!isChannel) {\n [claimId, claimName] = flipClaimNameAndIdForBackwardsCompatibility(claimId, claimName);\n }\n // log the request data for debugging\n logRequestData(responseType, claimName, channelName, claimId);\n // get the claim Id and then serve the asset\n getClaimIdAndServeAsset(channelName, channelClaimId, claimName, claimId, originalUrl, ip, res);\n};\n\nmodule.exports = serverAssetByIdentifierAndClaim;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/assets/serveAssetByIdentifierAndClaim.js","const handlePageRequest = require('./sendReactApp');\r\n\r\nmodule.exports = (app) => {\r\n app.get('*', handlePageRequest);\r\n};\r\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/fallback/index.js","const handlePageRender = require('helpers/handlePageRender.jsx');\n\nconst sendReactApp = (req, res) => {\n handlePageRender(req, res);\n};\n\nmodule.exports = sendReactApp;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/fallback/sendReactApp.js"],"sourceRoot":""} \ No newline at end of file +{"version":3,"sources":["webpack:///index.js","webpack:///webpack/bootstrap 82ebf2ca30129849e4ef","webpack:///external \"winston\"","webpack:///./server/helpers/errorHandlers.js","webpack:///./server/models/index.js","webpack:///./config/siteConfig.js","webpack:///./server/helpers/googleAnalytics.js","webpack:///./server/helpers/lbryApi.js","webpack:///./server/controllers/serveController.js","webpack:///./server/speechPassport/index.js","webpack:///./server/controllers/publishController.js","webpack:///./server/helpers/publishHelpers.js","webpack:///./config/mysqlConfig.js","webpack:///external \"passport-local\"","webpack:///./server/helpers/sequelizeHelpers.js","webpack:///external \"sequelize\"","webpack:///./server/helpers/handlePageRender.jsx","webpack:///external \"react\"","webpack:///external \"react-dom/server\"","webpack:///external \"redux\"","webpack:///external \"react-redux\"","webpack:///external \"react-router-dom\"","webpack:///external \"spee.ch-components\"","webpack:///./server/helpers/renderFullPage.js","webpack:///external \"react-helmet\"","webpack:///./server/helpers/serveHelpers.js","webpack:///./server/helpers/lbryUri.js","webpack:///./server/helpers/handleShowRender.jsx","webpack:///external \"babel-polyfill\"","webpack:///external \"whatwg-fetch\"","webpack:///./speech.js","webpack:///./server/index.js","webpack:///external \"express\"","webpack:///external \"body-parser\"","webpack:///external \"express-handlebars\"","webpack:///external \"handlebars\"","webpack:///external \"helmet\"","webpack:///external \"cookie-session\"","webpack:///external \"http\"","webpack:///./server/middleware/requestLogger.js","webpack:///external \"path\"","webpack:///./config/loggerConfig.js","webpack:///./config/slackConfig.js","webpack:///external \"winston-slack-webhook\"","webpack:///external \"passport\"","webpack:///./server/speechPassport/local-login.js","webpack:///./server/models/certificate.js","webpack:///./server/models/channel.js","webpack:///./server/models/claim.js","webpack:///./server/models/file.js","webpack:///./server/models/request.js","webpack:///./server/models/user.js","webpack:///external \"bcrypt\"","webpack:///./server/speechPassport/local-signup.js","webpack:///external \"axios\"","webpack:///./config/lbryConfig.js","webpack:///external \"universal-analytics\"","webpack:///./server/helpers/authHelpers.js","webpack:///./server/routes/auth/index.js","webpack:///./server/routes/auth/signup.js","webpack:///./server/routes/auth/login.js","webpack:///./server/routes/auth/logout.js","webpack:///./server/routes/auth/user.js","webpack:///./server/routes/api/index.js","webpack:///./server/routes/api/channelAvailability.js","webpack:///external \"fs\"","webpack:///./server/routes/api/channelClaims.js","webpack:///./server/helpers/channelPagination.js","webpack:///./server/routes/api/channelData.js","webpack:///./server/routes/api/channelShortId.js","webpack:///./server/routes/api/claimAvailability.js","webpack:///./server/routes/api/claimData.js","webpack:///./server/routes/api/claimGet.js","webpack:///./server/routes/api/claimLongId.js","webpack:///./server/routes/api/claimPublish.js","webpack:///./server/auth/authentication.js","webpack:///./server/routes/api/claimResolve.js","webpack:///./server/routes/api/claimShortId.js","webpack:///./server/routes/api/claimList.js","webpack:///./server/routes/api/fileAvailability.js","webpack:///./server/helpers/multipartMiddleware.js","webpack:///external \"connect-multiparty\"","webpack:///./server/routes/pages/index.js","webpack:///./server/routes/pages/sendReactApp.js","webpack:///./server/routes/pages/sendEmbedPage.js","webpack:///./server/routes/pages/redirect.js","webpack:///./server/routes/assets/index.js","webpack:///./server/routes/assets/serveAssetByClaim.js","webpack:///external \"redux-saga\"","webpack:///external \"redux-saga/effects\"","webpack:///./server/routes/assets/serveAssetByIdentifierAndClaim.js","webpack:///./server/routes/fallback/index.js","webpack:///./server/routes/fallback/sendReactApp.js"],"names":["module","exports","modules","__webpack_require__","moduleId","installedModules","i","l","call","m","c","d","name","getter","o","Object","defineProperty","configurable","enumerable","get","n","__esModule","object","property","prototype","hasOwnProperty","p","s","require","_slicedToArray","sliceIterator","arr","_arr","_n","_d","_e","undefined","_s","_i","Symbol","iterator","next","done","push","value","length","err","Array","isArray","TypeError","logger","handleErrorResponse","originalUrl","ip","error","res","useObjectPropertiesIfNoKeys","_module$exports$retur","returnErrorMessageAndStatus","_module$exports$retur2","status","message","json","createErrorResponsePayload","code","keys","newErrorObject","getOwnPropertyNames","forEach","key","success","Certificate","Channel","Claim","File","Request","User","Sequelize","_require","database","username","password","sequelize","host","dialect","dialectOptions","decimalNumbers","logging","pool","max","min","idle","acquire","authenticate","then","info","catch","db","import","modelName","associate","upsert","Model","values","condition","tableName","findOne","where","obj","debug","update","create","SiteConfig","_this","this","analytics","googleId","assetDefaults","description","thumbnail","title","auth","sessionKey","customComponents","components","containers","pages","details","port","twitter","publishing","additionalClaimAddresses","disabled","disabledMessage","primaryClaimAddress","thumbnailChannel","thumbnailChannelId","uploadDirectory","routes","config","console","log","createServeEventParams","headers","eventCategory","eventAction","eventLabel","ipOverride","userAgentOverride","createPublishTimingEventParams","category","variable","label","startTime","endTime","userTimingCategory","userTimingVariableName","userTimingTime","userTimingLabel","sendGoogleAnalyticsEvent","params","visitorId","replace","ua","strictCidFormat","https","event","sendGoogleAnalyticsTiming","timing","sendGAServeEvent","sendGATimingEvent","chooseGaLbrynetPublishLabel","_ref","channelName","channel_name","channelId","channel_id","axios","_require$api","api","apiHost","apiPort","lbryApiUri","_require2","handleLbrynetResponse","resolve","reject","data","result","Error","JSON","stringify","publishClaim","publishParams","gaStartTime","Date","now","Promise","post","method","response","getClaim","uri","timeout","getClaimList","claimName","resolveUri","_ref2","getDownloadDirectory","_ref3","download_directory","createChannel","amount","returnPaginatedChannelClaims","getClaimId","channelClaimId","claimId","getClaimIdByChannel","getClaimIdByClaim","getLongClaimId","longClaimId","getLongChannelId","longChannelId","all","getClaimIdByLongChannelId","getChannelData","page","longChannelClaimId","getShortChannelIdFromLongChannelId","_ref4","shortChannelClaimId","getChannelClaims","getAllChannelClaims","_ref5","_ref6","channelClaimsArray","paginatedChannelViewData","getLocalFileRecord","file","dataValues","passport","localLoginStrategy","localSignupStrategy","serializeSpeechUser","deserializeSpeechUser","deserializeUser","serializeUser","use","_defineProperty","writable","lbryApi","publishHelpers","_require$publishing","Op","publish","fileName","fileType","publishResults","certificateId","tx","channel","fileRecord","claim_id","metadata","address","claim_address","outpoint","txid","nout","height","filePath","file_path","nsfw","claimRecord","contentType","bid","upsertCriteria","claim","setClaim","setFile","deleteTemporaryFile","claimNameIsAvailable","claimAddresses","findAll","attributes","or","checkChannelAvailability","fs","parsePublishApiRequestBody","license","exec","parsePublishApiRequestFiles","path","type","size","test","validateFileTypeAndSize","thumbnailFileName","thumbnailFilePath","thumbnailFileType","createBasicPublishParams","trim","author","language","createThumbnailPublishParams","unlink","addGetResultsToFileData","fileInfo","getResult","file_name","download_path","createFileData","mysql","warn","returnShortId","claimsArray","longId","claimIndex","shortId","substring","shortIdLength","findIndex","element","possibleMatches","slice","filter","_interopRequireDefault","default","_react","_react2","_server","_redux","_reactRedux","_reactRouterDom","_spee","_renderFullPage","_renderFullPage2","_reactHelmet","_reactHelmet2","siteConfig","req","context","customizedReducers","Reducers","store","createStore","html","renderToString","createElement","Provider","StaticRouter","location","url","GAListener","App","helmet","renderStatic","redirect","preloadedState","getState","send","toString","meta","link","clientAcceptsHtml","accept","match","requestIsFromBrowser","clientWantsAsset","range","imageIsWanted","videoIsWanted","isValidClaimId","isValidShortId","isValidShortIdOrClaimId","input","serveAssetToClient","NO_FILE","verbose","sendFileOptions","X-Content-Type-Options","Content-Type","sendFile","getClaimIdAndServeAsset","fullClaimId","determineResponseType","hasFileExtension","responseType","flipClaimNameAndIdForBackwardsCompatibility","identifier","tempName","logRequestData","REGEXP_INVALID_CLAIM","REGEXP_INVALID_CHANNEL","REGEXP_ADDRESS","CHANNEL_CHAR","parseIdentifier","componentsRegex","RegExp","_componentsRegex$exec","map","_componentsRegex$exec2","proto","modifierSeperator","modifier","isChannel","startsWith","nameBadChars","join","parseClaim","_componentsRegex$exec3","_componentsRegex$exec4","parseModifier","_componentsRegex$exec5","_componentsRegex$exec6","_reduxSaga","_reduxSaga2","_effects","returnSagaWithParams","saga","regeneratorRuntime","mark","_callee","wrap","_context","prev","stop","sagaMiddleware","middleware","applyMiddleware","action","Actions","onHandleShowPageUri","Sagas","handleShowPageUri","run","Server","configureLogger","userConfig","loggerConfig","configureMysql","mysqlConfig","configureSiteDetails","configureSlack","slackConfig","configureClientBundle","configureModels","configureRoutes","createApp","app","express","enable","publicFolder","Path","process","cwd","static","publicPath","__dirname","bodyParser","urlencoded","extended","requestLogger","speechPassport","cookieSession","maxAge","initialize","session","hbs","expressHandlebars","defaultLayout","handlebars","Handlebars","engine","set","server","http","start","PORT","sync","listen","LoggerConfig","logLevel","configure","transports","Console","level","timestamp","colorize","prettyPrint","handleExceptions","humanReadableUnhandledException","silly","SlackConfig","slackWebHook","slackErrorChannel","slackInfoChannel","winston","add","winstonSlackWebHook","webhookUrl","iconEmoji","SlackWebHook","PassportLocalStrategy","Strategy","returnUserAndChannelInfo","userInstance","userInfo","id","userName","getChannel","shortChannelId","usernameField","passwordField","user","comparePassword","isMatch","STRING","BOOLEAN","INTEGER","TEXT","DECIMAL","define","claimSequence","decodedClaim","depth","effectiveAmount","hasSignature","hex","validAtHeight","valueVersion","claimType","certificateVersion","keyType","publicKey","freezeTableName","belongsTo","foreignKey","allowNull","order","getLongChannelIdFromShortChannelId","_this2","$like","getLongChannelIdFromChannelName","_this3","validateLongChannelId","_this4","hasOne","determineFileExtensionFromContentType","determineThumbnail","storedThumbnail","defaultThumbnail","prepareClaimData","licenseUrl","preview","metadataVersion","source","sourceType","sourceVersion","streamVersion","getShortClaimIdFromLongClaimId","raw","getLongClaimIdFromShortClaimId","getTopFreeClaimIdByClaimName","_this5","validateLongClaimId","_this6","resolveClaim","_this7","claimArray","defaultValue","trendingEligible","hasMany","getRecentClaims","limit","ipAddress","bcrypt","compare","changePassword","newPassword","genSalt","saltError","salt","hash","hashError","hook","options","userData","channelData","certificateData","newUser","newChannel","newCertificate","setChannel","setUser","lbryConfig","handleSignupRequest","handleLoginRequest","handleLogoutRequest","handleUserRequest","signup","login","logIn","logout","channelAvailability","channelClaims","channelShortId","claimAvailability","claimData","claimGet","claimLongId","claimPublish","claimResolve","claimShortId","claimList","fileAvailability","multipartMiddleware","_require3","availableName","body","claims","totalPages","determineTotalPages","paginationPage","getPageFromQuery","extractPageFromClaims","previousPage","determinePreviousPage","currentPage","nextPage","determineNextPage","totalResults","determineTotalClaims","parseInt","pageNumber","claimStartIndex","claimEndIndex","totalClaims","fullPages","Math","floor","channelShortIdRoute","claimInfo","resolveResult","fileData","_ref5$","completed","authenticateUser","_require4","_require5","_require6","files","channelPassword","_parsePublishApiReque","_parsePublishApiReque2","_ref3$","thumbnailPublishParams","lbryTx","authenticateChannelCredentials","userPassword","channelFindParams","resolvedUri","claimsList","multipart","uploadDir","handlePageRequest","handleEmbedRequest","handlePageRender","sendReactApp","sendEmbedPage","render","layout","route","serveAssetByClaim","serveAssetByIdentifierAndClaim","lbryUri","handleShowRender","serverAssetByClaim","serverAssetByIdentifierAndClaim","_lbryUri$parseIdentif","_flipClaimNameAndIdFo","_flipClaimNameAndIdFo2"],"mappings":"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","file":"index.js","sourcesContent":["module.exports =\n/******/ (function(modules) { // webpackBootstrap\n/******/ \t// The module cache\n/******/ \tvar installedModules = {};\n/******/\n/******/ \t// The require function\n/******/ \tfunction __webpack_require__(moduleId) {\n/******/\n/******/ \t\t// Check if module is in cache\n/******/ \t\tif(installedModules[moduleId]) {\n/******/ \t\t\treturn installedModules[moduleId].exports;\n/******/ \t\t}\n/******/ \t\t// Create a new module (and put it into the cache)\n/******/ \t\tvar module = installedModules[moduleId] = {\n/******/ \t\t\ti: moduleId,\n/******/ \t\t\tl: false,\n/******/ \t\t\texports: {}\n/******/ \t\t};\n/******/\n/******/ \t\t// Execute the module function\n/******/ \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n/******/\n/******/ \t\t// Flag the module as loaded\n/******/ \t\tmodule.l = true;\n/******/\n/******/ \t\t// Return the exports of the module\n/******/ \t\treturn module.exports;\n/******/ \t}\n/******/\n/******/\n/******/ \t// expose the modules object (__webpack_modules__)\n/******/ \t__webpack_require__.m = modules;\n/******/\n/******/ \t// expose the module cache\n/******/ \t__webpack_require__.c = installedModules;\n/******/\n/******/ \t// define getter function for harmony exports\n/******/ \t__webpack_require__.d = function(exports, name, getter) {\n/******/ \t\tif(!__webpack_require__.o(exports, name)) {\n/******/ \t\t\tObject.defineProperty(exports, name, {\n/******/ \t\t\t\tconfigurable: false,\n/******/ \t\t\t\tenumerable: true,\n/******/ \t\t\t\tget: getter\n/******/ \t\t\t});\n/******/ \t\t}\n/******/ \t};\n/******/\n/******/ \t// getDefaultExport function for compatibility with non-harmony modules\n/******/ \t__webpack_require__.n = function(module) {\n/******/ \t\tvar getter = module && module.__esModule ?\n/******/ \t\t\tfunction getDefault() { return module['default']; } :\n/******/ \t\t\tfunction getModuleExports() { return module; };\n/******/ \t\t__webpack_require__.d(getter, 'a', getter);\n/******/ \t\treturn getter;\n/******/ \t};\n/******/\n/******/ \t// Object.prototype.hasOwnProperty.call\n/******/ \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n/******/\n/******/ \t// __webpack_public_path__\n/******/ \t__webpack_require__.p = \"/\";\n/******/\n/******/ \t// Load entry module and return exports\n/******/ \treturn __webpack_require__(__webpack_require__.s = 26);\n/******/ })\n/************************************************************************/\n/******/ ([\n/* 0 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"winston\");\n\n/***/ }),\n/* 1 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i[\"return\"]) _i[\"return\"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError(\"Invalid attempt to destructure non-iterable instance\"); } }; }();\n\nvar logger = __webpack_require__(0);\n\nmodule.exports = {\n handleErrorResponse: function handleErrorResponse(originalUrl, ip, error, res) {\n logger.error('Error on ' + originalUrl, module.exports.useObjectPropertiesIfNoKeys(error));\n\n var _module$exports$retur = module.exports.returnErrorMessageAndStatus(error),\n _module$exports$retur2 = _slicedToArray(_module$exports$retur, 2),\n status = _module$exports$retur2[0],\n message = _module$exports$retur2[1];\n\n res.status(status).json(module.exports.createErrorResponsePayload(status, message));\n },\n returnErrorMessageAndStatus: function returnErrorMessageAndStatus(error) {\n var status = void 0,\n message = void 0;\n // check for daemon being turned off\n if (error.code === 'ECONNREFUSED') {\n status = 503;\n message = 'Connection refused. The daemon may not be running.';\n // fallback for everything else\n } else {\n status = 400;\n if (error.message) {\n message = error.message;\n } else {\n message = error;\n };\n };\n return [status, message];\n },\n useObjectPropertiesIfNoKeys: function useObjectPropertiesIfNoKeys(err) {\n if (Object.keys(err).length === 0) {\n var newErrorObject = {};\n Object.getOwnPropertyNames(err).forEach(function (key) {\n newErrorObject[key] = err[key];\n });\n return newErrorObject;\n }\n return err;\n },\n createErrorResponsePayload: function createErrorResponsePayload(status, message) {\n return {\n status: status,\n success: false,\n message: message\n };\n }\n};\n\n/***/ }),\n/* 2 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar Certificate = __webpack_require__(45);\nvar Channel = __webpack_require__(46);\nvar Claim = __webpack_require__(47);\nvar File = __webpack_require__(48);\nvar Request = __webpack_require__(49);\nvar User = __webpack_require__(50);\n\nvar Sequelize = __webpack_require__(13);\nvar logger = __webpack_require__(0);\n\nvar _require = __webpack_require__(10),\n database = _require.database,\n username = _require.username,\n password = _require.password;\n\n// set sequelize options\n\n\nvar sequelize = new Sequelize(database, username, password, {\n host: 'localhost',\n dialect: 'mysql',\n dialectOptions: { decimalNumbers: true },\n logging: false,\n pool: {\n max: 5,\n min: 0,\n idle: 10000,\n acquire: 10000\n }\n});\n\n// establish mysql connection\nsequelize.authenticate().then(function () {\n logger.info('Sequelize has established mysql connection successfully.');\n}).catch(function (err) {\n logger.error('Sequelize was unable to connect to the database:', err);\n});\n\n// manually add each model to the db object (note: make this dynamic)\nvar db = {};\ndb['Certificate'] = sequelize.import('Certificate', Certificate);\ndb['Channel'] = sequelize.import('Channel', Channel);\ndb['Claim'] = sequelize.import('Claim', Claim);\ndb['File'] = sequelize.import('File', File);\ndb['Request'] = sequelize.import('Request', Request);\ndb['User'] = sequelize.import('User', User);\n\n// run model.association for each model in the db object that has an association\nlogger.info('associating db models...');\nObject.keys(db).forEach(function (modelName) {\n if (db[modelName].associate) {\n logger.info('Associating model:', modelName);\n db[modelName].associate(db);\n }\n});\n\n// add sequelize/Sequelize to db\ndb.sequelize = sequelize;\ndb.Sequelize = Sequelize;\n// add an 'upsert' method to the db object\ndb.upsert = function (Model, values, condition, tableName) {\n return Model.findOne({\n where: condition\n }).then(function (obj) {\n if (obj) {\n // update\n logger.debug('updating record in db.' + tableName);\n return obj.update(values);\n } else {\n // insert\n logger.debug('creating record in db.' + tableName);\n return Model.create(values);\n }\n }).catch(function (error) {\n logger.error(tableName + '.upsert error', error);\n throw error;\n });\n};\n\nmodule.exports = db;\n\n/***/ }),\n/* 3 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nfunction SiteConfig() {\n var _this = this;\n\n this.analytics = {\n googleId: 'default'\n };\n this.assetDefaults = {\n description: 'An asset published on Spee.ch',\n thumbnail: 'https://spee.ch/assets/img/video_thumb_default.png',\n title: 'Spee.ch'\n };\n this.auth = {\n sessionKey: 'default'\n };\n this.customComponents = {\n components: {},\n containers: {},\n pages: {}\n };\n this.details = {\n description: 'Open-source, decentralized image and video sharing.',\n host: 'default',\n port: 3000,\n title: 'Spee.ch',\n twitter: '@spee_ch'\n };\n this.publishing = {\n additionalClaimAddresses: [],\n disabled: false,\n disabledMessage: 'Please check back soon.',\n primaryClaimAddress: 'default',\n thumbnailChannel: 'default',\n thumbnailChannelId: 'default',\n uploadDirectory: '/home/lbry/Uploads'\n };\n this.routes = {};\n this.update = function (config) {\n if (!config) {\n return console.log('No site config received.');\n }\n var analytics = config.analytics,\n assetDefaults = config.assetDefaults,\n auth = config.auth,\n customComponents = config.customComponents,\n details = config.details,\n publishing = config.publishing,\n routes = config.routes;\n\n console.log('Configuring site details...');\n _this.analytics = analytics;\n _this.assetDefaults = assetDefaults;\n _this.auth = auth;\n _this.details = details;\n _this.publishing = publishing;\n _this.customComponents = customComponents;\n _this.routes = routes;\n };\n};\n\nmodule.exports = new SiteConfig();\n\n/***/ }),\n/* 4 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar logger = __webpack_require__(0);\nvar ua = __webpack_require__(55);\n\nvar _require = __webpack_require__(3),\n googleId = _require.analytics.googleId,\n title = _require.details.title;\n\nfunction createServeEventParams(headers, ip, originalUrl) {\n return {\n eventCategory: 'client requests',\n eventAction: 'serve request',\n eventLabel: originalUrl,\n ipOverride: ip,\n userAgentOverride: headers['user-agent']\n };\n};\n\nfunction createPublishTimingEventParams(category, variable, label, startTime, endTime) {\n var duration = endTime - startTime;\n return {\n userTimingCategory: category,\n userTimingVariableName: variable,\n userTimingTime: duration,\n userTimingLabel: label\n };\n};\n\nfunction sendGoogleAnalyticsEvent(ip, params) {\n var visitorId = ip.replace(/\\./g, '-');\n var visitor = ua(googleId, visitorId, { strictCidFormat: false, https: true });\n visitor.event(params, function (err) {\n if (err) {\n logger.error('Google Analytics Event Error >>', err);\n }\n });\n};\n\nfunction sendGoogleAnalyticsTiming(visitorId, params) {\n var visitor = ua(googleId, visitorId, { strictCidFormat: false, https: true });\n visitor.timing(params, function (err) {\n if (err) {\n logger.error('Google Analytics Event Error >>', err);\n }\n logger.debug('Timing event successfully sent to google analytics');\n });\n};\n\nmodule.exports = {\n sendGAServeEvent: function sendGAServeEvent(headers, ip, originalUrl) {\n var params = createServeEventParams(headers, ip, originalUrl);\n sendGoogleAnalyticsEvent(ip, params);\n },\n sendGATimingEvent: function sendGATimingEvent(category, variable, label, startTime, endTime) {\n var params = createPublishTimingEventParams(category, variable, label, startTime, endTime);\n sendGoogleAnalyticsTiming(title, params);\n },\n chooseGaLbrynetPublishLabel: function chooseGaLbrynetPublishLabel(_ref) {\n var channelName = _ref.channel_name,\n channelId = _ref.channel_id;\n\n return channelName || channelId ? 'PUBLISH_IN_CHANNEL_CLAIM' : 'PUBLISH_ANONYMOUS_CLAIM';\n }\n};\n\n/***/ }),\n/* 5 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar axios = __webpack_require__(53);\nvar logger = __webpack_require__(0);\n\nvar _require = __webpack_require__(54),\n _require$api = _require.api,\n apiHost = _require$api.apiHost,\n apiPort = _require$api.apiPort;\n\nvar lbryApiUri = 'http://' + apiHost + ':' + apiPort;\n\nvar _require2 = __webpack_require__(4),\n chooseGaLbrynetPublishLabel = _require2.chooseGaLbrynetPublishLabel,\n sendGATimingEvent = _require2.sendGATimingEvent;\n\nvar handleLbrynetResponse = function handleLbrynetResponse(_ref, resolve, reject) {\n var data = _ref.data;\n\n logger.debug('lbry api data:', data);\n if (data.result) {\n // check for an error\n if (data.result.error) {\n logger.debug('Lbrynet api error:', data.result.error);\n reject(new Error(data.result.error));\n return;\n };\n resolve(data.result);\n return;\n }\n // fallback in case it just timed out\n reject(JSON.stringify(data));\n};\n\nmodule.exports = {\n publishClaim: function publishClaim(publishParams) {\n logger.debug('lbryApi >> Publishing claim to \"' + publishParams.name + '\"');\n var gaStartTime = Date.now();\n return new Promise(function (resolve, reject) {\n axios.post(lbryApiUri, {\n method: 'publish',\n params: publishParams\n }).then(function (response) {\n sendGATimingEvent('lbrynet', 'publish', chooseGaLbrynetPublishLabel(publishParams), gaStartTime, Date.now());\n handleLbrynetResponse(response, resolve, reject);\n }).catch(function (error) {\n reject(error);\n });\n });\n },\n getClaim: function getClaim(uri) {\n logger.debug('lbryApi >> Getting Claim for \"' + uri + '\"');\n var gaStartTime = Date.now();\n return new Promise(function (resolve, reject) {\n axios.post(lbryApiUri, {\n method: 'get',\n params: { uri: uri, timeout: 20 }\n }).then(function (response) {\n sendGATimingEvent('lbrynet', 'getClaim', 'GET', gaStartTime, Date.now());\n handleLbrynetResponse(response, resolve, reject);\n }).catch(function (error) {\n reject(error);\n });\n });\n },\n getClaimList: function getClaimList(claimName) {\n logger.debug('lbryApi >> Getting claim_list for \"' + claimName + '\"');\n var gaStartTime = Date.now();\n return new Promise(function (resolve, reject) {\n axios.post(lbryApiUri, {\n method: 'claim_list',\n params: { name: claimName }\n }).then(function (response) {\n sendGATimingEvent('lbrynet', 'getClaimList', 'CLAIM_LIST', gaStartTime, Date.now());\n handleLbrynetResponse(response, resolve, reject);\n }).catch(function (error) {\n reject(error);\n });\n });\n },\n resolveUri: function resolveUri(uri) {\n logger.debug('lbryApi >> Resolving URI for \"' + uri + '\"');\n var gaStartTime = Date.now();\n return new Promise(function (resolve, reject) {\n axios.post(lbryApiUri, {\n method: 'resolve',\n params: { uri: uri }\n }).then(function (_ref2) {\n var data = _ref2.data;\n\n sendGATimingEvent('lbrynet', 'resolveUri', 'RESOLVE', gaStartTime, Date.now());\n if (data.result[uri].error) {\n // check for errors\n reject(data.result[uri].error);\n } else {\n // if no errors, resolve\n resolve(data.result[uri]);\n }\n }).catch(function (error) {\n reject(error);\n });\n });\n },\n getDownloadDirectory: function getDownloadDirectory() {\n logger.debug('lbryApi >> Retrieving the download directory path from lbry daemon...');\n var gaStartTime = Date.now();\n return new Promise(function (resolve, reject) {\n axios.post(lbryApiUri, {\n method: 'settings_get'\n }).then(function (_ref3) {\n var data = _ref3.data;\n\n sendGATimingEvent('lbrynet', 'getDownloadDirectory', 'SETTINGS_GET', gaStartTime, Date.now());\n if (data.result) {\n resolve(data.result.download_directory);\n } else {\n return new Error('Successfully connected to lbry daemon, but unable to retrieve the download directory.');\n }\n }).catch(function (error) {\n logger.error('Lbrynet Error:', error);\n resolve('/home/lbry/Downloads/');\n });\n });\n },\n createChannel: function createChannel(name) {\n logger.debug('lbryApi >> Creating channel for ' + name + '...');\n var gaStartTime = Date.now();\n return new Promise(function (resolve, reject) {\n axios.post(lbryApiUri, {\n method: 'channel_new',\n params: {\n channel_name: name,\n amount: 0.1\n }\n }).then(function (response) {\n sendGATimingEvent('lbrynet', 'createChannel', 'CHANNEL_NEW', gaStartTime, Date.now());\n handleLbrynetResponse(response, resolve, reject);\n }).catch(function (error) {\n reject(error);\n });\n });\n }\n};\n\n/***/ }),\n/* 6 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i[\"return\"]) _i[\"return\"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError(\"Invalid attempt to destructure non-iterable instance\"); } }; }();\n\nvar db = __webpack_require__(2);\nvar logger = __webpack_require__(0);\n\nvar _require = __webpack_require__(66),\n returnPaginatedChannelClaims = _require.returnPaginatedChannelClaims;\n\nvar NO_CHANNEL = 'NO_CHANNEL';\nvar NO_CLAIM = 'NO_CLAIM';\nvar NO_FILE = 'NO_FILE';\n\nmodule.exports = {\n getClaimId: function getClaimId(channelName, channelClaimId, name, claimId) {\n if (channelName) {\n return module.exports.getClaimIdByChannel(channelName, channelClaimId, name);\n } else {\n return module.exports.getClaimIdByClaim(name, claimId);\n }\n },\n getClaimIdByClaim: function getClaimIdByClaim(claimName, claimId) {\n logger.debug('getClaimIdByClaim(' + claimName + ', ' + claimId + ')');\n return new Promise(function (resolve, reject) {\n db.Claim.getLongClaimId(claimName, claimId).then(function (longClaimId) {\n if (!longClaimId) {\n resolve(NO_CLAIM);\n }\n resolve(longClaimId);\n }).catch(function (error) {\n reject(error);\n });\n });\n },\n getClaimIdByChannel: function getClaimIdByChannel(channelName, channelClaimId, claimName) {\n logger.debug('getClaimIdByChannel(' + channelName + ', ' + channelClaimId + ', ' + claimName + ')');\n return new Promise(function (resolve, reject) {\n db.Certificate.getLongChannelId(channelName, channelClaimId) // 1. get the long channel id\n .then(function (longChannelId) {\n if (!longChannelId) {\n return [null, null];\n }\n return Promise.all([longChannelId, db.Claim.getClaimIdByLongChannelId(longChannelId, claimName)]); // 2. get the long claim id\n }).then(function (_ref) {\n var _ref2 = _slicedToArray(_ref, 2),\n longChannelId = _ref2[0],\n longClaimId = _ref2[1];\n\n if (!longChannelId) {\n return resolve(NO_CHANNEL);\n }\n if (!longClaimId) {\n return resolve(NO_CLAIM);\n }\n resolve(longClaimId);\n }).catch(function (error) {\n reject(error);\n });\n });\n },\n getChannelData: function getChannelData(channelName, channelClaimId, page) {\n return new Promise(function (resolve, reject) {\n // 1. get the long channel Id (make sure channel exists)\n db.Certificate.getLongChannelId(channelName, channelClaimId).then(function (longChannelClaimId) {\n if (!longChannelClaimId) {\n return [null, null, null];\n }\n // 2. get the short ID and all claims for that channel\n return Promise.all([longChannelClaimId, db.Certificate.getShortChannelIdFromLongChannelId(longChannelClaimId, channelName)]);\n }).then(function (_ref3) {\n var _ref4 = _slicedToArray(_ref3, 2),\n longChannelClaimId = _ref4[0],\n shortChannelClaimId = _ref4[1];\n\n if (!longChannelClaimId) {\n return resolve(NO_CHANNEL);\n }\n // 3. return all the channel information\n resolve({\n channelName: channelName,\n longChannelClaimId: longChannelClaimId,\n shortChannelClaimId: shortChannelClaimId\n });\n }).catch(function (error) {\n reject(error);\n });\n });\n },\n getChannelClaims: function getChannelClaims(channelName, channelClaimId, page) {\n return new Promise(function (resolve, reject) {\n // 1. get the long channel Id (make sure channel exists)\n db.Certificate.getLongChannelId(channelName, channelClaimId).then(function (longChannelClaimId) {\n if (!longChannelClaimId) {\n return [null, null, null];\n }\n // 2. get the short ID and all claims for that channel\n return Promise.all([longChannelClaimId, db.Claim.getAllChannelClaims(longChannelClaimId)]);\n }).then(function (_ref5) {\n var _ref6 = _slicedToArray(_ref5, 2),\n longChannelClaimId = _ref6[0],\n channelClaimsArray = _ref6[1];\n\n if (!longChannelClaimId) {\n return resolve(NO_CHANNEL);\n }\n // 3. format the data for the view, including pagination\n var paginatedChannelViewData = returnPaginatedChannelClaims(channelName, longChannelClaimId, channelClaimsArray, page);\n // 4. return all the channel information and contents\n resolve(paginatedChannelViewData);\n }).catch(function (error) {\n reject(error);\n });\n });\n },\n getLocalFileRecord: function getLocalFileRecord(claimId, name) {\n return db.File.findOne({ where: { claimId: claimId, name: name } }).then(function (file) {\n if (!file) {\n return NO_FILE;\n }\n return file.dataValues;\n });\n }\n};\n\n/***/ }),\n/* 7 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar passport = __webpack_require__(43);\nvar localLoginStrategy = __webpack_require__(44);\nvar localSignupStrategy = __webpack_require__(52);\n\nvar _require = __webpack_require__(56),\n serializeSpeechUser = _require.serializeSpeechUser,\n deserializeSpeechUser = _require.deserializeSpeechUser;\n\npassport.deserializeUser(deserializeSpeechUser);\npassport.serializeUser(serializeSpeechUser);\npassport.use('local-login', localLoginStrategy);\npassport.use('local-signup', localSignupStrategy);\n\nmodule.exports = passport;\n\n/***/ }),\n/* 8 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i[\"return\"]) _i[\"return\"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError(\"Invalid attempt to destructure non-iterable instance\"); } }; }();\n\nfunction _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }\n\nvar logger = __webpack_require__(0);\nvar db = __webpack_require__(2);\nvar lbryApi = __webpack_require__(5);\nvar publishHelpers = __webpack_require__(9);\n\nvar _require = __webpack_require__(3),\n _require$publishing = _require.publishing,\n primaryClaimAddress = _require$publishing.primaryClaimAddress,\n additionalClaimAddresses = _require$publishing.additionalClaimAddresses;\n\nvar Sequelize = __webpack_require__(13);\nvar Op = Sequelize.Op;\n\nmodule.exports = {\n publish: function publish(publishParams, fileName, fileType) {\n return new Promise(function (resolve, reject) {\n var publishResults = void 0,\n certificateId = void 0,\n channelName = void 0;\n // publish the file\n return lbryApi.publishClaim(publishParams).then(function (tx) {\n logger.info('Successfully published ' + publishParams.name + ' ' + fileName, tx);\n publishResults = tx;\n // get the channel information\n if (publishParams.channel_name) {\n logger.debug('this claim was published in channel: ' + publishParams.channel_name);\n return db.Channel.findOne({\n where: {\n channelName: publishParams.channel_name\n }\n });\n } else {\n logger.debug('this claim was not published in a channel');\n return null;\n }\n }).then(function (channel) {\n // set channel information\n certificateId = null;\n channelName = null;\n if (channel) {\n certificateId = channel.channelClaimId;\n channelName = channel.channelName;\n }\n logger.debug('certificateId: ' + certificateId);\n }).then(function () {\n // create the File record\n var fileRecord = {\n name: publishParams.name,\n claimId: publishResults.claim_id,\n title: publishParams.metadata.title,\n description: publishParams.metadata.description,\n address: publishParams.claim_address,\n outpoint: publishResults.txid + ':' + publishResults.nout,\n height: 0,\n fileName: fileName,\n filePath: publishParams.file_path,\n fileType: fileType,\n nsfw: publishParams.metadata.nsfw\n };\n // create the Claim record\n var claimRecord = {\n name: publishParams.name,\n claimId: publishResults.claim_id,\n title: publishParams.metadata.title,\n description: publishParams.metadata.description,\n address: publishParams.claim_address,\n thumbnail: publishParams.metadata.thumbnail,\n outpoint: publishResults.txid + ':' + publishResults.nout,\n height: 0,\n contentType: fileType,\n nsfw: publishParams.metadata.nsfw,\n amount: publishParams.bid,\n certificateId: certificateId,\n channelName: channelName\n };\n // upsert criteria\n var upsertCriteria = {\n name: publishParams.name,\n claimId: publishResults.claim_id\n };\n // upsert the records\n return Promise.all([db.upsert(db.File, fileRecord, upsertCriteria, 'File'), db.upsert(db.Claim, claimRecord, upsertCriteria, 'Claim')]);\n }).then(function (_ref) {\n var _ref2 = _slicedToArray(_ref, 2),\n file = _ref2[0],\n claim = _ref2[1];\n\n logger.debug('File and Claim records successfully created');\n return Promise.all([file.setClaim(claim), claim.setFile(file)]);\n }).then(function () {\n logger.debug('File and Claim records successfully associated');\n resolve(publishResults); // resolve the promise with the result from lbryApi.publishClaim;\n }).catch(function (error) {\n logger.error('PUBLISH ERROR', error);\n publishHelpers.deleteTemporaryFile(publishParams.file_path); // delete the local file\n reject(error);\n });\n });\n },\n claimNameIsAvailable: function claimNameIsAvailable(name) {\n var claimAddresses = additionalClaimAddresses || [];\n claimAddresses.push(primaryClaimAddress);\n // find any records where the name is used\n return db.Claim.findAll({\n attributes: ['address'],\n where: {\n name: name,\n address: _defineProperty({}, Op.or, claimAddresses)\n }\n }).then(function (result) {\n if (result.length >= 1) {\n throw new Error('That claim is already in use');\n };\n return name;\n }).catch(function (error) {\n throw error;\n });\n },\n checkChannelAvailability: function checkChannelAvailability(name) {\n return db.Channel.findAll({\n where: { channelName: name }\n }).then(function (result) {\n if (result.length >= 1) {\n throw new Error('That channel has already been claimed');\n }\n return name;\n }).catch(function (error) {\n throw error;\n });\n }\n};\n\n/***/ }),\n/* 9 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar logger = __webpack_require__(0);\nvar fs = __webpack_require__(64);\n\nvar _require = __webpack_require__(3),\n details = _require.details,\n publishing = _require.publishing;\n\nmodule.exports = {\n parsePublishApiRequestBody: function parsePublishApiRequestBody(_ref) {\n var name = _ref.name,\n nsfw = _ref.nsfw,\n license = _ref.license,\n title = _ref.title,\n description = _ref.description,\n thumbnail = _ref.thumbnail;\n\n // validate name\n if (!name) {\n throw new Error('no name field found in request');\n }\n var invalidNameCharacters = /[^A-Za-z0-9,-]/.exec(name);\n if (invalidNameCharacters) {\n throw new Error('The claim name you provided is not allowed. Only the following characters are allowed: A-Z, a-z, 0-9, and \"-\"');\n }\n // optional parameters\n nsfw = nsfw === 'true';\n license = license || null;\n title = title || null;\n description = description || null;\n thumbnail = thumbnail || null;\n // return results\n return {\n name: name,\n nsfw: nsfw,\n license: license,\n title: title,\n description: description,\n thumbnail: thumbnail\n };\n },\n parsePublishApiRequestFiles: function parsePublishApiRequestFiles(_ref2) {\n var file = _ref2.file,\n thumbnail = _ref2.thumbnail;\n\n // make sure a file was provided\n if (!file) {\n throw new Error('no file with key of [file] found in request');\n }\n if (!file.path) {\n throw new Error('no file path found');\n }\n if (!file.type) {\n throw new Error('no file type found');\n }\n if (!file.size) {\n throw new Error('no file type found');\n }\n // validate the file name\n if (/'/.test(file.name)) {\n throw new Error('apostrophes are not allowed in the file name');\n }\n // validate the file\n module.exports.validateFileTypeAndSize(file);\n // return results\n return {\n fileName: file.name,\n filePath: file.path,\n fileType: file.type,\n thumbnailFileName: thumbnail ? thumbnail.name : null,\n thumbnailFilePath: thumbnail ? thumbnail.path : null,\n thumbnailFileType: thumbnail ? thumbnail.type : null\n };\n },\n validateFileTypeAndSize: function validateFileTypeAndSize(file) {\n // check file type and size\n switch (file.type) {\n case 'image/jpeg':\n case 'image/jpg':\n case 'image/png':\n if (file.size > 10000000) {\n logger.debug('publish > file validation > .jpeg/.jpg/.png was too big');\n throw new Error('Sorry, images are limited to 10 megabytes.');\n }\n break;\n case 'image/gif':\n if (file.size > 50000000) {\n logger.debug('publish > file validation > .gif was too big');\n throw new Error('Sorry, .gifs are limited to 50 megabytes.');\n }\n break;\n case 'video/mp4':\n if (file.size > 50000000) {\n logger.debug('publish > file validation > .mp4 was too big');\n throw new Error('Sorry, videos are limited to 50 megabytes.');\n }\n break;\n default:\n logger.debug('publish > file validation > unrecognized file type');\n throw new Error('The ' + file.type + ' content type is not supported. Only, .jpeg, .png, .gif, and .mp4 files are currently supported.');\n }\n return file;\n },\n createBasicPublishParams: function createBasicPublishParams(filePath, name, title, description, license, nsfw, thumbnail) {\n logger.debug('Creating Publish Parameters');\n // provide defaults for title\n if (title === null || title.trim() === '') {\n title = name;\n }\n // provide default for description\n if (description === null || description.trim() === '') {\n description = '';\n }\n // provide default for license\n if (license === null || license.trim() === '') {\n license = ' '; // default to empty string\n }\n // create the publish params\n var publishParams = {\n name: name,\n file_path: filePath,\n bid: 0.01,\n metadata: {\n description: description,\n title: title,\n author: details.title,\n language: 'en',\n license: license,\n nsfw: nsfw\n },\n claim_address: publishing.primaryClaimAddress\n };\n // add thumbnail to channel if video\n if (thumbnail) {\n publishParams['metadata']['thumbnail'] = thumbnail;\n }\n return publishParams;\n },\n createThumbnailPublishParams: function createThumbnailPublishParams(thumbnailFilePath, claimName, license, nsfw) {\n if (!thumbnailFilePath) {\n return;\n }\n logger.debug('Creating Thumbnail Publish Parameters');\n // create the publish params\n return {\n name: claimName + '-thumb',\n file_path: thumbnailFilePath,\n bid: 0.01,\n metadata: {\n title: claimName + ' thumbnail',\n description: 'a thumbnail for ' + claimName,\n author: details.title,\n language: 'en',\n license: license,\n nsfw: nsfw\n },\n claim_address: publishing.primaryClaimAddress,\n channel_name: publishing.thumbnailChannel,\n channel_id: publishing.thumbnailChannelId\n };\n },\n deleteTemporaryFile: function deleteTemporaryFile(filePath) {\n fs.unlink(filePath, function (err) {\n if (err) {\n logger.error('error deleting temporary file ' + filePath);\n throw err;\n }\n logger.debug('successfully deleted ' + filePath);\n });\n },\n addGetResultsToFileData: function addGetResultsToFileData(fileInfo, getResult) {\n fileInfo.fileName = getResult.file_name;\n fileInfo.filePath = getResult.download_path;\n return fileInfo;\n },\n createFileData: function createFileData(_ref3) {\n var name = _ref3.name,\n claimId = _ref3.claimId,\n outpoint = _ref3.outpoint,\n height = _ref3.height,\n address = _ref3.address,\n nsfw = _ref3.nsfw,\n contentType = _ref3.contentType;\n\n return {\n name: name,\n claimId: claimId,\n outpoint: outpoint,\n height: height,\n address: address,\n fileName: '',\n filePath: '',\n fileType: contentType,\n nsfw: nsfw\n };\n }\n};\n\n/***/ }),\n/* 10 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar logger = __webpack_require__(0);\n\nfunction mysql() {\n var _this = this;\n\n this.database = 'default';\n this.username = 'default';\n this.password = 'default';\n this.update = function (config) {\n if (!config) {\n return logger.warn('No MySQL config received.');\n }\n // configure credentials\n logger.info('configuring mysql...');\n var database = config.database,\n username = config.username,\n password = config.password;\n\n _this.database = database;\n _this.username = username;\n _this.password = password;\n };\n};\n\nmodule.exports = new mysql();\n\n/***/ }),\n/* 11 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"passport-local\");\n\n/***/ }),\n/* 12 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nmodule.exports = {\n returnShortId: function returnShortId(claimsArray, longId) {\n var claimIndex = void 0;\n var shortId = longId.substring(0, 1); // default short id is the first letter\n var shortIdLength = 0;\n // find the index of this claim id\n claimIndex = claimsArray.findIndex(function (element) {\n return element.claimId === longId;\n });\n if (claimIndex < 0) {\n throw new Error('claim id not found in claims list');\n }\n // get an array of all claims with lower height\n var possibleMatches = claimsArray.slice(0, claimIndex);\n // remove certificates with the same prefixes until none are left.\n while (possibleMatches.length > 0) {\n shortIdLength += 1;\n shortId = longId.substring(0, shortIdLength);\n possibleMatches = possibleMatches.filter(function (element) {\n return element.claimId && element.claimId.substring(0, shortIdLength) === shortId;\n });\n }\n return shortId;\n }\n};\n\n/***/ }),\n/* 13 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"sequelize\");\n\n/***/ }),\n/* 14 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _react = __webpack_require__(15);\n\nvar _react2 = _interopRequireDefault(_react);\n\nvar _server = __webpack_require__(16);\n\nvar _redux = __webpack_require__(17);\n\nvar _reactRedux = __webpack_require__(18);\n\nvar _reactRouterDom = __webpack_require__(19);\n\nvar _spee = __webpack_require__(20);\n\nvar _renderFullPage = __webpack_require__(21);\n\nvar _renderFullPage2 = _interopRequireDefault(_renderFullPage);\n\nvar _reactHelmet = __webpack_require__(22);\n\nvar _reactHelmet2 = _interopRequireDefault(_reactHelmet);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\nvar siteConfig = __webpack_require__(3);\n\nmodule.exports = function (req, res) {\n var context = {};\n\n // customize the reducer by passing in intial state configs\n var customizedReducers = (0, _spee.Reducers)(siteConfig);\n\n // create a new Redux store instance\n var store = (0, _redux.createStore)(customizedReducers);\n\n // render component to a string\n var html = (0, _server.renderToString)(_react2.default.createElement(\n _reactRedux.Provider,\n { store: store },\n _react2.default.createElement(\n _reactRouterDom.StaticRouter,\n { location: req.url, context: context },\n _react2.default.createElement(\n _spee.GAListener,\n null,\n _react2.default.createElement(_spee.App, null)\n )\n )\n ));\n\n // get head tags from helmet\n var helmet = _reactHelmet2.default.renderStatic();\n\n // check for a redirect\n if (context.url) {\n // Somewhere a `` was rendered\n return res.redirect(301, context.url);\n } else {}\n // we're good, send the response\n\n\n // get the initial state from our Redux store\n var preloadedState = store.getState();\n\n // send the rendered page back to the client\n res.send((0, _renderFullPage2.default)(helmet, html, preloadedState));\n\n console.log('hello from spee.ch handlePageRender.jsx');\n};\n\n/***/ }),\n/* 15 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"react\");\n\n/***/ }),\n/* 16 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"react-dom/server\");\n\n/***/ }),\n/* 17 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"redux\");\n\n/***/ }),\n/* 18 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"react-redux\");\n\n/***/ }),\n/* 19 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"react-router-dom\");\n\n/***/ }),\n/* 20 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"spee.ch-components\");\n\n/***/ }),\n/* 21 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nmodule.exports = function (helmet, html, preloadedState) {\n // take the html and preloadedState and return the full page\n return '\\n \\n \\n \\n \\n \\n \\n \\n ' + helmet.title.toString() + '\\n ' + helmet.meta.toString() + '\\n ' + helmet.link.toString() + '\\n \\n \\n \\n \\n \\n \\n \\n \\n
\\n
' + html + '
\\n
\\n \\n \\n \\n \\n ';\n};\n\n/***/ }),\n/* 22 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"react-helmet\");\n\n/***/ }),\n/* 23 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar logger = __webpack_require__(0);\n\nvar _require = __webpack_require__(6),\n getClaimId = _require.getClaimId,\n getLocalFileRecord = _require.getLocalFileRecord;\n\nvar _require2 = __webpack_require__(1),\n handleErrorResponse = _require2.handleErrorResponse;\n\nvar SERVE = 'SERVE';\nvar SHOW = 'SHOW';\nvar NO_FILE = 'NO_FILE';\nvar NO_CHANNEL = 'NO_CHANNEL';\nvar NO_CLAIM = 'NO_CLAIM';\n\nfunction clientAcceptsHtml(_ref) {\n var accept = _ref.accept;\n\n return accept && accept.match(/text\\/html/);\n};\n\nfunction requestIsFromBrowser(headers) {\n return headers['user-agent'] && headers['user-agent'].match(/Mozilla/);\n};\n\nfunction clientWantsAsset(_ref2) {\n var accept = _ref2.accept,\n range = _ref2.range;\n\n var imageIsWanted = accept && accept.match(/image\\/.*/) && !accept.match(/text\\/html/) && !accept.match(/text\\/\\*/);\n var videoIsWanted = accept && range;\n return imageIsWanted || videoIsWanted;\n};\n\nfunction isValidClaimId(claimId) {\n return claimId.length === 40 && !/[^A-Za-z0-9]/g.test(claimId);\n};\n\nfunction isValidShortId(claimId) {\n return claimId.length === 1; // it should really evaluate the short url itself\n};\n\nfunction isValidShortIdOrClaimId(input) {\n return isValidClaimId(input) || isValidShortId(input);\n};\n\nfunction serveAssetToClient(claimId, name, res) {\n return getLocalFileRecord(claimId, name).then(function (fileRecord) {\n // check that a local record was found\n if (fileRecord === NO_FILE) {\n return res.status(307).redirect('/api/claim/get/' + name + '/' + claimId);\n }\n // serve the file\n var filePath = fileRecord.filePath,\n fileType = fileRecord.fileType;\n\n logger.verbose('serving file: ' + filePath);\n var sendFileOptions = {\n headers: {\n 'X-Content-Type-Options': 'nosniff',\n 'Content-Type': fileType || 'image/jpeg'\n }\n };\n res.status(200).sendFile(filePath, sendFileOptions);\n }).catch(function (error) {\n throw error;\n });\n};\n\nmodule.exports = {\n getClaimIdAndServeAsset: function getClaimIdAndServeAsset(channelName, channelClaimId, claimName, claimId, originalUrl, ip, res) {\n // get the claim Id and then serve the asset\n getClaimId(channelName, channelClaimId, claimName, claimId).then(function (fullClaimId) {\n if (fullClaimId === NO_CLAIM) {\n return res.status(404).json({ success: false, message: 'no claim id could be found' });\n } else if (fullClaimId === NO_CHANNEL) {\n return res.status(404).json({ success: false, message: 'no channel id could be found' });\n }\n serveAssetToClient(fullClaimId, claimName, res);\n // postToStats(responseType, originalUrl, ip, claimName, fullClaimId, 'success');\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n // postToStats(responseType, originalUrl, ip, claimName, fullClaimId, 'fail');\n });\n },\n determineResponseType: function determineResponseType(hasFileExtension, headers) {\n var responseType = void 0;\n if (hasFileExtension) {\n responseType = SERVE; // assume a serve request if file extension is present\n if (clientAcceptsHtml(headers)) {\n // if the request comes from a browser, change it to a show request\n responseType = SHOW;\n }\n } else {\n responseType = SHOW;\n if (clientWantsAsset(headers) && requestIsFromBrowser(headers)) {\n // this is in case someone embeds a show url\n logger.debug('Show request came from browser but wants an image/video. Changing response to serve...');\n responseType = SERVE;\n }\n }\n return responseType;\n },\n flipClaimNameAndIdForBackwardsCompatibility: function flipClaimNameAndIdForBackwardsCompatibility(identifier, name) {\n // this is a patch for backwards compatability with '/name/claim_id' url format\n if (isValidShortIdOrClaimId(name) && !isValidShortIdOrClaimId(identifier)) {\n var tempName = name;\n name = identifier;\n identifier = tempName;\n }\n return [identifier, name];\n },\n logRequestData: function logRequestData(responseType, claimName, channelName, claimId) {\n logger.debug('responseType ===', responseType);\n logger.debug('claim name === ', claimName);\n logger.debug('channel name ===', channelName);\n logger.debug('claim id ===', claimId);\n }\n};\n\n/***/ }),\n/* 24 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i[\"return\"]) _i[\"return\"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError(\"Invalid attempt to destructure non-iterable instance\"); } }; }();\n\nvar logger = __webpack_require__(0);\n\nmodule.exports = {\n REGEXP_INVALID_CLAIM: /[^A-Za-z0-9-]/g,\n REGEXP_INVALID_CHANNEL: /[^A-Za-z0-9-@]/g,\n REGEXP_ADDRESS: /^b(?=[^0OIl]{32,33})[0-9A-Za-z]{32,33}$/,\n CHANNEL_CHAR: '@',\n parseIdentifier: function parseIdentifier(identifier) {\n logger.debug('parsing identifier:', identifier);\n var componentsRegex = new RegExp('([^:$#/]*)' + // value (stops at the first separator or end)\n '([:$#]?)([^/]*)' // modifier separator, modifier (stops at the first path separator or end)\n );\n\n var _componentsRegex$exec = componentsRegex.exec(identifier).map(function (match) {\n return match || null;\n }),\n _componentsRegex$exec2 = _slicedToArray(_componentsRegex$exec, 4),\n proto = _componentsRegex$exec2[0],\n value = _componentsRegex$exec2[1],\n modifierSeperator = _componentsRegex$exec2[2],\n modifier = _componentsRegex$exec2[3];\n\n logger.debug(proto + ', ' + value + ', ' + modifierSeperator + ', ' + modifier);\n\n // Validate and process name\n if (!value) {\n throw new Error('Check your url. No channel name provided before \"' + modifierSeperator + '\"');\n }\n var isChannel = value.startsWith(module.exports.CHANNEL_CHAR);\n var channelName = isChannel ? value : null;\n var claimId = void 0;\n if (isChannel) {\n if (!channelName) {\n throw new Error('No channel name after @.');\n }\n var nameBadChars = channelName.match(module.exports.REGEXP_INVALID_CHANNEL);\n if (nameBadChars) {\n throw new Error('Invalid characters in channel name: ' + nameBadChars.join(', ') + '.');\n }\n } else {\n claimId = value;\n }\n\n // Validate and process modifier\n var channelClaimId = void 0;\n if (modifierSeperator) {\n if (!modifier) {\n throw new Error('No modifier provided after separator \"' + modifierSeperator + '\"');\n }\n\n if (modifierSeperator === ':') {\n channelClaimId = modifier;\n } else {\n throw new Error('The \"' + modifierSeperator + '\" modifier is not currently supported');\n }\n }\n return {\n isChannel: isChannel,\n channelName: channelName,\n channelClaimId: channelClaimId,\n claimId: claimId\n };\n },\n parseClaim: function parseClaim(claim) {\n logger.debug('parsing name:', claim);\n var componentsRegex = new RegExp('([^:$#/.]*)' + // name (stops at the first modifier)\n '([:$#.]?)([^/]*)' // modifier separator, modifier (stops at the first path separator or end)\n );\n\n var _componentsRegex$exec3 = componentsRegex.exec(claim).map(function (match) {\n return match || null;\n }),\n _componentsRegex$exec4 = _slicedToArray(_componentsRegex$exec3, 4),\n proto = _componentsRegex$exec4[0],\n claimName = _componentsRegex$exec4[1],\n modifierSeperator = _componentsRegex$exec4[2],\n modifier = _componentsRegex$exec4[3];\n\n logger.debug(proto + ', ' + claimName + ', ' + modifierSeperator + ', ' + modifier);\n\n // Validate and process name\n if (!claimName) {\n throw new Error('No claim name provided before .');\n }\n var nameBadChars = claimName.match(module.exports.REGEXP_INVALID_CLAIM);\n if (nameBadChars) {\n throw new Error('Invalid characters in claim name: ' + nameBadChars.join(', ') + '.');\n }\n // Validate and process modifier\n if (modifierSeperator) {\n if (!modifier) {\n throw new Error('No file extension provided after separator ' + modifierSeperator + '.');\n }\n if (modifierSeperator !== '.') {\n throw new Error('The ' + modifierSeperator + ' modifier is not supported in the claim name');\n }\n }\n // return results\n return {\n claimName: claimName\n };\n },\n parseModifier: function parseModifier(claim) {\n logger.debug('parsing modifier:', claim);\n var componentsRegex = new RegExp('([^:$#/.]*)' + // name (stops at the first modifier)\n '([:$#.]?)([^/]*)' // modifier separator, modifier (stops at the first path separator or end)\n );\n\n var _componentsRegex$exec5 = componentsRegex.exec(claim).map(function (match) {\n return match || null;\n }),\n _componentsRegex$exec6 = _slicedToArray(_componentsRegex$exec5, 4),\n proto = _componentsRegex$exec6[0],\n claimName = _componentsRegex$exec6[1],\n modifierSeperator = _componentsRegex$exec6[2],\n modifier = _componentsRegex$exec6[3];\n\n logger.debug(proto + ', ' + claimName + ', ' + modifierSeperator + ', ' + modifier);\n // Validate and process modifier\n var hasFileExtension = false;\n if (modifierSeperator) {\n hasFileExtension = true;\n }\n return {\n hasFileExtension: hasFileExtension\n };\n }\n};\n\n/***/ }),\n/* 25 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _react = __webpack_require__(15);\n\nvar _react2 = _interopRequireDefault(_react);\n\nvar _server = __webpack_require__(16);\n\nvar _redux = __webpack_require__(17);\n\nvar _reactRedux = __webpack_require__(18);\n\nvar _reactRouterDom = __webpack_require__(19);\n\nvar _renderFullPage = __webpack_require__(21);\n\nvar _renderFullPage2 = _interopRequireDefault(_renderFullPage);\n\nvar _reduxSaga = __webpack_require__(87);\n\nvar _reduxSaga2 = _interopRequireDefault(_reduxSaga);\n\nvar _effects = __webpack_require__(88);\n\nvar _spee = __webpack_require__(20);\n\nvar _reactHelmet = __webpack_require__(22);\n\nvar _reactHelmet2 = _interopRequireDefault(_reactHelmet);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\n// configure the reducers by passing initial state configs\nvar siteConfig = __webpack_require__(3);\nvar customizedReducers = (0, _spee.Reducers)(siteConfig);\n\nvar returnSagaWithParams = function returnSagaWithParams(saga, params) {\n return (/*#__PURE__*/regeneratorRuntime.mark(function _callee() {\n return regeneratorRuntime.wrap(function _callee$(_context) {\n while (1) {\n switch (_context.prev = _context.next) {\n case 0:\n _context.next = 2;\n return (0, _effects.call)(saga, params);\n\n case 2:\n case 'end':\n return _context.stop();\n }\n }\n }, _callee, this);\n })\n );\n};\n\nmodule.exports = function (req, res) {\n var context = {};\n\n // create and apply middleware\n var sagaMiddleware = (0, _reduxSaga2.default)();\n var middleware = (0, _redux.applyMiddleware)(sagaMiddleware);\n\n // create a new Redux store instance\n var store = (0, _redux.createStore)(customizedReducers, middleware);\n\n // create saga\n var action = _spee.Actions.onHandleShowPageUri(req.params);\n var saga = returnSagaWithParams(_spee.Sagas.handleShowPageUri, action);\n\n // run the saga middleware\n sagaMiddleware.run(saga).done.then(function () {\n // render component to a string\n var html = (0, _server.renderToString)(_react2.default.createElement(\n _reactRedux.Provider,\n { store: store },\n _react2.default.createElement(\n _reactRouterDom.StaticRouter,\n { location: req.url, context: context },\n _react2.default.createElement(\n _spee.GAListener,\n null,\n _react2.default.createElement(_spee.App, null)\n )\n )\n ));\n\n // get head tags from helmet\n var helmet = _reactHelmet2.default.renderStatic();\n\n // check for a redirect\n if (context.url) {\n return res.redirect(301, context.url);\n }\n\n // get the initial state from our Redux store\n var preloadedState = store.getState();\n\n // send the rendered page back to the client\n res.send((0, _renderFullPage2.default)(helmet, html, preloadedState));\n });\n\n console.log('hello from spee.ch handleShowRender.jsx');\n};\n\n/***/ }),\n/* 26 */\n/***/ (function(module, exports, __webpack_require__) {\n\n__webpack_require__(27);\n__webpack_require__(28);\nmodule.exports = __webpack_require__(29);\n\n\n/***/ }),\n/* 27 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"babel-polyfill\");\n\n/***/ }),\n/* 28 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"whatwg-fetch\");\n\n/***/ }),\n/* 29 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar Server = __webpack_require__(30);\n\nvar _exports = {\n Server: Server\n};\n\nmodule.exports = _exports;\n\n/***/ }),\n/* 30 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\n// app dependencies\nvar express = __webpack_require__(31);\nvar bodyParser = __webpack_require__(32);\nvar expressHandlebars = __webpack_require__(33);\nvar Handlebars = __webpack_require__(34);\nvar helmet = __webpack_require__(35);\nvar cookieSession = __webpack_require__(36);\nvar http = __webpack_require__(37);\nvar logger = __webpack_require__(0);\nvar requestLogger = __webpack_require__(38);\nvar Path = __webpack_require__(39);\nvar loggerConfig = __webpack_require__(40);\nvar mysqlConfig = __webpack_require__(10);\nvar siteConfig = __webpack_require__(3);\nvar slackConfig = __webpack_require__(41);\n\nfunction Server() {\n var _this = this;\n\n this.configureLogger = function (userConfig) {\n loggerConfig.update(userConfig);\n };\n this.configureMysql = function (userConfig) {\n mysqlConfig.update(userConfig);\n };\n this.configureSiteDetails = function (userConfig) {\n siteConfig.update(userConfig);\n };\n this.configureSlack = function (userConfig) {\n slackConfig.update(userConfig);\n };\n this.configureClientBundle = function () {\n logger.debug('configure the client here by passing in the bundle and configuring it, or better yet: taking in the components to use dynamically from here.');\n };\n this.configureModels = function () {\n logger.debug('here is where you could add/overwrite the default models');\n };\n this.configureRoutes = function () {\n logger.debug('here is where you could add/overwrite the default routes');\n };\n this.createApp = function () {\n // create an Express application\n var app = express();\n\n // trust the proxy to get ip address for us\n app.enable('trust proxy');\n\n /* add middleware */\n // set HTTP headers to protect against well-known web vulnerabilties\n app.use(helmet());\n // 'express.static' to serve static files from public directory\n if (siteConfig.routes.publicFolder) {\n // take in a different public folder, so it can serve it's own bundle if needed\n var publicFolder = Path.resolve(process.cwd(), siteConfig.routes.publicFolder);\n app.use('/static', express.static(publicFolder));\n logger.info('serving static files from custom path:', publicFolder);\n } else {\n var publicPath = Path.resolve(__dirname, 'public');\n app.use('/static', express.static(publicPath));\n logger.info('serving static files from default path:', publicPath);\n };\n // 'body parser' for parsing application/json\n app.use(bodyParser.json());\n // 'body parser' for parsing application/x-www-form-urlencoded\n app.use(bodyParser.urlencoded({ extended: true }));\n\n // add custom middleware (note: build out to accept dynamically use what is in server/middleware/\n app.use(requestLogger);\n\n // configure passport\n var speechPassport = __webpack_require__(7);\n // initialize passport\n var sessionKey = siteConfig.auth.sessionKey;\n app.use(cookieSession({\n name: 'session',\n keys: [sessionKey],\n maxAge: 24 * 60 * 60 * 1000 // i.e. 24 hours\n }));\n app.use(speechPassport.initialize());\n app.use(speechPassport.session());\n\n // configure handlebars & register it with express app\n var hbs = expressHandlebars.create({\n defaultLayout: 'embed',\n handlebars: Handlebars\n });\n app.engine('handlebars', hbs.engine);\n app.set('view engine', 'handlebars');\n\n // set the routes on the app\n __webpack_require__(57)(app);\n __webpack_require__(62)(app);\n __webpack_require__(81)(app);\n __webpack_require__(85)(app);\n __webpack_require__(90)(app);\n\n _this.app = app;\n };\n this.initialize = function () {\n _this.createApp();\n _this.server = http.Server(_this.app);\n };\n this.start = function () {\n var db = __webpack_require__(2);\n var PORT = siteConfig.details.port;\n // sync sequelize\n db.sequelize.sync()\n // start the server\n .then(function () {\n _this.server.listen(PORT, function () {\n logger.info('Server is listening on PORT ' + PORT);\n });\n }).catch(function (error) {\n logger.error('Startup Error:', error);\n });\n };\n};\n\nmodule.exports = Server;\n\n/***/ }),\n/* 31 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"express\");\n\n/***/ }),\n/* 32 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"body-parser\");\n\n/***/ }),\n/* 33 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"express-handlebars\");\n\n/***/ }),\n/* 34 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"handlebars\");\n\n/***/ }),\n/* 35 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"helmet\");\n\n/***/ }),\n/* 36 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"cookie-session\");\n\n/***/ }),\n/* 37 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"http\");\n\n/***/ }),\n/* 38 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar logger = __webpack_require__(0);\n\nvar requestLogger = function requestLogger(req, res, next) {\n // custom logging middleware to log all incoming http requests\n logger.verbose('Request on ' + req.originalUrl + ' from ' + req.ip);\n next();\n};\n\nmodule.exports = requestLogger;\n\n/***/ }),\n/* 39 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"path\");\n\n/***/ }),\n/* 40 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar logger = __webpack_require__(0);\n\nfunction LoggerConfig() {\n var _this = this;\n\n this.logLevel = 'debug';\n this.update = function (config) {\n if (!config) {\n return logger.warn('No logger config received.');\n }\n logger.info('configuring winston logger...');\n // update values with local config params\n var logLevel = config.logLevel;\n\n _this.logLevel = logLevel;\n // configure the winston logger\n logger.configure({\n transports: [new logger.transports.Console({\n level: _this.logLevel,\n timestamp: false,\n colorize: true,\n prettyPrint: true,\n handleExceptions: true,\n humanReadableUnhandledException: true\n })]\n });\n // test all the log levels\n logger.info('testing winston log levels...');\n logger.error('Level 0');\n logger.warn('Level 1');\n logger.info('Level 2');\n logger.verbose('Level 3');\n logger.debug('Level 4');\n logger.silly('Level 5');\n };\n};\n\nmodule.exports = new LoggerConfig();\n\n/***/ }),\n/* 41 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar winstonSlackWebHook = __webpack_require__(42).SlackWebHook;\nvar winston = __webpack_require__(0);\n\nfunction SlackConfig() {\n var _this = this;\n\n this.slackWebHook = 'default';\n this.slackErrorChannel = 'default';\n this.slackInfoChannel = 'default';\n this.update = function (config) {\n if (!config) {\n return winston.warn('No slack config received');\n }\n // update variables\n winston.info('configuring slack logger...');\n var slackWebHook = config.slackWebHook,\n slackErrorChannel = config.slackErrorChannel,\n slackInfoChannel = config.slackInfoChannel;\n\n _this.slackWebHook = slackWebHook;\n _this.slackErrorChannel = slackErrorChannel;\n _this.slackInfoChannel = slackInfoChannel;\n // update slack webhook settings\n if (_this.slackWebHook) {\n // add a transport for errors to slack\n if (_this.slackErrorChannel) {\n winston.add(winstonSlackWebHook, {\n name: 'slack-errors-transport',\n level: 'warn',\n webhookUrl: _this.slackWebHook,\n channel: _this.slackErrorChannel,\n username: 'spee.ch',\n iconEmoji: ':face_with_head_bandage:'\n });\n };\n if (slackInfoChannel) {\n winston.add(winstonSlackWebHook, {\n name: 'slack-info-transport',\n level: 'info',\n webhookUrl: _this.slackWebHook,\n channel: _this.slackInfoChannel,\n username: 'spee.ch',\n iconEmoji: ':nerd_face:'\n });\n };\n // send test messages\n winston.info('testing slack logger...');\n winston.error('Slack \"error\" logging is online.');\n winston.info('Slack \"info\" logging is online.');\n } else {\n winston.warn('Slack logging is not enabled because no slackWebHook config var provided.');\n }\n };\n};\n\nmodule.exports = new SlackConfig();\n\n/***/ }),\n/* 42 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"winston-slack-webhook\");\n\n/***/ }),\n/* 43 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"passport\");\n\n/***/ }),\n/* 44 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar PassportLocalStrategy = __webpack_require__(11).Strategy;\nvar logger = __webpack_require__(0);\nvar db = __webpack_require__(2);\n\nvar returnUserAndChannelInfo = function returnUserAndChannelInfo(userInstance) {\n return new Promise(function (resolve, reject) {\n var userInfo = {};\n userInfo['id'] = userInstance.id;\n userInfo['userName'] = userInstance.userName;\n userInstance.getChannel().then(function (_ref) {\n var channelName = _ref.channelName,\n channelClaimId = _ref.channelClaimId;\n\n userInfo['channelName'] = channelName;\n userInfo['channelClaimId'] = channelClaimId;\n return db.Certificate.getShortChannelIdFromLongChannelId(channelClaimId, channelName);\n }).then(function (shortChannelId) {\n userInfo['shortChannelId'] = shortChannelId;\n resolve(userInfo);\n }).catch(function (error) {\n reject(error);\n });\n });\n};\n\nmodule.exports = new PassportLocalStrategy({\n usernameField: 'username',\n passwordField: 'password'\n}, function (username, password, done) {\n return db.User.findOne({\n where: { userName: username }\n }).then(function (user) {\n if (!user) {\n logger.debug('no user found');\n return done(null, false, { message: 'Incorrect username or password' });\n }\n return user.comparePassword(password).then(function (isMatch) {\n if (!isMatch) {\n logger.debug('incorrect password');\n return done(null, false, { message: 'Incorrect username or password' });\n }\n logger.debug('Password was a match, returning User');\n return returnUserAndChannelInfo(user).then(function (userInfo) {\n return done(null, userInfo);\n }).catch(function (error) {\n return error;\n });\n }).catch(function (error) {\n return error;\n });\n }).catch(function (error) {\n return done(error);\n });\n});\n\n/***/ }),\n/* 45 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar logger = __webpack_require__(0);\n\nvar _require = __webpack_require__(12),\n returnShortId = _require.returnShortId;\n\nmodule.exports = function (sequelize, _ref) {\n var STRING = _ref.STRING,\n BOOLEAN = _ref.BOOLEAN,\n INTEGER = _ref.INTEGER,\n TEXT = _ref.TEXT,\n DECIMAL = _ref.DECIMAL;\n\n var Certificate = sequelize.define('Certificate', {\n address: {\n type: STRING,\n default: null\n },\n amount: {\n type: DECIMAL(19, 8),\n default: null\n },\n claimId: {\n type: STRING,\n default: null\n },\n claimSequence: {\n type: INTEGER,\n default: null\n },\n decodedClaim: {\n type: BOOLEAN,\n default: null\n },\n depth: {\n type: INTEGER,\n default: null\n },\n effectiveAmount: {\n type: DECIMAL(19, 8),\n default: null\n },\n hasSignature: {\n type: BOOLEAN,\n default: null\n },\n height: {\n type: INTEGER,\n default: null\n },\n hex: {\n type: TEXT('long'),\n default: null\n },\n name: {\n type: STRING,\n default: null\n },\n nout: {\n type: INTEGER,\n default: null\n },\n txid: {\n type: STRING,\n default: null\n },\n validAtHeight: {\n type: INTEGER,\n default: null\n },\n outpoint: {\n type: STRING,\n default: null\n },\n valueVersion: {\n type: STRING,\n default: null\n },\n claimType: {\n type: STRING,\n default: null\n },\n certificateVersion: {\n type: STRING,\n default: null\n },\n keyType: {\n type: STRING,\n default: null\n },\n publicKey: {\n type: TEXT('long'),\n default: null\n }\n }, {\n freezeTableName: true\n });\n\n Certificate.associate = function (db) {\n Certificate.belongsTo(db.Channel, {\n foreignKey: {\n allowNull: true\n }\n });\n };\n\n Certificate.getShortChannelIdFromLongChannelId = function (longChannelId, channelName) {\n var _this = this;\n\n logger.debug('getShortChannelIdFromLongChannelId ' + channelName + ':' + longChannelId);\n return new Promise(function (resolve, reject) {\n _this.findAll({\n where: { name: channelName },\n order: [['height', 'ASC']]\n }).then(function (result) {\n switch (result.length) {\n case 0:\n throw new Error('No channel(s) found with that channel name');\n default:\n return resolve(returnShortId(result, longChannelId));\n }\n }).catch(function (error) {\n reject(error);\n });\n });\n };\n\n Certificate.getLongChannelIdFromShortChannelId = function (channelName, channelClaimId) {\n var _this2 = this;\n\n logger.debug('getLongChannelIdFromShortChannelId(' + channelName + ', ' + channelClaimId + ')');\n return new Promise(function (resolve, reject) {\n _this2.findAll({\n where: {\n name: channelName,\n claimId: {\n $like: channelClaimId + '%'\n }\n },\n order: [['height', 'ASC']]\n }).then(function (result) {\n switch (result.length) {\n case 0:\n return resolve(null);\n default:\n // note results must be sorted\n return resolve(result[0].claimId);\n }\n }).catch(function (error) {\n reject(error);\n });\n });\n };\n\n Certificate.getLongChannelIdFromChannelName = function (channelName) {\n var _this3 = this;\n\n logger.debug('getLongChannelIdFromChannelName(' + channelName + ')');\n return new Promise(function (resolve, reject) {\n _this3.findAll({\n where: { name: channelName },\n order: [['effectiveAmount', 'DESC'], ['height', 'ASC']]\n }).then(function (result) {\n switch (result.length) {\n case 0:\n return resolve(null);\n default:\n return resolve(result[0].claimId);\n }\n }).catch(function (error) {\n reject(error);\n });\n });\n };\n\n Certificate.validateLongChannelId = function (name, claimId) {\n var _this4 = this;\n\n logger.debug('validateLongChannelId(' + name + ', ' + claimId + ')');\n return new Promise(function (resolve, reject) {\n _this4.findOne({\n where: { name: name, claimId: claimId }\n }).then(function (result) {\n if (!result) {\n return resolve(null);\n };\n resolve(claimId);\n }).catch(function (error) {\n reject(error);\n });\n });\n };\n\n Certificate.getLongChannelId = function (channelName, channelClaimId) {\n logger.debug('getLongChannelId(' + channelName + ', ' + channelClaimId + ')');\n if (channelClaimId && channelClaimId.length === 40) {\n // if a full channel id is provided\n return this.validateLongChannelId(channelName, channelClaimId);\n } else if (channelClaimId && channelClaimId.length < 40) {\n // if a short channel id is provided\n return this.getLongChannelIdFromShortChannelId(channelName, channelClaimId);\n } else {\n return this.getLongChannelIdFromChannelName(channelName); // if no channel id provided\n }\n };\n\n return Certificate;\n};\n\n/***/ }),\n/* 46 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nmodule.exports = function (sequelize, _ref) {\n var STRING = _ref.STRING;\n\n var Channel = sequelize.define('Channel', {\n channelName: {\n type: STRING,\n allowNull: false\n },\n channelClaimId: {\n type: STRING,\n allowNull: false\n }\n }, {\n freezeTableName: true\n });\n\n Channel.associate = function (db) {\n Channel.belongsTo(db.User);\n Channel.hasOne(db.Certificate);\n };\n\n return Channel;\n};\n\n/***/ }),\n/* 47 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar logger = __webpack_require__(0);\n\nvar _require = __webpack_require__(12),\n returnShortId = _require.returnShortId;\n\nvar _require2 = __webpack_require__(3),\n defaultThumbnail = _require2.assetDefaults.thumbnail,\n host = _require2.details.host;\n\nfunction determineFileExtensionFromContentType(contentType) {\n switch (contentType) {\n case 'image/jpeg':\n case 'image/jpg':\n return 'jpeg';\n case 'image/png':\n return 'png';\n case 'image/gif':\n return 'gif';\n case 'video/mp4':\n return 'mp4';\n default:\n logger.debug('setting unknown file type as file extension jpeg');\n return 'jpeg';\n }\n};\n\nfunction determineThumbnail(storedThumbnail, defaultThumbnail) {\n if (storedThumbnail === '') {\n return defaultThumbnail;\n }\n return storedThumbnail;\n};\n\nfunction prepareClaimData(claim) {\n // logger.debug('preparing claim data based on resolved data:', claim);\n claim['thumbnail'] = determineThumbnail(claim.thumbnail, defaultThumbnail);\n claim['fileExt'] = determineFileExtensionFromContentType(claim.contentType);\n claim['host'] = host;\n return claim;\n};\n\nmodule.exports = function (sequelize, _ref) {\n var STRING = _ref.STRING,\n BOOLEAN = _ref.BOOLEAN,\n INTEGER = _ref.INTEGER,\n TEXT = _ref.TEXT,\n DECIMAL = _ref.DECIMAL;\n\n var Claim = sequelize.define('Claim', {\n address: {\n type: STRING,\n default: null\n },\n amount: {\n type: DECIMAL(19, 8),\n default: null\n },\n claimId: {\n type: STRING,\n default: null\n },\n claimSequence: {\n type: INTEGER,\n default: null\n },\n decodedClaim: {\n type: BOOLEAN,\n default: null\n },\n depth: {\n type: INTEGER,\n default: null\n },\n effectiveAmount: {\n type: DECIMAL(19, 8),\n default: null\n },\n hasSignature: {\n type: BOOLEAN,\n default: null\n },\n height: {\n type: INTEGER,\n default: null\n },\n hex: {\n type: TEXT('long'),\n default: null\n },\n name: {\n type: STRING,\n default: null\n },\n nout: {\n type: INTEGER,\n default: null\n },\n txid: {\n type: STRING,\n default: null\n },\n validAtHeight: {\n type: INTEGER,\n default: null\n },\n outpoint: {\n type: STRING,\n default: null\n },\n claimType: {\n type: STRING,\n default: null\n },\n certificateId: {\n type: STRING,\n default: null\n },\n author: {\n type: STRING,\n default: null\n },\n description: {\n type: TEXT('long'),\n default: null\n },\n language: {\n type: STRING,\n default: null\n },\n license: {\n type: STRING,\n default: null\n },\n licenseUrl: {\n type: STRING,\n default: null\n },\n nsfw: {\n type: BOOLEAN,\n default: null\n },\n preview: {\n type: STRING,\n default: null\n },\n thumbnail: {\n type: STRING,\n default: null\n },\n title: {\n type: STRING,\n default: null\n },\n metadataVersion: {\n type: STRING,\n default: null\n },\n contentType: {\n type: STRING,\n default: null\n },\n source: {\n type: STRING,\n default: null\n },\n sourceType: {\n type: STRING,\n default: null\n },\n sourceVersion: {\n type: STRING,\n default: null\n },\n streamVersion: {\n type: STRING,\n default: null\n },\n valueVersion: {\n type: STRING,\n default: null\n },\n channelName: {\n type: STRING,\n allowNull: true,\n default: null\n }\n }, {\n freezeTableName: true\n });\n\n Claim.associate = function (db) {\n Claim.belongsTo(db.File, {\n foreignKey: {\n allowNull: true\n }\n });\n };\n\n Claim.getShortClaimIdFromLongClaimId = function (claimId, claimName) {\n var _this = this;\n\n logger.debug('Claim.getShortClaimIdFromLongClaimId for ' + claimName + '#' + claimId);\n return new Promise(function (resolve, reject) {\n _this.findAll({\n where: { name: claimName },\n order: [['height', 'ASC']]\n }).then(function (result) {\n switch (result.length) {\n case 0:\n throw new Error('No claim(s) found with that claim name');\n default:\n resolve(returnShortId(result, claimId));\n }\n }).catch(function (error) {\n reject(error);\n });\n });\n };\n\n Claim.getAllChannelClaims = function (channelClaimId) {\n var _this2 = this;\n\n logger.debug('Claim.getAllChannelClaims for ' + channelClaimId);\n return new Promise(function (resolve, reject) {\n _this2.findAll({\n where: { certificateId: channelClaimId },\n order: [['height', 'ASC']],\n raw: true // returns an array of only data, not an array of instances\n }).then(function (channelClaimsArray) {\n // logger.debug('channelclaimsarray length:', channelClaimsArray.length);\n switch (channelClaimsArray.length) {\n case 0:\n return resolve(null);\n default:\n channelClaimsArray.forEach(function (claim) {\n claim['fileExt'] = determineFileExtensionFromContentType(claim.contentType);\n claim['thumbnail'] = determineThumbnail(claim.thumbnail, defaultThumbnail);\n return claim;\n });\n return resolve(channelClaimsArray);\n }\n }).catch(function (error) {\n reject(error);\n });\n });\n };\n\n Claim.getClaimIdByLongChannelId = function (channelClaimId, claimName) {\n var _this3 = this;\n\n logger.debug('finding claim id for claim ' + claimName + ' from channel ' + channelClaimId);\n return new Promise(function (resolve, reject) {\n _this3.findAll({\n where: { name: claimName, certificateId: channelClaimId },\n order: [['id', 'ASC']]\n }).then(function (result) {\n switch (result.length) {\n case 0:\n return resolve(null);\n case 1:\n return resolve(result[0].claimId);\n default:\n logger.error(result.length + ' records found for \"' + claimName + '\" in channel \"' + channelClaimId + '\"');\n return resolve(result[0].claimId);\n }\n }).catch(function (error) {\n reject(error);\n });\n });\n };\n\n Claim.getLongClaimIdFromShortClaimId = function (name, shortId) {\n var _this4 = this;\n\n return new Promise(function (resolve, reject) {\n _this4.findAll({\n where: {\n name: name,\n claimId: {\n $like: shortId + '%'\n } },\n order: [['height', 'ASC']]\n }).then(function (result) {\n switch (result.length) {\n case 0:\n return resolve(null);\n default:\n // note results must be sorted\n return resolve(result[0].claimId);\n }\n }).catch(function (error) {\n reject(error);\n });\n });\n };\n\n Claim.getTopFreeClaimIdByClaimName = function (name) {\n var _this5 = this;\n\n return new Promise(function (resolve, reject) {\n _this5.findAll({\n where: { name: name },\n order: [['effectiveAmount', 'DESC'], ['height', 'ASC']] // note: maybe height and effective amount need to switch?\n }).then(function (result) {\n logger.debug('length of result', result.length);\n switch (result.length) {\n case 0:\n return resolve(null);\n default:\n return resolve(result[0].dataValues.claimId);\n }\n }).catch(function (error) {\n reject(error);\n });\n });\n };\n\n Claim.validateLongClaimId = function (name, claimId) {\n var _this6 = this;\n\n return new Promise(function (resolve, reject) {\n _this6.findOne({\n where: { name: name, claimId: claimId }\n }).then(function (result) {\n if (!result) {\n return resolve(null);\n };\n resolve(claimId);\n }).catch(function (error) {\n reject(error);\n });\n });\n };\n\n Claim.getLongClaimId = function (claimName, claimId) {\n logger.debug('getLongClaimId(' + claimName + ', ' + claimId + ')');\n if (claimId && claimId.length === 40) {\n // if a full claim id is provided\n return this.validateLongClaimId(claimName, claimId);\n } else if (claimId && claimId.length < 40) {\n return this.getLongClaimIdFromShortClaimId(claimName, claimId); // if a short claim id is provided\n } else {\n return this.getTopFreeClaimIdByClaimName(claimName); // if no claim id is provided\n }\n };\n\n Claim.resolveClaim = function (name, claimId) {\n var _this7 = this;\n\n logger.debug('Claim.resolveClaim: ' + name + ' ' + claimId);\n return new Promise(function (resolve, reject) {\n _this7.findAll({\n where: { name: name, claimId: claimId }\n }).then(function (claimArray) {\n switch (claimArray.length) {\n case 0:\n return resolve(null);\n case 1:\n return resolve(prepareClaimData(claimArray[0].dataValues));\n default:\n logger.error('more than one record matches ' + name + '#' + claimId + ' in db.Claim');\n return resolve(prepareClaimData(claimArray[0].dataValues));\n }\n }).catch(function (error) {\n reject(error);\n });\n });\n };\n\n return Claim;\n};\n\n/***/ }),\n/* 48 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nmodule.exports = function (sequelize, _ref) {\n var STRING = _ref.STRING,\n BOOLEAN = _ref.BOOLEAN,\n INTEGER = _ref.INTEGER;\n\n var File = sequelize.define('File', {\n name: {\n type: STRING,\n allowNull: false\n },\n claimId: {\n type: STRING,\n allowNull: false\n },\n address: {\n type: STRING,\n allowNull: false\n },\n outpoint: {\n type: STRING,\n allowNull: false\n },\n height: {\n type: INTEGER,\n allowNull: false,\n default: 0\n },\n fileName: {\n type: STRING,\n allowNull: false\n },\n filePath: {\n type: STRING,\n allowNull: false\n },\n fileType: {\n type: STRING\n },\n nsfw: {\n type: BOOLEAN,\n allowNull: false,\n defaultValue: false\n },\n trendingEligible: {\n type: BOOLEAN,\n allowNull: false,\n defaultValue: true\n }\n }, {\n freezeTableName: true\n });\n\n File.associate = function (db) {\n File.hasMany(db.Request);\n File.hasOne(db.Claim);\n };\n\n File.getRecentClaims = function () {\n return this.findAll({\n where: { nsfw: false, trendingEligible: true },\n order: [['createdAt', 'DESC']],\n limit: 25\n });\n };\n\n return File;\n};\n\n/***/ }),\n/* 49 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nmodule.exports = function (sequelize, _ref) {\n var STRING = _ref.STRING,\n BOOLEAN = _ref.BOOLEAN,\n TEXT = _ref.TEXT;\n\n var Request = sequelize.define('Request', {\n action: {\n type: STRING,\n allowNull: false\n },\n url: {\n type: STRING,\n allowNull: false\n },\n ipAddress: {\n type: STRING,\n allowNull: true\n },\n result: {\n type: TEXT('long'),\n allowNull: true,\n default: null\n }\n }, {\n freezeTableName: true\n });\n\n Request.associate = function (db) {\n Request.belongsTo(db.File, {\n foreignKey: {\n allowNull: true\n }\n });\n };\n\n return Request;\n};\n\n/***/ }),\n/* 50 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar bcrypt = __webpack_require__(51);\nvar logger = __webpack_require__(0);\n\nmodule.exports = function (sequelize, _ref) {\n var STRING = _ref.STRING;\n\n var User = sequelize.define('User', {\n userName: {\n type: STRING,\n allowNull: false\n },\n password: {\n type: STRING,\n allowNull: false\n }\n }, {\n freezeTableName: true\n });\n\n User.associate = function (db) {\n User.hasOne(db.Channel);\n };\n\n User.prototype.comparePassword = function (password) {\n return bcrypt.compare(password, this.password);\n };\n\n User.prototype.changePassword = function (newPassword) {\n var _this = this;\n\n return new Promise(function (resolve, reject) {\n // generate a salt string to use for hashing\n bcrypt.genSalt(function (saltError, salt) {\n if (saltError) {\n logger.error('salt error', saltError);\n reject(saltError);\n return;\n }\n // generate a hashed version of the user's password\n bcrypt.hash(newPassword, salt, function (hashError, hash) {\n // if there is an error with the hash generation return the error\n if (hashError) {\n logger.error('hash error', hashError);\n reject(hashError);\n return;\n }\n // replace the current password with the new hash\n _this.update({ password: hash }).then(function () {\n resolve();\n }).catch(function (error) {\n reject(error);\n });\n });\n });\n });\n };\n\n // pre-save hook method to hash the user's password before the user's info is saved to the db.\n User.hook('beforeCreate', function (user, options) {\n logger.debug('User.beforeCreate hook...');\n return new Promise(function (resolve, reject) {\n // generate a salt string to use for hashing\n bcrypt.genSalt(function (saltError, salt) {\n if (saltError) {\n logger.error('salt error', saltError);\n reject(saltError);\n return;\n }\n // generate a hashed version of the user's password\n bcrypt.hash(user.password, salt, function (hashError, hash) {\n // if there is an error with the hash generation return the error\n if (hashError) {\n logger.error('hash error', hashError);\n reject(hashError);\n return;\n }\n // replace the password string with the hash password value\n user.password = hash;\n resolve();\n });\n });\n });\n });\n\n return User;\n};\n\n/***/ }),\n/* 51 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"bcrypt\");\n\n/***/ }),\n/* 52 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i[\"return\"]) _i[\"return\"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError(\"Invalid attempt to destructure non-iterable instance\"); } }; }();\n\nvar PassportLocalStrategy = __webpack_require__(11).Strategy;\nvar lbryApi = __webpack_require__(5);\nvar logger = __webpack_require__(0);\nvar db = __webpack_require__(2);\n\nmodule.exports = new PassportLocalStrategy({\n usernameField: 'username',\n passwordField: 'password'\n}, function (username, password, done) {\n logger.verbose('new channel signup request. user: ' + username + ' pass: ' + password + ' .');\n var userInfo = {};\n // server-side validaton of inputs (username, password)\n\n // create the channel and retrieve the metadata\n return lbryApi.createChannel('@' + username).then(function (tx) {\n // create user record\n var userData = {\n userName: username,\n password: password\n };\n logger.verbose('userData >', userData);\n // create user record\n var channelData = {\n channelName: '@' + username,\n channelClaimId: tx.claim_id\n };\n logger.verbose('channelData >', channelData);\n // create certificate record\n var certificateData = {\n claimId: tx.claim_id,\n name: '@' + username\n // address,\n };\n logger.verbose('certificateData >', certificateData);\n // save user and certificate to db\n return Promise.all([db.User.create(userData), db.Channel.create(channelData), db.Certificate.create(certificateData)]);\n }).then(function (_ref) {\n var _ref2 = _slicedToArray(_ref, 3),\n newUser = _ref2[0],\n newChannel = _ref2[1],\n newCertificate = _ref2[2];\n\n logger.verbose('user and certificate successfully created');\n // store the relevant newUser info to be passed back for req.User\n userInfo['id'] = newUser.id;\n userInfo['userName'] = newUser.userName;\n userInfo['channelName'] = newChannel.channelName;\n userInfo['channelClaimId'] = newChannel.channelClaimId;\n // associate the instances\n return Promise.all([newCertificate.setChannel(newChannel), newChannel.setUser(newUser)]);\n }).then(function () {\n logger.verbose('user and certificate successfully associated');\n return db.Certificate.getShortChannelIdFromLongChannelId(userInfo.channelClaimId, userInfo.channelName);\n }).then(function (shortChannelId) {\n userInfo['shortChannelId'] = shortChannelId;\n return done(null, userInfo);\n }).catch(function (error) {\n logger.error('signup error', error);\n return done(error);\n });\n});\n\n/***/ }),\n/* 53 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"axios\");\n\n/***/ }),\n/* 54 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar lbryConfig = {\n api: {\n apiHost: 'localhost',\n apiPort: '5279'\n }\n};\n\nmodule.exports = lbryConfig;\n\n/***/ }),\n/* 55 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"universal-analytics\");\n\n/***/ }),\n/* 56 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nmodule.exports = {\n serializeSpeechUser: function serializeSpeechUser(user, done) {\n // returns user data to be serialized into session\n console.log('serializing user');\n done(null, user);\n },\n deserializeSpeechUser: function deserializeSpeechUser(user, done) {\n // deserializes session and populates additional info to req.user\n console.log('deserializing user');\n done(null, user);\n }\n};\n\n/***/ }),\n/* 57 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar speechPassport = __webpack_require__(7);\nvar handleSignupRequest = __webpack_require__(58);\nvar handleLoginRequest = __webpack_require__(59);\nvar handleLogoutRequest = __webpack_require__(60);\nvar handleUserRequest = __webpack_require__(61);\n\nmodule.exports = function (app) {\n app.post('/signup', speechPassport.authenticate('local-signup'), handleSignupRequest);\n app.post('/login', handleLoginRequest);\n app.get('/logout', handleLogoutRequest);\n app.get('/user', handleUserRequest);\n};\n\n/***/ }),\n/* 58 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar signup = function signup(req, res) {\n res.status(200).json({\n success: true,\n channelName: req.user.channelName,\n channelClaimId: req.user.channelClaimId,\n shortChannelId: req.user.shortChannelId\n });\n};\n\nmodule.exports = signup;\n\n/***/ }),\n/* 59 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar speechPassport = __webpack_require__(7);\n\nvar login = function login(req, res, next) {\n speechPassport.authenticate('local-login', function (err, user, info) {\n if (err) {\n return next(err);\n }\n if (!user) {\n return res.status(400).json({\n success: false,\n message: info.message\n });\n }\n req.logIn(user, function (err) {\n if (err) {\n return next(err);\n }\n return res.status(200).json({\n success: true,\n channelName: req.user.channelName,\n channelClaimId: req.user.channelClaimId,\n shortChannelId: req.user.shortChannelId\n });\n });\n })(req, res, next);\n};\n\nmodule.exports = login;\n\n/***/ }),\n/* 60 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar logout = function logout(req, res) {\n req.logout();\n res.status(200).json({ success: true, message: 'you successfully logged out' });\n};\n\nmodule.exports = logout;\n\n/***/ }),\n/* 61 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar user = function user(req, res) {\n if (req.user) {\n res.status(200).json({ success: true, data: req.user });\n } else {\n res.status(401).json({ success: false, message: 'user is not logged in' });\n }\n};\n\nmodule.exports = user;\n\n/***/ }),\n/* 62 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar channelAvailability = __webpack_require__(63);\nvar channelClaims = __webpack_require__(65);\nvar channelData = __webpack_require__(67);\nvar channelShortId = __webpack_require__(68);\nvar claimAvailability = __webpack_require__(69);\nvar claimData = __webpack_require__(70);\nvar claimGet = __webpack_require__(71);\nvar claimLongId = __webpack_require__(72);\nvar claimPublish = __webpack_require__(73);\nvar claimResolve = __webpack_require__(75);\nvar claimShortId = __webpack_require__(76);\nvar claimList = __webpack_require__(77);\nvar fileAvailability = __webpack_require__(78);\n\nvar multipartMiddleware = __webpack_require__(79);\n\nmodule.exports = function (app) {\n // channel routes\n app.get('/api/channel/availability/:name', channelAvailability);\n app.get('/api/channel/short-id/:longId/:name', channelShortId);\n app.get('/api/channel/data/:channelName/:channelClaimId', channelData);\n app.get('/api/channel/claims/:channelName/:channelClaimId/:page', channelClaims);\n // claim routes\n app.get('/api/claim/list/:name', claimList);\n app.get('/api/claim/get/:name/:claimId', claimGet);\n app.get('/api/claim/availability/:name', claimAvailability);\n app.get('/api/claim/resolve/:name/:claimId', claimResolve);\n app.post('/api/claim/publish', multipartMiddleware, claimPublish);\n app.get('/api/claim/short-id/:longId/:name', claimShortId);\n app.post('/api/claim/long-id', claimLongId);\n app.get('/api/claim/data/:claimName/:claimId', claimData);\n // file routes\n app.get('/api/file/availability/:name/:claimId', fileAvailability);\n};\n\n/***/ }),\n/* 63 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(8),\n checkChannelAvailability = _require.checkChannelAvailability;\n\nvar _require2 = __webpack_require__(4),\n sendGATimingEvent = _require2.sendGATimingEvent;\n\nvar _require3 = __webpack_require__(1),\n handleErrorResponse = _require3.handleErrorResponse;\n\n/*\n\n route to check whether site has published to a channel\n\n*/\n\nvar channelAvailability = function channelAvailability(_ref, res) {\n var ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n name = _ref.params.name;\n\n var gaStartTime = Date.now();\n checkChannelAvailability(name).then(function (availableName) {\n res.status(200).json(availableName);\n sendGATimingEvent('end-to-end', 'claim name availability', name, gaStartTime, Date.now());\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = channelAvailability;\n\n/***/ }),\n/* 64 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"fs\");\n\n/***/ }),\n/* 65 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(6),\n getChannelClaims = _require.getChannelClaims;\n\nvar _require2 = __webpack_require__(1),\n handleErrorResponse = _require2.handleErrorResponse;\n\nvar NO_CHANNEL = 'NO_CHANNEL';\n\n/*\n\n route to get all claims for channel\n\n*/\n\nvar channelClaims = function channelClaims(_ref, res) {\n var ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n body = _ref.body,\n params = _ref.params;\n\n var channelName = params.channelName;\n var channelClaimId = params.channelClaimId;\n if (channelClaimId === 'none') channelClaimId = null;\n var page = params.page;\n getChannelClaims(channelName, channelClaimId, page).then(function (data) {\n if (data === NO_CHANNEL) {\n return res.status(404).json({ success: false, message: 'No matching channel was found' });\n }\n res.status(200).json({ success: true, data: data });\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = channelClaims;\n\n/***/ }),\n/* 66 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar CLAIMS_PER_PAGE = 12;\n\nmodule.exports = {\n returnPaginatedChannelClaims: function returnPaginatedChannelClaims(channelName, longChannelClaimId, claims, page) {\n var totalPages = module.exports.determineTotalPages(claims);\n var paginationPage = module.exports.getPageFromQuery(page);\n var viewData = {\n channelName: channelName,\n longChannelClaimId: longChannelClaimId,\n claims: module.exports.extractPageFromClaims(claims, paginationPage),\n previousPage: module.exports.determinePreviousPage(paginationPage),\n currentPage: paginationPage,\n nextPage: module.exports.determineNextPage(totalPages, paginationPage),\n totalPages: totalPages,\n totalResults: module.exports.determineTotalClaims(claims)\n };\n return viewData;\n },\n getPageFromQuery: function getPageFromQuery(page) {\n if (page) {\n return parseInt(page);\n }\n return 1;\n },\n extractPageFromClaims: function extractPageFromClaims(claims, pageNumber) {\n if (!claims) {\n return []; // if no claims, return this default\n }\n // logger.debug('claims is array?', Array.isArray(claims));\n // logger.debug(`pageNumber ${pageNumber} is number?`, Number.isInteger(pageNumber));\n var claimStartIndex = (pageNumber - 1) * CLAIMS_PER_PAGE;\n var claimEndIndex = claimStartIndex + CLAIMS_PER_PAGE;\n var pageOfClaims = claims.slice(claimStartIndex, claimEndIndex);\n return pageOfClaims;\n },\n determineTotalPages: function determineTotalPages(claims) {\n if (!claims) {\n return 0;\n } else {\n var totalClaims = claims.length;\n if (totalClaims < CLAIMS_PER_PAGE) {\n return 1;\n }\n var fullPages = Math.floor(totalClaims / CLAIMS_PER_PAGE);\n var remainder = totalClaims % CLAIMS_PER_PAGE;\n if (remainder === 0) {\n return fullPages;\n }\n return fullPages + 1;\n }\n },\n determinePreviousPage: function determinePreviousPage(currentPage) {\n if (currentPage === 1) {\n return null;\n }\n return currentPage - 1;\n },\n determineNextPage: function determineNextPage(totalPages, currentPage) {\n if (currentPage === totalPages) {\n return null;\n }\n return currentPage + 1;\n },\n determineTotalClaims: function determineTotalClaims(claims) {\n if (!claims) {\n return 0;\n }\n return claims.length;\n }\n};\n\n/***/ }),\n/* 67 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(6),\n getChannelData = _require.getChannelData;\n\nvar _require2 = __webpack_require__(1),\n handleErrorResponse = _require2.handleErrorResponse;\n\nvar NO_CHANNEL = 'NO_CHANNEL';\n\n/*\n\n route to get data for a channel\n\n*/\n\nvar channelData = function channelData(_ref, res) {\n var ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n body = _ref.body,\n params = _ref.params;\n\n var channelName = params.channelName;\n var channelClaimId = params.channelClaimId;\n if (channelClaimId === 'none') channelClaimId = null;\n getChannelData(channelName, channelClaimId, 0).then(function (data) {\n if (data === NO_CHANNEL) {\n return res.status(404).json({ success: false, message: 'No matching channel was found' });\n }\n res.status(200).json({ success: true, data: data });\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = channelData;\n\n/***/ }),\n/* 68 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(1),\n handleErrorResponse = _require.handleErrorResponse;\n\nvar db = __webpack_require__(2);\n\n/*\n\nroute to get a short channel id from long channel Id\n\n*/\n\nvar channelShortIdRoute = function channelShortIdRoute(_ref, res) {\n var ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n params = _ref.params;\n\n db.Certificate.getShortChannelIdFromLongChannelId(params.longId, params.name).then(function (shortId) {\n res.status(200).json(shortId);\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = channelShortIdRoute;\n\n/***/ }),\n/* 69 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(8),\n claimNameIsAvailable = _require.claimNameIsAvailable;\n\nvar _require2 = __webpack_require__(4),\n sendGATimingEvent = _require2.sendGATimingEvent;\n\nvar _require3 = __webpack_require__(1),\n handleErrorResponse = _require3.handleErrorResponse;\n\n/*\n\n route to check whether this site published to a claim\n\n*/\n\nvar claimAvailability = function claimAvailability(_ref, res) {\n var ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n name = _ref.params.name;\n\n var gaStartTime = Date.now();\n claimNameIsAvailable(name).then(function (result) {\n res.status(200).json(result);\n sendGATimingEvent('end-to-end', 'claim name availability', name, gaStartTime, Date.now());\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimAvailability;\n\n/***/ }),\n/* 70 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(1),\n handleErrorResponse = _require.handleErrorResponse;\n\nvar db = __webpack_require__(2);\n\n/*\n\n route to return data for a claim\n\n*/\n\nvar claimData = function claimData(_ref, res) {\n var ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n body = _ref.body,\n params = _ref.params;\n\n var claimName = params.claimName;\n var claimId = params.claimId;\n if (claimId === 'none') claimId = null;\n db.Claim.resolveClaim(claimName, claimId).then(function (claimInfo) {\n if (!claimInfo) {\n return res.status(404).json({ success: false, message: 'No claim could be found' });\n }\n res.status(200).json({ success: true, data: claimInfo });\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimData;\n\n/***/ }),\n/* 71 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i[\"return\"]) _i[\"return\"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError(\"Invalid attempt to destructure non-iterable instance\"); } }; }();\n\nvar _require = __webpack_require__(5),\n getClaim = _require.getClaim;\n\nvar _require2 = __webpack_require__(9),\n addGetResultsToFileData = _require2.addGetResultsToFileData,\n createFileData = _require2.createFileData;\n\nvar _require3 = __webpack_require__(1),\n handleErrorResponse = _require3.handleErrorResponse;\n\nvar db = __webpack_require__(2);\n\n/*\n\n route to get a claim\n\n*/\n\nvar claimGet = function claimGet(_ref, res) {\n var ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n params = _ref.params;\n\n var name = params.name;\n var claimId = params.claimId;\n // resolve the claim\n db.Claim.resolveClaim(name, claimId).then(function (resolveResult) {\n // make sure a claim actually exists at that uri\n if (!resolveResult) {\n throw new Error('No matching uri found in Claim table');\n }\n var fileData = createFileData(resolveResult);\n // get the claim\n return Promise.all([fileData, getClaim(name + '#' + claimId)]);\n }).then(function (_ref2) {\n var _ref3 = _slicedToArray(_ref2, 2),\n fileData = _ref3[0],\n getResult = _ref3[1];\n\n fileData = addGetResultsToFileData(fileData, getResult);\n return Promise.all([db.upsert(db.File, fileData, { name: name, claimId: claimId }, 'File'), getResult]);\n }).then(function (_ref4) {\n var _ref5 = _slicedToArray(_ref4, 2),\n fileRecord = _ref5[0],\n _ref5$ = _ref5[1],\n message = _ref5$.message,\n completed = _ref5$.completed;\n\n res.status(200).json({ success: true, message: message, completed: completed });\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimGet;\n\n/***/ }),\n/* 72 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(6),\n getClaimId = _require.getClaimId;\n\nvar _require2 = __webpack_require__(1),\n handleErrorResponse = _require2.handleErrorResponse;\n\nvar NO_CHANNEL = 'NO_CHANNEL';\nvar NO_CLAIM = 'NO_CLAIM';\n\n/*\n\n route to get a long claim id\n\n*/\n\nvar claimLongId = function claimLongId(_ref, res) {\n var ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n body = _ref.body,\n params = _ref.params;\n\n var channelName = body.channelName;\n var channelClaimId = body.channelClaimId;\n var claimName = body.claimName;\n var claimId = body.claimId;\n getClaimId(channelName, channelClaimId, claimName, claimId).then(function (result) {\n if (result === NO_CHANNEL) {\n return res.status(404).json({ success: false, message: 'No matching channel could be found' });\n }\n if (result === NO_CLAIM) {\n return res.status(404).json({ success: false, message: 'No matching claim id could be found' });\n }\n res.status(200).json({ success: true, data: result });\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimLongId;\n\n/***/ }),\n/* 73 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i[\"return\"]) _i[\"return\"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError(\"Invalid attempt to destructure non-iterable instance\"); } }; }();\n\nvar _require = __webpack_require__(9),\n createBasicPublishParams = _require.createBasicPublishParams,\n createThumbnailPublishParams = _require.createThumbnailPublishParams,\n parsePublishApiRequestBody = _require.parsePublishApiRequestBody,\n parsePublishApiRequestFiles = _require.parsePublishApiRequestFiles;\n\nvar _require2 = __webpack_require__(8),\n claimNameIsAvailable = _require2.claimNameIsAvailable,\n publish = _require2.publish;\n\nvar _require3 = __webpack_require__(74),\n authenticateUser = _require3.authenticateUser;\n\nvar _require4 = __webpack_require__(4),\n sendGATimingEvent = _require4.sendGATimingEvent;\n\nvar _require5 = __webpack_require__(1),\n handleErrorResponse = _require5.handleErrorResponse;\n\nvar _require6 = __webpack_require__(3),\n host = _require6.details.host;\n\n/*\n\n route to publish a claim through the daemon\n\n*/\n\nvar claimPublish = function claimPublish(_ref, res) {\n var body = _ref.body,\n files = _ref.files,\n headers = _ref.headers,\n ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n user = _ref.user;\n\n // define variables\n var channelName = void 0,\n channelId = void 0,\n channelPassword = void 0,\n description = void 0,\n fileName = void 0,\n filePath = void 0,\n fileType = void 0,\n gaStartTime = void 0,\n license = void 0,\n name = void 0,\n nsfw = void 0,\n thumbnail = void 0,\n thumbnailFileName = void 0,\n thumbnailFilePath = void 0,\n thumbnailFileType = void 0,\n title = void 0;\n // record the start time of the request\n gaStartTime = Date.now();\n // validate the body and files of the request\n try {\n var _parsePublishApiReque = parsePublishApiRequestBody(body);\n // validateApiPublishRequest(body, files);\n\n\n name = _parsePublishApiReque.name;\n nsfw = _parsePublishApiReque.nsfw;\n license = _parsePublishApiReque.license;\n title = _parsePublishApiReque.title;\n description = _parsePublishApiReque.description;\n thumbnail = _parsePublishApiReque.thumbnail;\n\n var _parsePublishApiReque2 = parsePublishApiRequestFiles(files);\n\n fileName = _parsePublishApiReque2.fileName;\n filePath = _parsePublishApiReque2.filePath;\n fileType = _parsePublishApiReque2.fileType;\n thumbnailFileName = _parsePublishApiReque2.thumbnailFileName;\n thumbnailFilePath = _parsePublishApiReque2.thumbnailFilePath;\n thumbnailFileType = _parsePublishApiReque2.thumbnailFileType;\n channelName = body.channelName;\n channelId = body.channelId;\n channelPassword = body.channelPassword;\n } catch (error) {\n return res.status(400).json({ success: false, message: error.message });\n }\n // check channel authorization\n Promise.all([authenticateUser(channelName, channelId, channelPassword, user), claimNameIsAvailable(name), createBasicPublishParams(filePath, name, title, description, license, nsfw, thumbnail), createThumbnailPublishParams(thumbnailFilePath, name, license, nsfw)]).then(function (_ref2) {\n var _ref3 = _slicedToArray(_ref2, 4),\n _ref3$ = _ref3[0],\n channelName = _ref3$.channelName,\n channelClaimId = _ref3$.channelClaimId,\n validatedClaimName = _ref3[1],\n publishParams = _ref3[2],\n thumbnailPublishParams = _ref3[3];\n\n // add channel details to the publish params\n if (channelName && channelClaimId) {\n publishParams['channel_name'] = channelName;\n publishParams['channel_id'] = channelClaimId;\n }\n // publish the thumbnail\n if (thumbnailPublishParams) {\n publish(thumbnailPublishParams, thumbnailFileName, thumbnailFileType);\n }\n // publish the asset\n return publish(publishParams, fileName, fileType);\n }).then(function (result) {\n res.status(200).json({\n success: true,\n message: 'publish completed successfully',\n data: {\n name: name,\n claimId: result.claim_id,\n url: host + '/' + result.claim_id + '/' + name,\n lbryTx: result\n }\n });\n // record the publish end time and send to google analytics\n sendGATimingEvent('end-to-end', 'publish', fileType, gaStartTime, Date.now());\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimPublish;\n\n/***/ }),\n/* 74 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar db = __webpack_require__(2);\nvar logger = __webpack_require__(0);\n\nmodule.exports = {\n authenticateUser: function authenticateUser(channelName, channelId, channelPassword, user) {\n // case: no channelName or channel Id are provided (anonymous), regardless of whether user token is provided\n if (!channelName && !channelId) {\n return {\n channelName: null,\n channelClaimId: null\n };\n }\n // case: channelName or channel Id are provided with user token\n if (user) {\n if (channelName && channelName !== user.channelName) {\n throw new Error('the provided channel name does not match user credentials');\n }\n if (channelId && channelId !== user.channelClaimId) {\n throw new Error('the provided channel id does not match user credentials');\n }\n return {\n channelName: user.channelName,\n channelClaimId: user.channelClaimId\n };\n }\n // case: channelName or channel Id are provided with password instead of user token\n if (!channelPassword) throw new Error('no channel password provided');\n return module.exports.authenticateChannelCredentials(channelName, channelId, channelPassword);\n },\n authenticateChannelCredentials: function authenticateChannelCredentials(channelName, channelId, userPassword) {\n return new Promise(function (resolve, reject) {\n // hoisted variables\n var channelData = void 0;\n // build the params for finding the channel\n var channelFindParams = {};\n if (channelName) channelFindParams['channelName'] = channelName;\n if (channelId) channelFindParams['channelClaimId'] = channelId;\n // find the channel\n db.Channel.findOne({\n where: channelFindParams\n }).then(function (channel) {\n if (!channel) {\n logger.debug('no channel found');\n throw new Error('Authentication failed, you do not have access to that channel');\n }\n channelData = channel.get();\n logger.debug('channel data:', channelData);\n return db.User.findOne({\n where: { userName: channelData.channelName.substring(1) }\n });\n }).then(function (user) {\n if (!user) {\n logger.debug('no user found');\n throw new Error('Authentication failed, you do not have access to that channel');\n }\n return user.comparePassword(userPassword);\n }).then(function (isMatch) {\n if (!isMatch) {\n logger.debug('incorrect password');\n throw new Error('Authentication failed, you do not have access to that channel');\n }\n logger.debug('...password was a match...');\n resolve(channelData);\n }).catch(function (error) {\n reject(error);\n });\n });\n }\n};\n\n/***/ }),\n/* 75 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(5),\n resolveUri = _require.resolveUri;\n\nvar _require2 = __webpack_require__(1),\n handleErrorResponse = _require2.handleErrorResponse;\n\n/*\n\n route to run a resolve request on the daemon\n\n*/\n\nvar claimResolve = function claimResolve(_ref, res) {\n var headers = _ref.headers,\n ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n params = _ref.params;\n\n resolveUri(params.name + '#' + params.claimId).then(function (resolvedUri) {\n res.status(200).json(resolvedUri);\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimResolve;\n\n/***/ }),\n/* 76 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(1),\n handleErrorResponse = _require.handleErrorResponse;\n\nvar db = __webpack_require__(2);\n\n/*\n\n route to get a short claim id from long claim Id\n\n*/\n\nvar claimShortId = function claimShortId(_ref, res) {\n var ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n body = _ref.body,\n params = _ref.params;\n\n db.Claim.getShortClaimIdFromLongClaimId(params.longId, params.name).then(function (shortId) {\n res.status(200).json({ success: true, data: shortId });\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimShortId;\n\n/***/ }),\n/* 77 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(5),\n getClaimList = _require.getClaimList;\n\nvar _require2 = __webpack_require__(1),\n handleErrorResponse = _require2.handleErrorResponse;\n\n/*\n\n route to get list of claims\n\n*/\n\nvar claimList = function claimList(_ref, res) {\n var ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n params = _ref.params;\n\n getClaimList(params.name).then(function (claimsList) {\n res.status(200).json(claimsList);\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimList;\n\n/***/ }),\n/* 78 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(1),\n handleErrorResponse = _require.handleErrorResponse;\n\nvar db = __webpack_require__(2);\n\n/*\n\n route to see if asset is available locally\n\n*/\n\nvar fileAvailability = function fileAvailability(_ref, res) {\n var ip = _ref.ip,\n originalUrl = _ref.originalUrl,\n params = _ref.params;\n\n var name = params.name;\n var claimId = params.claimId;\n db.File.findOne({\n where: {\n name: name,\n claimId: claimId\n }\n }).then(function (result) {\n if (result) {\n return res.status(200).json({ success: true, data: true });\n }\n res.status(200).json({ success: true, data: false });\n }).catch(function (error) {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = fileAvailability;\n\n/***/ }),\n/* 79 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar multipart = __webpack_require__(80);\n\nvar _require = __webpack_require__(3),\n uploadDirectory = _require.publishing.uploadDirectory;\n\nvar multipartMiddleware = multipart({ uploadDir: uploadDirectory });\n\nmodule.exports = multipartMiddleware;\n\n/***/ }),\n/* 80 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"connect-multiparty\");\n\n/***/ }),\n/* 81 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar handlePageRequest = __webpack_require__(82);\nvar handleEmbedRequest = __webpack_require__(83);\nvar redirect = __webpack_require__(84);\n\nmodule.exports = function (app) {\n app.get('/', handlePageRequest);\n app.get('/login', handlePageRequest);\n app.get('/about', handlePageRequest);\n app.get('/trending', redirect('/popular'));\n app.get('/popular', handlePageRequest);\n app.get('/new', handlePageRequest);\n app.get('/embed/:claimId/:name', handleEmbedRequest); // route to send embedable video player (for twitter)\n};\n\n/***/ }),\n/* 82 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar handlePageRender = __webpack_require__(14);\n\nvar sendReactApp = function sendReactApp(req, res) {\n handlePageRender(req, res);\n};\n\nmodule.exports = sendReactApp;\n\n/***/ }),\n/* 83 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(3),\n host = _require.details.host;\n\nvar sendEmbedPage = function sendEmbedPage(_ref, res) {\n var params = _ref.params;\n\n var claimId = params.claimId;\n var name = params.name;\n // get and render the content\n res.status(200).render('embed', { layout: 'embed', host: host, claimId: claimId, name: name });\n};\n\nmodule.exports = sendEmbedPage;\n\n/***/ }),\n/* 84 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar redirect = function redirect(route) {\n return function (req, res) {\n res.status(301).redirect(route);\n };\n};\n\nmodule.exports = redirect;\n\n/***/ }),\n/* 85 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar serveAssetByClaim = __webpack_require__(86);\nvar serveAssetByIdentifierAndClaim = __webpack_require__(89);\n\nmodule.exports = function (app, db) {\n app.get('/:identifier/:claim', serveAssetByIdentifierAndClaim);\n app.get('/:claim', serveAssetByClaim);\n};\n\n/***/ }),\n/* 86 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _require = __webpack_require__(4),\n sendGAServeEvent = _require.sendGAServeEvent;\n\nvar _require2 = __webpack_require__(23),\n determineResponseType = _require2.determineResponseType,\n logRequestData = _require2.logRequestData,\n getClaimIdAndServeAsset = _require2.getClaimIdAndServeAsset;\n\nvar lbryUri = __webpack_require__(24);\nvar handleShowRender = __webpack_require__(25);\nvar SERVE = 'SERVE';\n\n/*\n\n route to serve an asset or the react app via the claim name only\n\n*/\n\nvar serverAssetByClaim = function serverAssetByClaim(req, res) {\n var headers = req.headers,\n ip = req.ip,\n originalUrl = req.originalUrl,\n params = req.params;\n // decide if this is a show request\n\n var hasFileExtension = void 0;\n try {\n var _lbryUri$parseModifie = lbryUri.parseModifier(params.claim);\n\n hasFileExtension = _lbryUri$parseModifie.hasFileExtension;\n } catch (error) {\n return res.status(400).json({ success: false, message: error.message });\n }\n var responseType = determineResponseType(hasFileExtension, headers);\n if (responseType !== SERVE) {\n return handleShowRender(req, res);\n }\n // handle serve request\n // send google analytics\n sendGAServeEvent(headers, ip, originalUrl);\n // parse the claim\n var claimName = void 0;\n try {\n var _lbryUri$parseClaim = lbryUri.parseClaim(params.claim);\n\n claimName = _lbryUri$parseClaim.claimName;\n } catch (error) {\n return res.status(400).json({ success: false, message: error.message });\n }\n // log the request data for debugging\n logRequestData(responseType, claimName, null, null);\n // get the claim Id and then serve the asset\n getClaimIdAndServeAsset(null, null, claimName, null, originalUrl, ip, res);\n};\n\nmodule.exports = serverAssetByClaim;\n\n/***/ }),\n/* 87 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"redux-saga\");\n\n/***/ }),\n/* 88 */\n/***/ (function(module, exports) {\n\nmodule.exports = require(\"redux-saga/effects\");\n\n/***/ }),\n/* 89 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i[\"return\"]) _i[\"return\"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError(\"Invalid attempt to destructure non-iterable instance\"); } }; }();\n\nvar _require = __webpack_require__(4),\n sendGAServeEvent = _require.sendGAServeEvent;\n\nvar _require2 = __webpack_require__(23),\n determineResponseType = _require2.determineResponseType,\n flipClaimNameAndIdForBackwardsCompatibility = _require2.flipClaimNameAndIdForBackwardsCompatibility,\n logRequestData = _require2.logRequestData,\n getClaimIdAndServeAsset = _require2.getClaimIdAndServeAsset;\n\nvar lbryUri = __webpack_require__(24);\nvar handleShowRender = __webpack_require__(25);\n\nvar SERVE = 'SERVE';\n\n/*\n\n route to serve an asset or the react app via the claim name and an identifier\n\n*/\n\nvar serverAssetByIdentifierAndClaim = function serverAssetByIdentifierAndClaim(req, res) {\n var headers = req.headers,\n ip = req.ip,\n originalUrl = req.originalUrl,\n params = req.params;\n // decide if this is a show request\n\n var hasFileExtension = void 0;\n try {\n var _lbryUri$parseModifie = lbryUri.parseModifier(params.claim);\n\n hasFileExtension = _lbryUri$parseModifie.hasFileExtension;\n } catch (error) {\n return res.status(400).json({ success: false, message: error.message });\n }\n var responseType = determineResponseType(hasFileExtension, headers);\n if (responseType !== SERVE) {\n return handleShowRender(req, res);\n }\n // handle serve request\n // send google analytics\n sendGAServeEvent(headers, ip, originalUrl);\n // parse the claim\n var claimName = void 0;\n try {\n var _lbryUri$parseClaim = lbryUri.parseClaim(params.claim);\n\n claimName = _lbryUri$parseClaim.claimName;\n } catch (error) {\n return res.status(400).json({ success: false, message: error.message });\n }\n // parse the identifier\n var isChannel = void 0,\n channelName = void 0,\n channelClaimId = void 0,\n claimId = void 0;\n try {\n var _lbryUri$parseIdentif = lbryUri.parseIdentifier(params.identifier);\n\n isChannel = _lbryUri$parseIdentif.isChannel;\n channelName = _lbryUri$parseIdentif.channelName;\n channelClaimId = _lbryUri$parseIdentif.channelClaimId;\n claimId = _lbryUri$parseIdentif.claimId;\n } catch (error) {\n return res.status(400).json({ success: false, message: error.message });\n }\n if (!isChannel) {\n var _flipClaimNameAndIdFo = flipClaimNameAndIdForBackwardsCompatibility(claimId, claimName);\n\n var _flipClaimNameAndIdFo2 = _slicedToArray(_flipClaimNameAndIdFo, 2);\n\n claimId = _flipClaimNameAndIdFo2[0];\n claimName = _flipClaimNameAndIdFo2[1];\n }\n // log the request data for debugging\n logRequestData(responseType, claimName, channelName, claimId);\n // get the claim Id and then serve the asset\n getClaimIdAndServeAsset(channelName, channelClaimId, claimName, claimId, originalUrl, ip, res);\n};\n\nmodule.exports = serverAssetByIdentifierAndClaim;\n\n/***/ }),\n/* 90 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar handlePageRequest = __webpack_require__(91);\n\nmodule.exports = function (app) {\n app.get('*', handlePageRequest);\n};\n\n/***/ }),\n/* 91 */\n/***/ (function(module, exports, __webpack_require__) {\n\n\"use strict\";\n\n\nvar handlePageRender = __webpack_require__(14);\n\nvar sendReactApp = function sendReactApp(req, res) {\n handlePageRender(req, res);\n};\n\nmodule.exports = sendReactApp;\n\n/***/ })\n/******/ ]);\n\n\n// WEBPACK FOOTER //\n// index.js"," \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId]) {\n \t\t\treturn installedModules[moduleId].exports;\n \t\t}\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\ti: moduleId,\n \t\t\tl: false,\n \t\t\texports: {}\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.l = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// define getter function for harmony exports\n \t__webpack_require__.d = function(exports, name, getter) {\n \t\tif(!__webpack_require__.o(exports, name)) {\n \t\t\tObject.defineProperty(exports, name, {\n \t\t\t\tconfigurable: false,\n \t\t\t\tenumerable: true,\n \t\t\t\tget: getter\n \t\t\t});\n \t\t}\n \t};\n\n \t// getDefaultExport function for compatibility with non-harmony modules\n \t__webpack_require__.n = function(module) {\n \t\tvar getter = module && module.__esModule ?\n \t\t\tfunction getDefault() { return module['default']; } :\n \t\t\tfunction getModuleExports() { return module; };\n \t\t__webpack_require__.d(getter, 'a', getter);\n \t\treturn getter;\n \t};\n\n \t// Object.prototype.hasOwnProperty.call\n \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"/\";\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(__webpack_require__.s = 26);\n\n\n\n// WEBPACK FOOTER //\n// webpack/bootstrap 82ebf2ca30129849e4ef","module.exports = require(\"winston\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"winston\"\n// module id = 0\n// module chunks = 0","const logger = require('winston');\n\nmodule.exports = {\n handleErrorResponse: function (originalUrl, ip, error, res) {\n logger.error(`Error on ${originalUrl}`, module.exports.useObjectPropertiesIfNoKeys(error));\n const [status, message] = module.exports.returnErrorMessageAndStatus(error);\n res\n .status(status)\n .json(module.exports.createErrorResponsePayload(status, message));\n },\n returnErrorMessageAndStatus: function (error) {\n let status, message;\n // check for daemon being turned off\n if (error.code === 'ECONNREFUSED') {\n status = 503;\n message = 'Connection refused. The daemon may not be running.';\n // fallback for everything else\n } else {\n status = 400;\n if (error.message) {\n message = error.message;\n } else {\n message = error;\n };\n };\n return [status, message];\n },\n useObjectPropertiesIfNoKeys: function (err) {\n if (Object.keys(err).length === 0) {\n let newErrorObject = {};\n Object.getOwnPropertyNames(err).forEach((key) => {\n newErrorObject[key] = err[key];\n });\n return newErrorObject;\n }\n return err;\n },\n createErrorResponsePayload (status, message) {\n return {\n status,\n success: false,\n message,\n };\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/errorHandlers.js","const Certificate = require('models/certificate.js');\nconst Channel = require('models/channel.js');\nconst Claim = require('models/claim.js');\nconst File = require('models/file.js');\nconst Request = require('models/request.js');\nconst User = require('models/user.js');\n\nconst Sequelize = require('sequelize');\nconst logger = require('winston');\n\nconst {database, username, password} = require('mysqlConfig.js');\n\n// set sequelize options\nconst sequelize = new Sequelize(database, username, password, {\n host : 'localhost',\n dialect : 'mysql',\n dialectOptions: {decimalNumbers: true},\n logging : false,\n pool : {\n max : 5,\n min : 0,\n idle : 10000,\n acquire: 10000,\n },\n});\n\n// establish mysql connection\nsequelize\n .authenticate()\n .then(() => {\n logger.info('Sequelize has established mysql connection successfully.');\n })\n .catch(err => {\n logger.error('Sequelize was unable to connect to the database:', err);\n });\n\n// manually add each model to the db object (note: make this dynamic)\nconst db = {};\ndb['Certificate'] = sequelize.import('Certificate', Certificate);\ndb['Channel'] = sequelize.import('Channel', Channel);\ndb['Claim'] = sequelize.import('Claim', Claim);\ndb['File'] = sequelize.import('File', File);\ndb['Request'] = sequelize.import('Request', Request);\ndb['User'] = sequelize.import('User', User);\n\n// run model.association for each model in the db object that has an association\nlogger.info('associating db models...');\nObject.keys(db).forEach(modelName => {\n if (db[modelName].associate) {\n logger.info('Associating model:', modelName);\n db[modelName].associate(db);\n }\n});\n\n// add sequelize/Sequelize to db\ndb.sequelize = sequelize;\ndb.Sequelize = Sequelize;\n// add an 'upsert' method to the db object\ndb.upsert = (Model, values, condition, tableName) => {\n return Model\n .findOne({\n where: condition,\n })\n .then(obj => {\n if (obj) { // update\n logger.debug(`updating record in db.${tableName}`);\n return obj.update(values);\n } else { // insert\n logger.debug(`creating record in db.${tableName}`);\n return Model.create(values);\n }\n })\n .catch(function (error) {\n logger.error(`${tableName}.upsert error`, error);\n throw error;\n });\n};\n\nmodule.exports = db;\n\n\n\n// WEBPACK FOOTER //\n// ./server/models/index.js","function SiteConfig () {\n this.analytics = {\n googleId: 'default',\n };\n this.assetDefaults = {\n description: 'An asset published on Spee.ch',\n thumbnail : 'https://spee.ch/assets/img/video_thumb_default.png',\n title : 'Spee.ch',\n };\n this.auth = {\n sessionKey: 'default',\n };\n this.customComponents = {\n components: {},\n containers: {},\n pages : {},\n };\n this.details = {\n description: 'Open-source, decentralized image and video sharing.',\n host : 'default',\n port : 3000,\n title : 'Spee.ch',\n twitter : '@spee_ch',\n };\n this.publishing = {\n additionalClaimAddresses: [],\n disabled : false,\n disabledMessage : 'Please check back soon.',\n primaryClaimAddress : 'default',\n thumbnailChannel : 'default',\n thumbnailChannelId : 'default',\n uploadDirectory : '/home/lbry/Uploads',\n };\n this.routes = {};\n this.update = (config) => {\n if (!config) {\n return console.log('No site config received.');\n }\n const { analytics, assetDefaults, auth, customComponents, details, publishing, routes } = config;\n console.log('Configuring site details...');\n this.analytics = analytics;\n this.assetDefaults = assetDefaults;\n this.auth = auth;\n this.details = details;\n this.publishing = publishing;\n this.customComponents = customComponents;\n this.routes = routes;\n };\n};\n\nmodule.exports = new SiteConfig();\n\n\n\n// WEBPACK FOOTER //\n// ./config/siteConfig.js","const logger = require('winston');\nconst ua = require('universal-analytics');\nconst { analytics : { googleId }, details: { title } } = require('../../config/siteConfig.js');\n\nfunction createServeEventParams (headers, ip, originalUrl) {\n return {\n eventCategory : 'client requests',\n eventAction : 'serve request',\n eventLabel : originalUrl,\n ipOverride : ip,\n userAgentOverride: headers['user-agent'],\n };\n};\n\nfunction createPublishTimingEventParams (category, variable, label, startTime, endTime) {\n const duration = endTime - startTime;\n return {\n userTimingCategory : category,\n userTimingVariableName: variable,\n userTimingTime : duration,\n userTimingLabel : label,\n };\n};\n\nfunction sendGoogleAnalyticsEvent (ip, params) {\n const visitorId = ip.replace(/\\./g, '-');\n const visitor = ua(googleId, visitorId, { strictCidFormat: false, https: true });\n visitor.event(params, (err) => {\n if (err) {\n logger.error('Google Analytics Event Error >>', err);\n }\n });\n};\n\nfunction sendGoogleAnalyticsTiming (visitorId, params) {\n const visitor = ua(googleId, visitorId, { strictCidFormat: false, https: true });\n visitor.timing(params, (err) => {\n if (err) {\n logger.error('Google Analytics Event Error >>', err);\n }\n logger.debug(`Timing event successfully sent to google analytics`);\n });\n};\n\nmodule.exports = {\n sendGAServeEvent (headers, ip, originalUrl) {\n const params = createServeEventParams(headers, ip, originalUrl);\n sendGoogleAnalyticsEvent(ip, params);\n },\n sendGATimingEvent (category, variable, label, startTime, endTime) {\n const params = createPublishTimingEventParams(category, variable, label, startTime, endTime);\n sendGoogleAnalyticsTiming(title, params);\n },\n chooseGaLbrynetPublishLabel ({ channel_name: channelName, channel_id: channelId }) {\n return (channelName || channelId ? 'PUBLISH_IN_CHANNEL_CLAIM' : 'PUBLISH_ANONYMOUS_CLAIM');\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/googleAnalytics.js","const axios = require('axios');\nconst logger = require('winston');\nconst { api: { apiHost, apiPort } } = require('../../config/lbryConfig.js');\nconst lbryApiUri = 'http://' + apiHost + ':' + apiPort;\nconst { chooseGaLbrynetPublishLabel, sendGATimingEvent } = require('./googleAnalytics.js');\n\nconst handleLbrynetResponse = ({ data }, resolve, reject) => {\n logger.debug('lbry api data:', data);\n if (data.result) {\n // check for an error\n if (data.result.error) {\n logger.debug('Lbrynet api error:', data.result.error);\n reject(new Error(data.result.error));\n return;\n };\n resolve(data.result);\n return;\n }\n // fallback in case it just timed out\n reject(JSON.stringify(data));\n};\n\nmodule.exports = {\n publishClaim (publishParams) {\n logger.debug(`lbryApi >> Publishing claim to \"${publishParams.name}\"`);\n const gaStartTime = Date.now();\n return new Promise((resolve, reject) => {\n axios\n .post(lbryApiUri, {\n method: 'publish',\n params: publishParams,\n })\n .then(response => {\n sendGATimingEvent('lbrynet', 'publish', chooseGaLbrynetPublishLabel(publishParams), gaStartTime, Date.now());\n handleLbrynetResponse(response, resolve, reject);\n })\n .catch(error => {\n reject(error);\n });\n });\n },\n getClaim (uri) {\n logger.debug(`lbryApi >> Getting Claim for \"${uri}\"`);\n const gaStartTime = Date.now();\n return new Promise((resolve, reject) => {\n axios\n .post(lbryApiUri, {\n method: 'get',\n params: { uri, timeout: 20 },\n })\n .then(response => {\n sendGATimingEvent('lbrynet', 'getClaim', 'GET', gaStartTime, Date.now());\n handleLbrynetResponse(response, resolve, reject);\n })\n .catch(error => {\n reject(error);\n });\n });\n },\n getClaimList (claimName) {\n logger.debug(`lbryApi >> Getting claim_list for \"${claimName}\"`);\n const gaStartTime = Date.now();\n return new Promise((resolve, reject) => {\n axios\n .post(lbryApiUri, {\n method: 'claim_list',\n params: { name: claimName },\n })\n .then(response => {\n sendGATimingEvent('lbrynet', 'getClaimList', 'CLAIM_LIST', gaStartTime, Date.now());\n handleLbrynetResponse(response, resolve, reject);\n })\n .catch(error => {\n reject(error);\n });\n });\n },\n resolveUri (uri) {\n logger.debug(`lbryApi >> Resolving URI for \"${uri}\"`);\n const gaStartTime = Date.now();\n return new Promise((resolve, reject) => {\n axios\n .post(lbryApiUri, {\n method: 'resolve',\n params: { uri },\n })\n .then(({ data }) => {\n sendGATimingEvent('lbrynet', 'resolveUri', 'RESOLVE', gaStartTime, Date.now());\n if (data.result[uri].error) { // check for errors\n reject(data.result[uri].error);\n } else { // if no errors, resolve\n resolve(data.result[uri]);\n }\n })\n .catch(error => {\n reject(error);\n });\n });\n },\n getDownloadDirectory () {\n logger.debug('lbryApi >> Retrieving the download directory path from lbry daemon...');\n const gaStartTime = Date.now();\n return new Promise((resolve, reject) => {\n axios\n .post(lbryApiUri, {\n method: 'settings_get',\n })\n .then(({ data }) => {\n sendGATimingEvent('lbrynet', 'getDownloadDirectory', 'SETTINGS_GET', gaStartTime, Date.now());\n if (data.result) {\n resolve(data.result.download_directory);\n } else {\n return new Error('Successfully connected to lbry daemon, but unable to retrieve the download directory.');\n }\n })\n .catch(error => {\n logger.error('Lbrynet Error:', error);\n resolve('/home/lbry/Downloads/');\n });\n });\n },\n createChannel (name) {\n logger.debug(`lbryApi >> Creating channel for ${name}...`);\n const gaStartTime = Date.now();\n return new Promise((resolve, reject) => {\n axios\n .post(lbryApiUri, {\n method: 'channel_new',\n params: {\n channel_name: name,\n amount : 0.1,\n },\n })\n .then(response => {\n sendGATimingEvent('lbrynet', 'createChannel', 'CHANNEL_NEW', gaStartTime, Date.now());\n handleLbrynetResponse(response, resolve, reject);\n })\n .catch(error => {\n reject(error);\n });\n });\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/lbryApi.js","const db = require('models');\nconst logger = require('winston');\nconst { returnPaginatedChannelClaims } = require('helpers/channelPagination.js');\n\nconst NO_CHANNEL = 'NO_CHANNEL';\nconst NO_CLAIM = 'NO_CLAIM';\nconst NO_FILE = 'NO_FILE';\n\nmodule.exports = {\n getClaimId (channelName, channelClaimId, name, claimId) {\n if (channelName) {\n return module.exports.getClaimIdByChannel(channelName, channelClaimId, name);\n } else {\n return module.exports.getClaimIdByClaim(name, claimId);\n }\n },\n getClaimIdByClaim (claimName, claimId) {\n logger.debug(`getClaimIdByClaim(${claimName}, ${claimId})`);\n return new Promise((resolve, reject) => {\n db.Claim.getLongClaimId(claimName, claimId)\n .then(longClaimId => {\n if (!longClaimId) {\n resolve(NO_CLAIM);\n }\n resolve(longClaimId);\n })\n .catch(error => {\n reject(error);\n });\n });\n },\n getClaimIdByChannel (channelName, channelClaimId, claimName) {\n logger.debug(`getClaimIdByChannel(${channelName}, ${channelClaimId}, ${claimName})`);\n return new Promise((resolve, reject) => {\n db.Certificate.getLongChannelId(channelName, channelClaimId) // 1. get the long channel id\n .then(longChannelId => {\n if (!longChannelId) {\n return [null, null];\n }\n return Promise.all([longChannelId, db.Claim.getClaimIdByLongChannelId(longChannelId, claimName)]); // 2. get the long claim id\n })\n .then(([longChannelId, longClaimId]) => {\n if (!longChannelId) {\n return resolve(NO_CHANNEL);\n }\n if (!longClaimId) {\n return resolve(NO_CLAIM);\n }\n resolve(longClaimId);\n })\n .catch(error => {\n reject(error);\n });\n });\n },\n getChannelData (channelName, channelClaimId, page) {\n return new Promise((resolve, reject) => {\n // 1. get the long channel Id (make sure channel exists)\n db.Certificate.getLongChannelId(channelName, channelClaimId)\n .then(longChannelClaimId => {\n if (!longChannelClaimId) {\n return [null, null, null];\n }\n // 2. get the short ID and all claims for that channel\n return Promise.all([longChannelClaimId, db.Certificate.getShortChannelIdFromLongChannelId(longChannelClaimId, channelName)]);\n })\n .then(([longChannelClaimId, shortChannelClaimId]) => {\n if (!longChannelClaimId) {\n return resolve(NO_CHANNEL);\n }\n // 3. return all the channel information\n resolve({\n channelName,\n longChannelClaimId,\n shortChannelClaimId,\n });\n })\n .catch(error => {\n reject(error);\n });\n });\n },\n getChannelClaims (channelName, channelClaimId, page) {\n return new Promise((resolve, reject) => {\n // 1. get the long channel Id (make sure channel exists)\n db.Certificate.getLongChannelId(channelName, channelClaimId)\n .then(longChannelClaimId => {\n if (!longChannelClaimId) {\n return [null, null, null];\n }\n // 2. get the short ID and all claims for that channel\n return Promise.all([longChannelClaimId, db.Claim.getAllChannelClaims(longChannelClaimId)]);\n })\n .then(([longChannelClaimId, channelClaimsArray]) => {\n if (!longChannelClaimId) {\n return resolve(NO_CHANNEL);\n }\n // 3. format the data for the view, including pagination\n let paginatedChannelViewData = returnPaginatedChannelClaims(channelName, longChannelClaimId, channelClaimsArray, page);\n // 4. return all the channel information and contents\n resolve(paginatedChannelViewData);\n })\n .catch(error => {\n reject(error);\n });\n });\n },\n getLocalFileRecord (claimId, name) {\n return db.File.findOne({where: {claimId, name}})\n .then(file => {\n if (!file) {\n return NO_FILE;\n }\n return file.dataValues;\n });\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/controllers/serveController.js","const passport = require('passport');\nconst localLoginStrategy = require('./local-login.js');\nconst localSignupStrategy = require('./local-signup.js');\nconst { serializeSpeechUser, deserializeSpeechUser } = require('helpers/authHelpers.js');\n\npassport.deserializeUser(deserializeSpeechUser);\npassport.serializeUser(serializeSpeechUser);\npassport.use('local-login', localLoginStrategy);\npassport.use('local-signup', localSignupStrategy);\n\nmodule.exports = passport;\n\n\n\n// WEBPACK FOOTER //\n// ./server/speechPassport/index.js","const logger = require('winston');\nconst db = require('models');\nconst lbryApi = require('helpers/lbryApi.js');\nconst publishHelpers = require('helpers/publishHelpers.js');\nconst { publishing: { primaryClaimAddress, additionalClaimAddresses } } = require('siteConfig.js');\nconst Sequelize = require('sequelize');\nconst Op = Sequelize.Op;\n\nmodule.exports = {\n publish (publishParams, fileName, fileType) {\n return new Promise((resolve, reject) => {\n let publishResults, certificateId, channelName;\n // publish the file\n return lbryApi.publishClaim(publishParams)\n .then(tx => {\n logger.info(`Successfully published ${publishParams.name} ${fileName}`, tx);\n publishResults = tx;\n // get the channel information\n if (publishParams.channel_name) {\n logger.debug(`this claim was published in channel: ${publishParams.channel_name}`);\n return db.Channel.findOne({\n where: {\n channelName: publishParams.channel_name,\n },\n });\n } else {\n logger.debug('this claim was not published in a channel');\n return null;\n }\n })\n .then(channel => {\n // set channel information\n certificateId = null;\n channelName = null;\n if (channel) {\n certificateId = channel.channelClaimId;\n channelName = channel.channelName;\n }\n logger.debug(`certificateId: ${certificateId}`);\n })\n .then(() => {\n // create the File record\n const fileRecord = {\n name : publishParams.name,\n claimId : publishResults.claim_id,\n title : publishParams.metadata.title,\n description: publishParams.metadata.description,\n address : publishParams.claim_address,\n outpoint : `${publishResults.txid}:${publishResults.nout}`,\n height : 0,\n fileName,\n filePath : publishParams.file_path,\n fileType,\n nsfw : publishParams.metadata.nsfw,\n };\n // create the Claim record\n const claimRecord = {\n name : publishParams.name,\n claimId : publishResults.claim_id,\n title : publishParams.metadata.title,\n description: publishParams.metadata.description,\n address : publishParams.claim_address,\n thumbnail : publishParams.metadata.thumbnail,\n outpoint : `${publishResults.txid}:${publishResults.nout}`,\n height : 0,\n contentType: fileType,\n nsfw : publishParams.metadata.nsfw,\n amount : publishParams.bid,\n certificateId,\n channelName,\n };\n // upsert criteria\n const upsertCriteria = {\n name : publishParams.name,\n claimId: publishResults.claim_id,\n };\n // upsert the records\n return Promise.all([db.upsert(db.File, fileRecord, upsertCriteria, 'File'), db.upsert(db.Claim, claimRecord, upsertCriteria, 'Claim')]);\n })\n .then(([file, claim]) => {\n logger.debug('File and Claim records successfully created');\n return Promise.all([file.setClaim(claim), claim.setFile(file)]);\n })\n .then(() => {\n logger.debug('File and Claim records successfully associated');\n resolve(publishResults); // resolve the promise with the result from lbryApi.publishClaim;\n })\n .catch(error => {\n logger.error('PUBLISH ERROR', error);\n publishHelpers.deleteTemporaryFile(publishParams.file_path); // delete the local file\n reject(error);\n });\n });\n },\n claimNameIsAvailable (name) {\n const claimAddresses = additionalClaimAddresses || [];\n claimAddresses.push(primaryClaimAddress);\n // find any records where the name is used\n return db.Claim\n .findAll({\n attributes: ['address'],\n where : {\n name,\n address: {\n [Op.or]: claimAddresses,\n },\n },\n })\n .then(result => {\n if (result.length >= 1) {\n throw new Error('That claim is already in use');\n };\n return name;\n })\n .catch(error => {\n throw error;\n });\n },\n checkChannelAvailability (name) {\n return db.Channel\n .findAll({\n where: { channelName: name },\n })\n .then(result => {\n if (result.length >= 1) {\n throw new Error('That channel has already been claimed');\n }\n return name;\n })\n .catch(error => {\n throw error;\n });\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/controllers/publishController.js","const logger = require('winston');\nconst fs = require('fs');\n\nconst { details, publishing } = require('../../config/siteConfig.js');\n\nmodule.exports = {\n parsePublishApiRequestBody ({name, nsfw, license, title, description, thumbnail}) {\n // validate name\n if (!name) {\n throw new Error('no name field found in request');\n }\n const invalidNameCharacters = /[^A-Za-z0-9,-]/.exec(name);\n if (invalidNameCharacters) {\n throw new Error('The claim name you provided is not allowed. Only the following characters are allowed: A-Z, a-z, 0-9, and \"-\"');\n }\n // optional parameters\n nsfw = (nsfw === 'true');\n license = license || null;\n title = title || null;\n description = description || null;\n thumbnail = thumbnail || null;\n // return results\n return {\n name,\n nsfw,\n license,\n title,\n description,\n thumbnail,\n };\n },\n parsePublishApiRequestFiles ({file, thumbnail}) {\n // make sure a file was provided\n if (!file) {\n throw new Error('no file with key of [file] found in request');\n }\n if (!file.path) {\n throw new Error('no file path found');\n }\n if (!file.type) {\n throw new Error('no file type found');\n }\n if (!file.size) {\n throw new Error('no file type found');\n }\n // validate the file name\n if (/'/.test(file.name)) {\n throw new Error('apostrophes are not allowed in the file name');\n }\n // validate the file\n module.exports.validateFileTypeAndSize(file);\n // return results\n return {\n fileName : file.name,\n filePath : file.path,\n fileType : file.type,\n thumbnailFileName: (thumbnail ? thumbnail.name : null),\n thumbnailFilePath: (thumbnail ? thumbnail.path : null),\n thumbnailFileType: (thumbnail ? thumbnail.type : null),\n };\n },\n validateFileTypeAndSize (file) {\n // check file type and size\n switch (file.type) {\n case 'image/jpeg':\n case 'image/jpg':\n case 'image/png':\n if (file.size > 10000000) {\n logger.debug('publish > file validation > .jpeg/.jpg/.png was too big');\n throw new Error('Sorry, images are limited to 10 megabytes.');\n }\n break;\n case 'image/gif':\n if (file.size > 50000000) {\n logger.debug('publish > file validation > .gif was too big');\n throw new Error('Sorry, .gifs are limited to 50 megabytes.');\n }\n break;\n case 'video/mp4':\n if (file.size > 50000000) {\n logger.debug('publish > file validation > .mp4 was too big');\n throw new Error('Sorry, videos are limited to 50 megabytes.');\n }\n break;\n default:\n logger.debug('publish > file validation > unrecognized file type');\n throw new Error('The ' + file.type + ' content type is not supported. Only, .jpeg, .png, .gif, and .mp4 files are currently supported.');\n }\n return file;\n },\n createBasicPublishParams (filePath, name, title, description, license, nsfw, thumbnail) {\n logger.debug(`Creating Publish Parameters`);\n // provide defaults for title\n if (title === null || title.trim() === '') {\n title = name;\n }\n // provide default for description\n if (description === null || description.trim() === '') {\n description = '';\n }\n // provide default for license\n if (license === null || license.trim() === '') {\n license = ' '; // default to empty string\n }\n // create the publish params\n const publishParams = {\n name,\n file_path: filePath,\n bid : 0.01,\n metadata : {\n description,\n title,\n author : details.title,\n language: 'en',\n license,\n nsfw,\n },\n claim_address: publishing.primaryClaimAddress,\n };\n // add thumbnail to channel if video\n if (thumbnail) {\n publishParams['metadata']['thumbnail'] = thumbnail;\n }\n return publishParams;\n },\n createThumbnailPublishParams (thumbnailFilePath, claimName, license, nsfw) {\n if (!thumbnailFilePath) {\n return;\n }\n logger.debug(`Creating Thumbnail Publish Parameters`);\n // create the publish params\n return {\n name : `${claimName}-thumb`,\n file_path: thumbnailFilePath,\n bid : 0.01,\n metadata : {\n title : `${claimName} thumbnail`,\n description: `a thumbnail for ${claimName}`,\n author : details.title,\n language : 'en',\n license,\n nsfw,\n },\n claim_address: publishing.primaryClaimAddress,\n channel_name : publishing.thumbnailChannel,\n channel_id : publishing.thumbnailChannelId,\n };\n },\n deleteTemporaryFile (filePath) {\n fs.unlink(filePath, err => {\n if (err) {\n logger.error(`error deleting temporary file ${filePath}`);\n throw err;\n }\n logger.debug(`successfully deleted ${filePath}`);\n });\n },\n addGetResultsToFileData (fileInfo, getResult) {\n fileInfo.fileName = getResult.file_name;\n fileInfo.filePath = getResult.download_path;\n return fileInfo;\n },\n createFileData ({ name, claimId, outpoint, height, address, nsfw, contentType }) {\n return {\n name,\n claimId,\n outpoint,\n height,\n address,\n fileName: '',\n filePath: '',\n fileType: contentType,\n nsfw,\n };\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/publishHelpers.js","const logger = require('winston');\n\nfunction mysql () {\n this.database = 'default';\n this.username = 'default';\n this.password = 'default';\n this.update = (config) => {\n if (!config) {\n return logger.warn('No MySQL config received.');\n }\n // configure credentials\n logger.info('configuring mysql...');\n const { database, username, password } = config;\n this.database = database;\n this.username = username;\n this.password = password;\n };\n};\n\nmodule.exports = new mysql();\n\n\n\n// WEBPACK FOOTER //\n// ./config/mysqlConfig.js","module.exports = require(\"passport-local\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"passport-local\"\n// module id = 11\n// module chunks = 0","module.exports = {\n returnShortId: function (claimsArray, longId) {\n let claimIndex;\n let shortId = longId.substring(0, 1); // default short id is the first letter\n let shortIdLength = 0;\n // find the index of this claim id\n claimIndex = claimsArray.findIndex(element => {\n return element.claimId === longId;\n });\n if (claimIndex < 0) {\n throw new Error('claim id not found in claims list');\n }\n // get an array of all claims with lower height\n let possibleMatches = claimsArray.slice(0, claimIndex);\n // remove certificates with the same prefixes until none are left.\n while (possibleMatches.length > 0) {\n shortIdLength += 1;\n shortId = longId.substring(0, shortIdLength);\n possibleMatches = possibleMatches.filter(element => {\n return (element.claimId && (element.claimId.substring(0, shortIdLength) === shortId));\n });\n }\n return shortId;\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/sequelizeHelpers.js","module.exports = require(\"sequelize\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"sequelize\"\n// module id = 13\n// module chunks = 0","import React from 'react';\nimport { renderToString } from 'react-dom/server';\nimport { createStore } from 'redux';\nimport { Provider } from 'react-redux';\nimport { StaticRouter } from 'react-router-dom';\nimport { Reducers, GAListener, App } from 'spee.ch-components';\nimport renderFullPage from './renderFullPage.js';\nimport Helmet from 'react-helmet';\n\nconst siteConfig = require('siteConfig.js');\n\nmodule.exports = (req, res) => {\n let context = {};\n\n // customize the reducer by passing in intial state configs\n const customizedReducers = Reducers(siteConfig);\n\n // create a new Redux store instance\n const store = createStore(customizedReducers);\n\n // render component to a string\n const html = renderToString(\n \n \n \n \n \n \n \n );\n\n // get head tags from helmet\n const helmet = Helmet.renderStatic();\n\n // check for a redirect\n if (context.url) {\n // Somewhere a `` was rendered\n return res.redirect(301, context.url);\n } else {\n // we're good, send the response\n }\n\n // get the initial state from our Redux store\n const preloadedState = store.getState();\n\n // send the rendered page back to the client\n res.send(renderFullPage(helmet, html, preloadedState));\n\n console.log('hello from spee.ch handlePageRender.jsx');\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/handlePageRender.jsx","module.exports = require(\"react\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"react\"\n// module id = 15\n// module chunks = 0","module.exports = require(\"react-dom/server\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"react-dom/server\"\n// module id = 16\n// module chunks = 0","module.exports = require(\"redux\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"redux\"\n// module id = 17\n// module chunks = 0","module.exports = require(\"react-redux\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"react-redux\"\n// module id = 18\n// module chunks = 0","module.exports = require(\"react-router-dom\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"react-router-dom\"\n// module id = 19\n// module chunks = 0","module.exports = require(\"spee.ch-components\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"spee.ch-components\"\n// module id = 20\n// module chunks = 0","module.exports = (helmet, html, preloadedState) => {\n // take the html and preloadedState and return the full page\n return `\n \n \n \n \n \n \n \n ${helmet.title.toString()}\n ${helmet.meta.toString()}\n ${helmet.link.toString()}\n \n \n \n \n \n \n \n \n
\n
${html}
\n
\n \n \n \n \n `;\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/renderFullPage.js","module.exports = require(\"react-helmet\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"react-helmet\"\n// module id = 22\n// module chunks = 0","const logger = require('winston');\nconst { getClaimId, getLocalFileRecord } = require('../controllers/serveController.js');\nconst { handleErrorResponse } = require('./errorHandlers.js');\n\nconst SERVE = 'SERVE';\nconst SHOW = 'SHOW';\nconst NO_FILE = 'NO_FILE';\nconst NO_CHANNEL = 'NO_CHANNEL';\nconst NO_CLAIM = 'NO_CLAIM';\n\nfunction clientAcceptsHtml ({accept}) {\n return accept && accept.match(/text\\/html/);\n};\n\nfunction requestIsFromBrowser (headers) {\n return headers['user-agent'] && headers['user-agent'].match(/Mozilla/);\n};\n\nfunction clientWantsAsset ({accept, range}) {\n const imageIsWanted = accept && accept.match(/image\\/.*/) && !accept.match(/text\\/html/) && !accept.match(/text\\/\\*/);\n const videoIsWanted = accept && range;\n return imageIsWanted || videoIsWanted;\n};\n\nfunction isValidClaimId (claimId) {\n return ((claimId.length === 40) && !/[^A-Za-z0-9]/g.test(claimId));\n};\n\nfunction isValidShortId (claimId) {\n return claimId.length === 1; // it should really evaluate the short url itself\n};\n\nfunction isValidShortIdOrClaimId (input) {\n return (isValidClaimId(input) || isValidShortId(input));\n};\n\nfunction serveAssetToClient (claimId, name, res) {\n return getLocalFileRecord(claimId, name)\n .then(fileRecord => {\n // check that a local record was found\n if (fileRecord === NO_FILE) {\n return res.status(307).redirect(`/api/claim/get/${name}/${claimId}`);\n }\n // serve the file\n const {filePath, fileType} = fileRecord;\n logger.verbose(`serving file: ${filePath}`);\n const sendFileOptions = {\n headers: {\n 'X-Content-Type-Options': 'nosniff',\n 'Content-Type' : fileType || 'image/jpeg',\n },\n };\n res.status(200).sendFile(filePath, sendFileOptions);\n })\n .catch(error => {\n throw error;\n });\n};\n\nmodule.exports = {\n getClaimIdAndServeAsset (channelName, channelClaimId, claimName, claimId, originalUrl, ip, res) {\n // get the claim Id and then serve the asset\n getClaimId(channelName, channelClaimId, claimName, claimId)\n .then(fullClaimId => {\n if (fullClaimId === NO_CLAIM) {\n return res.status(404).json({success: false, message: 'no claim id could be found'});\n } else if (fullClaimId === NO_CHANNEL) {\n return res.status(404).json({success: false, message: 'no channel id could be found'});\n }\n serveAssetToClient(fullClaimId, claimName, res);\n // postToStats(responseType, originalUrl, ip, claimName, fullClaimId, 'success');\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n // postToStats(responseType, originalUrl, ip, claimName, fullClaimId, 'fail');\n });\n },\n determineResponseType (hasFileExtension, headers) {\n let responseType;\n if (hasFileExtension) {\n responseType = SERVE; // assume a serve request if file extension is present\n if (clientAcceptsHtml(headers)) { // if the request comes from a browser, change it to a show request\n responseType = SHOW;\n }\n } else {\n responseType = SHOW;\n if (clientWantsAsset(headers) && requestIsFromBrowser(headers)) { // this is in case someone embeds a show url\n logger.debug('Show request came from browser but wants an image/video. Changing response to serve...');\n responseType = SERVE;\n }\n }\n return responseType;\n },\n flipClaimNameAndIdForBackwardsCompatibility (identifier, name) {\n // this is a patch for backwards compatability with '/name/claim_id' url format\n if (isValidShortIdOrClaimId(name) && !isValidShortIdOrClaimId(identifier)) {\n const tempName = name;\n name = identifier;\n identifier = tempName;\n }\n return [identifier, name];\n },\n logRequestData (responseType, claimName, channelName, claimId) {\n logger.debug('responseType ===', responseType);\n logger.debug('claim name === ', claimName);\n logger.debug('channel name ===', channelName);\n logger.debug('claim id ===', claimId);\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/serveHelpers.js","const logger = require('winston');\n\nmodule.exports = {\n REGEXP_INVALID_CLAIM : /[^A-Za-z0-9-]/g,\n REGEXP_INVALID_CHANNEL: /[^A-Za-z0-9-@]/g,\n REGEXP_ADDRESS : /^b(?=[^0OIl]{32,33})[0-9A-Za-z]{32,33}$/,\n CHANNEL_CHAR : '@',\n parseIdentifier : function (identifier) {\n logger.debug('parsing identifier:', identifier);\n const componentsRegex = new RegExp(\n '([^:$#/]*)' + // value (stops at the first separator or end)\n '([:$#]?)([^/]*)' // modifier separator, modifier (stops at the first path separator or end)\n );\n const [proto, value, modifierSeperator, modifier] = componentsRegex\n .exec(identifier)\n .map(match => match || null);\n logger.debug(`${proto}, ${value}, ${modifierSeperator}, ${modifier}`);\n\n // Validate and process name\n if (!value) {\n throw new Error(`Check your url. No channel name provided before \"${modifierSeperator}\"`);\n }\n const isChannel = value.startsWith(module.exports.CHANNEL_CHAR);\n const channelName = isChannel ? value : null;\n let claimId;\n if (isChannel) {\n if (!channelName) {\n throw new Error('No channel name after @.');\n }\n const nameBadChars = (channelName).match(module.exports.REGEXP_INVALID_CHANNEL);\n if (nameBadChars) {\n throw new Error(`Invalid characters in channel name: ${nameBadChars.join(', ')}.`);\n }\n } else {\n claimId = value;\n }\n\n // Validate and process modifier\n let channelClaimId;\n if (modifierSeperator) {\n if (!modifier) {\n throw new Error(`No modifier provided after separator \"${modifierSeperator}\"`);\n }\n\n if (modifierSeperator === ':') {\n channelClaimId = modifier;\n } else {\n throw new Error(`The \"${modifierSeperator}\" modifier is not currently supported`);\n }\n }\n return {\n isChannel,\n channelName,\n channelClaimId,\n claimId,\n };\n },\n parseClaim: function (claim) {\n logger.debug('parsing name:', claim);\n const componentsRegex = new RegExp(\n '([^:$#/.]*)' + // name (stops at the first modifier)\n '([:$#.]?)([^/]*)' // modifier separator, modifier (stops at the first path separator or end)\n );\n const [proto, claimName, modifierSeperator, modifier] = componentsRegex\n .exec(claim)\n .map(match => match || null);\n logger.debug(`${proto}, ${claimName}, ${modifierSeperator}, ${modifier}`);\n\n // Validate and process name\n if (!claimName) {\n throw new Error('No claim name provided before .');\n }\n const nameBadChars = (claimName).match(module.exports.REGEXP_INVALID_CLAIM);\n if (nameBadChars) {\n throw new Error(`Invalid characters in claim name: ${nameBadChars.join(', ')}.`);\n }\n // Validate and process modifier\n if (modifierSeperator) {\n if (!modifier) {\n throw new Error(`No file extension provided after separator ${modifierSeperator}.`);\n }\n if (modifierSeperator !== '.') {\n throw new Error(`The ${modifierSeperator} modifier is not supported in the claim name`);\n }\n }\n // return results\n return {\n claimName,\n };\n },\n parseModifier: function (claim) {\n logger.debug('parsing modifier:', claim);\n const componentsRegex = new RegExp(\n '([^:$#/.]*)' + // name (stops at the first modifier)\n '([:$#.]?)([^/]*)' // modifier separator, modifier (stops at the first path separator or end)\n );\n const [proto, claimName, modifierSeperator, modifier] = componentsRegex\n .exec(claim)\n .map(match => match || null);\n logger.debug(`${proto}, ${claimName}, ${modifierSeperator}, ${modifier}`);\n // Validate and process modifier\n let hasFileExtension = false;\n if (modifierSeperator) {\n hasFileExtension = true;\n }\n return {\n hasFileExtension,\n };\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/lbryUri.js","import React from 'react';\nimport { renderToString } from 'react-dom/server';\nimport { createStore, applyMiddleware } from 'redux';\nimport { Provider } from 'react-redux';\nimport { StaticRouter } from 'react-router-dom';\nimport renderFullPage from './renderFullPage';\nimport createSagaMiddleware from 'redux-saga';\nimport { call } from 'redux-saga/effects';\nimport { Reducers, GAListener, App, Sagas, Actions } from 'spee.ch-components';\n\nimport Helmet from 'react-helmet';\n\n// configure the reducers by passing initial state configs\nconst siteConfig = require('siteConfig.js');\nconst customizedReducers = Reducers(siteConfig);\n\nconst returnSagaWithParams = (saga, params) => {\n return function * () {\n yield call(saga, params);\n };\n};\n\nmodule.exports = (req, res) => {\n let context = {};\n\n // create and apply middleware\n const sagaMiddleware = createSagaMiddleware();\n const middleware = applyMiddleware(sagaMiddleware);\n\n // create a new Redux store instance\n const store = createStore(customizedReducers, middleware);\n\n // create saga\n const action = Actions.onHandleShowPageUri(req.params);\n const saga = returnSagaWithParams(Sagas.handleShowPageUri, action);\n\n // run the saga middleware\n sagaMiddleware\n .run(saga)\n .done\n .then(() => {\n // render component to a string\n const html = renderToString(\n \n \n \n \n \n \n \n );\n\n // get head tags from helmet\n const helmet = Helmet.renderStatic();\n\n // check for a redirect\n if (context.url) {\n return res.redirect(301, context.url);\n }\n\n // get the initial state from our Redux store\n const preloadedState = store.getState();\n\n // send the rendered page back to the client\n res.send(renderFullPage(helmet, html, preloadedState));\n });\n\n console.log('hello from spee.ch handleShowRender.jsx');\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/handleShowRender.jsx","module.exports = require(\"babel-polyfill\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"babel-polyfill\"\n// module id = 27\n// module chunks = 0","module.exports = require(\"whatwg-fetch\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"whatwg-fetch\"\n// module id = 28\n// module chunks = 0","const Server = require('./server');\n\nconst exports = {\n Server,\n};\n\nmodule.exports = exports;\n\n\n\n// WEBPACK FOOTER //\n// ./speech.js","// app dependencies\nconst express = require('express');\nconst bodyParser = require('body-parser');\nconst expressHandlebars = require('express-handlebars');\nconst Handlebars = require('handlebars');\nconst helmet = require('helmet');\nconst cookieSession = require('cookie-session');\nconst http = require('http');\nconst logger = require('winston');\nconst requestLogger = require('middleware/requestLogger.js');\nconst Path = require('path');\nconst loggerConfig = require('loggerConfig.js');\nconst mysqlConfig = require('mysqlConfig.js');\nconst siteConfig = require('siteConfig.js');\nconst slackConfig = require('slackConfig.js');\n\nfunction Server () {\n this.configureLogger = (userConfig) => {\n loggerConfig.update(userConfig);\n };\n this.configureMysql = (userConfig) => {\n mysqlConfig.update(userConfig);\n };\n this.configureSiteDetails = (userConfig) => {\n siteConfig.update(userConfig);\n };\n this.configureSlack = (userConfig) => {\n slackConfig.update(userConfig);\n };\n this.configureClientBundle = () => {\n logger.debug('configure the client here by passing in the bundle and configuring it, or better yet: taking in the components to use dynamically from here.');\n };\n this.configureModels = () => {\n logger.debug('here is where you could add/overwrite the default models')\n };\n this.configureRoutes = () => {\n logger.debug('here is where you could add/overwrite the default routes')\n };\n this.createApp = () => {\n // create an Express application\n const app = express();\n\n // trust the proxy to get ip address for us\n app.enable('trust proxy');\n\n /* add middleware */\n // set HTTP headers to protect against well-known web vulnerabilties\n app.use(helmet());\n // 'express.static' to serve static files from public directory\n if (siteConfig.routes.publicFolder) {\n // take in a different public folder, so it can serve it's own bundle if needed\n const publicFolder = Path.resolve(process.cwd(), siteConfig.routes.publicFolder);\n app.use('/static', express.static(publicFolder));\n logger.info('serving static files from custom path:', publicFolder);\n } else {\n const publicPath = Path.resolve(__dirname, 'public');\n app.use('/static', express.static(publicPath));\n logger.info('serving static files from default path:', publicPath);\n };\n // 'body parser' for parsing application/json\n app.use(bodyParser.json());\n // 'body parser' for parsing application/x-www-form-urlencoded\n app.use(bodyParser.urlencoded({ extended: true }));\n\n // add custom middleware (note: build out to accept dynamically use what is in server/middleware/\n app.use(requestLogger);\n\n // configure passport\n const speechPassport = require('speechPassport');\n // initialize passport\n const sessionKey = siteConfig.auth.sessionKey;\n app.use(cookieSession({\n name : 'session',\n keys : [sessionKey],\n maxAge: 24 * 60 * 60 * 1000, // i.e. 24 hours\n }));\n app.use(speechPassport.initialize());\n app.use(speechPassport.session());\n\n // configure handlebars & register it with express app\n const hbs = expressHandlebars.create({\n defaultLayout: 'embed',\n handlebars : Handlebars,\n });\n app.engine('handlebars', hbs.engine);\n app.set('view engine', 'handlebars');\n\n // set the routes on the app\n require('./routes/auth/')(app);\n require('./routes/api/')(app);\n require('./routes/pages/')(app);\n require('./routes/assets/')(app);\n require('./routes/fallback/')(app);\n\n this.app = app;\n };\n this.initialize = () => {\n this.createApp();\n this.server = http.Server(this.app);\n };\n this.start = () => {\n const db = require('models');\n const PORT = siteConfig.details.port;\n // sync sequelize\n db.sequelize.sync()\n // start the server\n .then(() => {\n this.server.listen(PORT, () => {\n logger.info(`Server is listening on PORT ${PORT}`);\n });\n })\n .catch((error) => {\n logger.error(`Startup Error:`, error);\n });\n };\n};\n\nmodule.exports = Server;\n\n\n\n// WEBPACK FOOTER //\n// ./server/index.js","module.exports = require(\"express\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"express\"\n// module id = 31\n// module chunks = 0","module.exports = require(\"body-parser\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"body-parser\"\n// module id = 32\n// module chunks = 0","module.exports = require(\"express-handlebars\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"express-handlebars\"\n// module id = 33\n// module chunks = 0","module.exports = require(\"handlebars\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"handlebars\"\n// module id = 34\n// module chunks = 0","module.exports = require(\"helmet\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"helmet\"\n// module id = 35\n// module chunks = 0","module.exports = require(\"cookie-session\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"cookie-session\"\n// module id = 36\n// module chunks = 0","module.exports = require(\"http\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"http\"\n// module id = 37\n// module chunks = 0","const logger = require('winston');\n\nconst requestLogger = (req, res, next) => { // custom logging middleware to log all incoming http requests\n logger.verbose(`Request on ${req.originalUrl} from ${req.ip}`);\n next();\n};\n\nmodule.exports = requestLogger;\n\n\n\n// WEBPACK FOOTER //\n// ./server/middleware/requestLogger.js","module.exports = require(\"path\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"path\"\n// module id = 39\n// module chunks = 0","const logger = require('winston');\n\nfunction LoggerConfig () {\n this.logLevel = 'debug';\n this.update = (config) => {\n if (!config) {\n return logger.warn('No logger config received.');\n }\n logger.info('configuring winston logger...');\n // update values with local config params\n const {logLevel} = config;\n this.logLevel = logLevel;\n // configure the winston logger\n logger.configure({\n transports: [\n new (logger.transports.Console)({\n level : this.logLevel,\n timestamp : false,\n colorize : true,\n prettyPrint : true,\n handleExceptions : true,\n humanReadableUnhandledException: true,\n }),\n ],\n });\n // test all the log levels\n logger.info('testing winston log levels...');\n logger.error('Level 0');\n logger.warn('Level 1');\n logger.info('Level 2');\n logger.verbose('Level 3');\n logger.debug('Level 4');\n logger.silly('Level 5');\n };\n};\n\nmodule.exports = new LoggerConfig();\n\n\n\n// WEBPACK FOOTER //\n// ./config/loggerConfig.js","const winstonSlackWebHook = require('winston-slack-webhook').SlackWebHook;\nconst winston = require('winston');\n\nfunction SlackConfig () {\n this.slackWebHook = 'default';\n this.slackErrorChannel = 'default';\n this.slackInfoChannel = 'default';\n this.update = (config) => {\n if (!config) {\n return winston.warn('No slack config received');\n }\n // update variables\n winston.info('configuring slack logger...');\n const {slackWebHook, slackErrorChannel, slackInfoChannel} = config;\n this.slackWebHook = slackWebHook;\n this.slackErrorChannel = slackErrorChannel;\n this.slackInfoChannel = slackInfoChannel;\n // update slack webhook settings\n if (this.slackWebHook) {\n // add a transport for errors to slack\n if (this.slackErrorChannel) {\n winston.add(winstonSlackWebHook, {\n name : 'slack-errors-transport',\n level : 'warn',\n webhookUrl: this.slackWebHook,\n channel : this.slackErrorChannel,\n username : 'spee.ch',\n iconEmoji : ':face_with_head_bandage:',\n });\n };\n if (slackInfoChannel) {\n winston.add(winstonSlackWebHook, {\n name : 'slack-info-transport',\n level : 'info',\n webhookUrl: this.slackWebHook,\n channel : this.slackInfoChannel,\n username : 'spee.ch',\n iconEmoji : ':nerd_face:',\n });\n };\n // send test messages\n winston.info('testing slack logger...');\n winston.error('Slack \"error\" logging is online.');\n winston.info('Slack \"info\" logging is online.');\n } else {\n winston.warn('Slack logging is not enabled because no slackWebHook config var provided.');\n }\n };\n};\n\nmodule.exports = new SlackConfig();\n\n\n\n// WEBPACK FOOTER //\n// ./config/slackConfig.js","module.exports = require(\"winston-slack-webhook\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"winston-slack-webhook\"\n// module id = 42\n// module chunks = 0","module.exports = require(\"passport\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"passport\"\n// module id = 43\n// module chunks = 0","const PassportLocalStrategy = require('passport-local').Strategy;\nconst logger = require('winston');\nconst db = require('models');\n\nconst returnUserAndChannelInfo = (userInstance) => {\n return new Promise((resolve, reject) => {\n let userInfo = {};\n userInfo['id'] = userInstance.id;\n userInfo['userName'] = userInstance.userName;\n userInstance\n .getChannel()\n .then(({channelName, channelClaimId}) => {\n userInfo['channelName'] = channelName;\n userInfo['channelClaimId'] = channelClaimId;\n return db.Certificate.getShortChannelIdFromLongChannelId(channelClaimId, channelName);\n })\n .then(shortChannelId => {\n userInfo['shortChannelId'] = shortChannelId;\n resolve(userInfo);\n })\n .catch(error => {\n reject(error);\n });\n });\n};\n\nmodule.exports = new PassportLocalStrategy(\n {\n usernameField: 'username',\n passwordField: 'password',\n },\n (username, password, done) => {\n return db.User\n .findOne({\n where: {userName: username},\n })\n .then(user => {\n if (!user) {\n logger.debug('no user found');\n return done(null, false, {message: 'Incorrect username or password'});\n }\n return user.comparePassword(password)\n .then(isMatch => {\n if (!isMatch) {\n logger.debug('incorrect password');\n return done(null, false, {message: 'Incorrect username or password'});\n }\n logger.debug('Password was a match, returning User');\n return returnUserAndChannelInfo(user)\n .then(userInfo => {\n return done(null, userInfo);\n })\n .catch(error => {\n return error;\n });\n })\n .catch(error => {\n return error;\n });\n })\n .catch(error => {\n return done(error);\n });\n },\n);\n\n\n\n// WEBPACK FOOTER //\n// ./server/speechPassport/local-login.js","const logger = require('winston');\nconst { returnShortId } = require('../helpers/sequelizeHelpers.js');\n\nmodule.exports = (sequelize, { STRING, BOOLEAN, INTEGER, TEXT, DECIMAL }) => {\n const Certificate = sequelize.define(\n 'Certificate',\n {\n address: {\n type : STRING,\n default: null,\n },\n amount: {\n type : DECIMAL(19, 8),\n default: null,\n },\n claimId: {\n type : STRING,\n default: null,\n },\n claimSequence: {\n type : INTEGER,\n default: null,\n },\n decodedClaim: {\n type : BOOLEAN,\n default: null,\n },\n depth: {\n type : INTEGER,\n default: null,\n },\n effectiveAmount: {\n type : DECIMAL(19, 8),\n default: null,\n },\n hasSignature: {\n type : BOOLEAN,\n default: null,\n },\n height: {\n type : INTEGER,\n default: null,\n },\n hex: {\n type : TEXT('long'),\n default: null,\n },\n name: {\n type : STRING,\n default: null,\n },\n nout: {\n type : INTEGER,\n default: null,\n },\n txid: {\n type : STRING,\n default: null,\n },\n validAtHeight: {\n type : INTEGER,\n default: null,\n },\n outpoint: {\n type : STRING,\n default: null,\n },\n valueVersion: {\n type : STRING,\n default: null,\n },\n claimType: {\n type : STRING,\n default: null,\n },\n certificateVersion: {\n type : STRING,\n default: null,\n },\n keyType: {\n type : STRING,\n default: null,\n },\n publicKey: {\n type : TEXT('long'),\n default: null,\n },\n },\n {\n freezeTableName: true,\n }\n );\n\n Certificate.associate = db => {\n Certificate.belongsTo(db.Channel, {\n foreignKey: {\n allowNull: true,\n },\n });\n };\n\n Certificate.getShortChannelIdFromLongChannelId = function (longChannelId, channelName) {\n logger.debug(`getShortChannelIdFromLongChannelId ${channelName}:${longChannelId}`);\n return new Promise((resolve, reject) => {\n this\n .findAll({\n where: {name: channelName},\n order: [['height', 'ASC']],\n })\n .then(result => {\n switch (result.length) {\n case 0:\n throw new Error('No channel(s) found with that channel name');\n default:\n return resolve(returnShortId(result, longChannelId));\n }\n })\n .catch(error => {\n reject(error);\n });\n });\n };\n\n Certificate.getLongChannelIdFromShortChannelId = function (channelName, channelClaimId) {\n logger.debug(`getLongChannelIdFromShortChannelId(${channelName}, ${channelClaimId})`);\n return new Promise((resolve, reject) => {\n this\n .findAll({\n where: {\n name : channelName,\n claimId: {\n $like: `${channelClaimId}%`,\n },\n },\n order: [['height', 'ASC']],\n })\n .then(result => {\n switch (result.length) {\n case 0:\n return resolve(null);\n default: // note results must be sorted\n return resolve(result[0].claimId);\n }\n })\n .catch(error => {\n reject(error);\n });\n });\n };\n\n Certificate.getLongChannelIdFromChannelName = function (channelName) {\n logger.debug(`getLongChannelIdFromChannelName(${channelName})`);\n return new Promise((resolve, reject) => {\n this\n .findAll({\n where: { name: channelName },\n order: [['effectiveAmount', 'DESC'], ['height', 'ASC']],\n })\n .then(result => {\n switch (result.length) {\n case 0:\n return resolve(null);\n default:\n return resolve(result[0].claimId);\n }\n })\n .catch(error => {\n reject(error);\n });\n });\n };\n\n Certificate.validateLongChannelId = function (name, claimId) {\n logger.debug(`validateLongChannelId(${name}, ${claimId})`);\n return new Promise((resolve, reject) => {\n this.findOne({\n where: {name, claimId},\n })\n .then(result => {\n if (!result) {\n return resolve(null);\n };\n resolve(claimId);\n })\n .catch(error => {\n reject(error);\n });\n });\n };\n\n Certificate.getLongChannelId = function (channelName, channelClaimId) {\n logger.debug(`getLongChannelId(${channelName}, ${channelClaimId})`);\n if (channelClaimId && (channelClaimId.length === 40)) { // if a full channel id is provided\n return this.validateLongChannelId(channelName, channelClaimId);\n } else if (channelClaimId && channelClaimId.length < 40) { // if a short channel id is provided\n return this.getLongChannelIdFromShortChannelId(channelName, channelClaimId);\n } else {\n return this.getLongChannelIdFromChannelName(channelName); // if no channel id provided\n }\n };\n\n return Certificate;\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/models/certificate.js","module.exports = (sequelize, { STRING }) => {\n const Channel = sequelize.define(\n 'Channel',\n {\n channelName: {\n type : STRING,\n allowNull: false,\n },\n channelClaimId: {\n type : STRING,\n allowNull: false,\n },\n },\n {\n freezeTableName: true,\n }\n );\n\n Channel.associate = db => {\n Channel.belongsTo(db.User);\n Channel.hasOne(db.Certificate);\n };\n\n return Channel;\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/models/channel.js","const logger = require('winston');\nconst { returnShortId } = require('../helpers/sequelizeHelpers.js');\nconst { assetDefaults: { thumbnail: defaultThumbnail }, details: { host } } = require('../../config/siteConfig.js');\n\nfunction determineFileExtensionFromContentType (contentType) {\n switch (contentType) {\n case 'image/jpeg':\n case 'image/jpg':\n return 'jpeg';\n case 'image/png':\n return 'png';\n case 'image/gif':\n return 'gif';\n case 'video/mp4':\n return 'mp4';\n default:\n logger.debug('setting unknown file type as file extension jpeg');\n return 'jpeg';\n }\n};\n\nfunction determineThumbnail (storedThumbnail, defaultThumbnail) {\n if (storedThumbnail === '') {\n return defaultThumbnail;\n }\n return storedThumbnail;\n};\n\nfunction prepareClaimData (claim) {\n // logger.debug('preparing claim data based on resolved data:', claim);\n claim['thumbnail'] = determineThumbnail(claim.thumbnail, defaultThumbnail);\n claim['fileExt'] = determineFileExtensionFromContentType(claim.contentType);\n claim['host'] = host;\n return claim;\n};\n\nmodule.exports = (sequelize, { STRING, BOOLEAN, INTEGER, TEXT, DECIMAL }) => {\n const Claim = sequelize.define(\n 'Claim',\n {\n address: {\n type : STRING,\n default: null,\n },\n amount: {\n type : DECIMAL(19, 8),\n default: null,\n },\n claimId: {\n type : STRING,\n default: null,\n },\n claimSequence: {\n type : INTEGER,\n default: null,\n },\n decodedClaim: {\n type : BOOLEAN,\n default: null,\n },\n depth: {\n type : INTEGER,\n default: null,\n },\n effectiveAmount: {\n type : DECIMAL(19, 8),\n default: null,\n },\n hasSignature: {\n type : BOOLEAN,\n default: null,\n },\n height: {\n type : INTEGER,\n default: null,\n },\n hex: {\n type : TEXT('long'),\n default: null,\n },\n name: {\n type : STRING,\n default: null,\n },\n nout: {\n type : INTEGER,\n default: null,\n },\n txid: {\n type : STRING,\n default: null,\n },\n validAtHeight: {\n type : INTEGER,\n default: null,\n },\n outpoint: {\n type : STRING,\n default: null,\n },\n claimType: {\n type : STRING,\n default: null,\n },\n certificateId: {\n type : STRING,\n default: null,\n },\n author: {\n type : STRING,\n default: null,\n },\n description: {\n type : TEXT('long'),\n default: null,\n },\n language: {\n type : STRING,\n default: null,\n },\n license: {\n type : STRING,\n default: null,\n },\n licenseUrl: {\n type : STRING,\n default: null,\n },\n nsfw: {\n type : BOOLEAN,\n default: null,\n },\n preview: {\n type : STRING,\n default: null,\n },\n thumbnail: {\n type : STRING,\n default: null,\n },\n title: {\n type : STRING,\n default: null,\n },\n metadataVersion: {\n type : STRING,\n default: null,\n },\n contentType: {\n type : STRING,\n default: null,\n },\n source: {\n type : STRING,\n default: null,\n },\n sourceType: {\n type : STRING,\n default: null,\n },\n sourceVersion: {\n type : STRING,\n default: null,\n },\n streamVersion: {\n type : STRING,\n default: null,\n },\n valueVersion: {\n type : STRING,\n default: null,\n },\n channelName: {\n type : STRING,\n allowNull: true,\n default : null,\n },\n },\n {\n freezeTableName: true,\n }\n );\n\n Claim.associate = db => {\n Claim.belongsTo(db.File, {\n foreignKey: {\n allowNull: true,\n },\n });\n };\n\n Claim.getShortClaimIdFromLongClaimId = function (claimId, claimName) {\n logger.debug(`Claim.getShortClaimIdFromLongClaimId for ${claimName}#${claimId}`);\n return new Promise((resolve, reject) => {\n this\n .findAll({\n where: { name: claimName },\n order: [['height', 'ASC']],\n })\n .then(result => {\n switch (result.length) {\n case 0:\n throw new Error('No claim(s) found with that claim name');\n default:\n resolve(returnShortId(result, claimId));\n }\n })\n .catch(error => {\n reject(error);\n });\n });\n };\n\n Claim.getAllChannelClaims = function (channelClaimId) {\n logger.debug(`Claim.getAllChannelClaims for ${channelClaimId}`);\n return new Promise((resolve, reject) => {\n this\n .findAll({\n where: { certificateId: channelClaimId },\n order: [['height', 'ASC']],\n raw : true, // returns an array of only data, not an array of instances\n })\n .then(channelClaimsArray => {\n // logger.debug('channelclaimsarray length:', channelClaimsArray.length);\n switch (channelClaimsArray.length) {\n case 0:\n return resolve(null);\n default:\n channelClaimsArray.forEach(claim => {\n claim['fileExt'] = determineFileExtensionFromContentType(claim.contentType);\n claim['thumbnail'] = determineThumbnail(claim.thumbnail, defaultThumbnail);\n return claim;\n });\n return resolve(channelClaimsArray);\n }\n })\n .catch(error => {\n reject(error);\n });\n });\n };\n\n Claim.getClaimIdByLongChannelId = function (channelClaimId, claimName) {\n logger.debug(`finding claim id for claim ${claimName} from channel ${channelClaimId}`);\n return new Promise((resolve, reject) => {\n this\n .findAll({\n where: { name: claimName, certificateId: channelClaimId },\n order: [['id', 'ASC']],\n })\n .then(result => {\n switch (result.length) {\n case 0:\n return resolve(null);\n case 1:\n return resolve(result[0].claimId);\n default:\n logger.error(`${result.length} records found for \"${claimName}\" in channel \"${channelClaimId}\"`);\n return resolve(result[0].claimId);\n }\n })\n .catch(error => {\n reject(error);\n });\n });\n };\n\n Claim.getLongClaimIdFromShortClaimId = function (name, shortId) {\n return new Promise((resolve, reject) => {\n this\n .findAll({\n where: {\n name,\n claimId: {\n $like: `${shortId}%`,\n }},\n order: [['height', 'ASC']],\n })\n .then(result => {\n switch (result.length) {\n case 0:\n return resolve(null);\n default: // note results must be sorted\n return resolve(result[0].claimId);\n }\n })\n .catch(error => {\n reject(error);\n });\n });\n };\n\n Claim.getTopFreeClaimIdByClaimName = function (name) {\n return new Promise((resolve, reject) => {\n this\n .findAll({\n where: { name },\n order: [['effectiveAmount', 'DESC'], ['height', 'ASC']], // note: maybe height and effective amount need to switch?\n })\n .then(result => {\n logger.debug('length of result', result.length);\n switch (result.length) {\n case 0:\n return resolve(null);\n default:\n return resolve(result[0].dataValues.claimId);\n }\n })\n .catch(error => {\n reject(error);\n });\n });\n };\n\n Claim.validateLongClaimId = function (name, claimId) {\n return new Promise((resolve, reject) => {\n this.findOne({\n where: {name, claimId},\n })\n .then(result => {\n if (!result) {\n return resolve(null);\n };\n resolve(claimId);\n })\n .catch(error => {\n reject(error);\n });\n });\n };\n\n Claim.getLongClaimId = function (claimName, claimId) {\n logger.debug(`getLongClaimId(${claimName}, ${claimId})`);\n if (claimId && (claimId.length === 40)) { // if a full claim id is provided\n return this.validateLongClaimId(claimName, claimId);\n } else if (claimId && claimId.length < 40) {\n return this.getLongClaimIdFromShortClaimId(claimName, claimId); // if a short claim id is provided\n } else {\n return this.getTopFreeClaimIdByClaimName(claimName); // if no claim id is provided\n }\n };\n\n Claim.resolveClaim = function (name, claimId) {\n logger.debug(`Claim.resolveClaim: ${name} ${claimId}`);\n return new Promise((resolve, reject) => {\n this\n .findAll({\n where: { name, claimId },\n })\n .then(claimArray => {\n switch (claimArray.length) {\n case 0:\n return resolve(null);\n case 1:\n return resolve(prepareClaimData(claimArray[0].dataValues));\n default:\n logger.error(`more than one record matches ${name}#${claimId} in db.Claim`);\n return resolve(prepareClaimData(claimArray[0].dataValues));\n }\n })\n .catch(error => {\n reject(error);\n });\n });\n };\n\n return Claim;\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/models/claim.js","module.exports = (sequelize, { STRING, BOOLEAN, INTEGER }) => {\n const File = sequelize.define(\n 'File',\n {\n name: {\n type : STRING,\n allowNull: false,\n },\n claimId: {\n type : STRING,\n allowNull: false,\n },\n address: {\n type : STRING,\n allowNull: false,\n },\n outpoint: {\n type : STRING,\n allowNull: false,\n },\n height: {\n type : INTEGER,\n allowNull: false,\n default : 0,\n },\n fileName: {\n type : STRING,\n allowNull: false,\n },\n filePath: {\n type : STRING,\n allowNull: false,\n },\n fileType: {\n type: STRING,\n },\n nsfw: {\n type : BOOLEAN,\n allowNull : false,\n defaultValue: false,\n },\n trendingEligible: {\n type : BOOLEAN,\n allowNull : false,\n defaultValue: true,\n },\n },\n {\n freezeTableName: true,\n }\n );\n\n File.associate = db => {\n File.hasMany(db.Request);\n File.hasOne(db.Claim);\n };\n\n File.getRecentClaims = function () {\n return this.findAll({\n where: { nsfw: false, trendingEligible: true },\n order: [['createdAt', 'DESC']],\n limit: 25,\n });\n };\n\n return File;\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/models/file.js","module.exports = (sequelize, { STRING, BOOLEAN, TEXT }) => {\n const Request = sequelize.define(\n 'Request',\n {\n action: {\n type : STRING,\n allowNull: false,\n },\n url: {\n type : STRING,\n allowNull: false,\n },\n ipAddress: {\n type : STRING,\n allowNull: true,\n },\n result: {\n type : TEXT('long'),\n allowNull: true,\n default : null,\n },\n },\n {\n freezeTableName: true,\n }\n );\n\n Request.associate = db => {\n Request.belongsTo(db.File, {\n foreignKey: {\n allowNull: true,\n },\n });\n };\n\n return Request;\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/models/request.js","'use strict';\nconst bcrypt = require('bcrypt');\nconst logger = require('winston');\n\nmodule.exports = (sequelize, { STRING }) => {\n const User = sequelize.define(\n 'User',\n {\n userName: {\n type : STRING,\n allowNull: false,\n },\n password: {\n type : STRING,\n allowNull: false,\n },\n },\n {\n freezeTableName: true,\n }\n );\n\n User.associate = db => {\n User.hasOne(db.Channel);\n };\n\n User.prototype.comparePassword = function (password) {\n return bcrypt.compare(password, this.password);\n };\n\n User.prototype.changePassword = function (newPassword) {\n return new Promise((resolve, reject) => {\n // generate a salt string to use for hashing\n bcrypt.genSalt((saltError, salt) => {\n if (saltError) {\n logger.error('salt error', saltError);\n reject(saltError);\n return;\n }\n // generate a hashed version of the user's password\n bcrypt.hash(newPassword, salt, (hashError, hash) => {\n // if there is an error with the hash generation return the error\n if (hashError) {\n logger.error('hash error', hashError);\n reject(hashError);\n return;\n }\n // replace the current password with the new hash\n this\n .update({password: hash})\n .then(() => {\n resolve();\n })\n .catch(error => {\n reject(error);\n });\n });\n });\n });\n };\n\n // pre-save hook method to hash the user's password before the user's info is saved to the db.\n User.hook('beforeCreate', (user, options) => {\n logger.debug('User.beforeCreate hook...');\n return new Promise((resolve, reject) => {\n // generate a salt string to use for hashing\n bcrypt.genSalt((saltError, salt) => {\n if (saltError) {\n logger.error('salt error', saltError);\n reject(saltError);\n return;\n }\n // generate a hashed version of the user's password\n bcrypt.hash(user.password, salt, (hashError, hash) => {\n // if there is an error with the hash generation return the error\n if (hashError) {\n logger.error('hash error', hashError);\n reject(hashError);\n return;\n }\n // replace the password string with the hash password value\n user.password = hash;\n resolve();\n });\n });\n });\n });\n\n return User;\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/models/user.js","module.exports = require(\"bcrypt\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"bcrypt\"\n// module id = 51\n// module chunks = 0","const PassportLocalStrategy = require('passport-local').Strategy;\nconst lbryApi = require('../helpers/lbryApi.js');\nconst logger = require('winston');\nconst db = require('models');\n\nmodule.exports = new PassportLocalStrategy(\n {\n usernameField: 'username',\n passwordField: 'password',\n },\n (username, password, done) => {\n logger.verbose(`new channel signup request. user: ${username} pass: ${password} .`);\n let userInfo = {};\n // server-side validaton of inputs (username, password)\n\n // create the channel and retrieve the metadata\n return lbryApi.createChannel(`@${username}`)\n .then(tx => {\n // create user record\n const userData = {\n userName: username,\n password: password,\n };\n logger.verbose('userData >', userData);\n // create user record\n const channelData = {\n channelName : `@${username}`,\n channelClaimId: tx.claim_id,\n };\n logger.verbose('channelData >', channelData);\n // create certificate record\n const certificateData = {\n claimId: tx.claim_id,\n name : `@${username}`,\n // address,\n };\n logger.verbose('certificateData >', certificateData);\n // save user and certificate to db\n return Promise.all([db.User.create(userData), db.Channel.create(channelData), db.Certificate.create(certificateData)]);\n })\n .then(([newUser, newChannel, newCertificate]) => {\n logger.verbose('user and certificate successfully created');\n // store the relevant newUser info to be passed back for req.User\n userInfo['id'] = newUser.id;\n userInfo['userName'] = newUser.userName;\n userInfo['channelName'] = newChannel.channelName;\n userInfo['channelClaimId'] = newChannel.channelClaimId;\n // associate the instances\n return Promise.all([newCertificate.setChannel(newChannel), newChannel.setUser(newUser)]);\n })\n .then(() => {\n logger.verbose('user and certificate successfully associated');\n return db.Certificate.getShortChannelIdFromLongChannelId(userInfo.channelClaimId, userInfo.channelName);\n })\n .then(shortChannelId => {\n userInfo['shortChannelId'] = shortChannelId;\n return done(null, userInfo);\n })\n .catch(error => {\n logger.error('signup error', error);\n return done(error);\n });\n }\n);\n\n\n\n// WEBPACK FOOTER //\n// ./server/speechPassport/local-signup.js","module.exports = require(\"axios\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"axios\"\n// module id = 53\n// module chunks = 0","const lbryConfig = {\n api: {\n apiHost: 'localhost',\n apiPort: '5279',\n },\n};\n\nmodule.exports = lbryConfig;\n\n\n\n// WEBPACK FOOTER //\n// ./config/lbryConfig.js","module.exports = require(\"universal-analytics\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"universal-analytics\"\n// module id = 55\n// module chunks = 0","module.exports = {\n serializeSpeechUser (user, done) { // returns user data to be serialized into session\n console.log('serializing user');\n done(null, user);\n },\n deserializeSpeechUser (user, done) { // deserializes session and populates additional info to req.user\n console.log('deserializing user');\n done(null, user);\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/authHelpers.js","const speechPassport = require('speechPassport');\nconst handleSignupRequest = require('./signup');\nconst handleLoginRequest = require('./login');\nconst handleLogoutRequest = require('./logout');\nconst handleUserRequest = require('./user');\n\nmodule.exports = (app) => {\n app.post('/signup', speechPassport.authenticate('local-signup'), handleSignupRequest);\n app.post('/login', handleLoginRequest);\n app.get('/logout', handleLogoutRequest);\n app.get('/user', handleUserRequest);\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/auth/index.js","const signup = (req, res) => {\n res.status(200).json({\n success : true,\n channelName : req.user.channelName,\n channelClaimId: req.user.channelClaimId,\n shortChannelId: req.user.shortChannelId,\n });\n};\n\nmodule.exports = signup;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/auth/signup.js","const speechPassport = require('speechPassport');\n\nconst login = (req, res, next) => {\n speechPassport.authenticate('local-login', (err, user, info) => {\n if (err) {\n return next(err);\n }\n if (!user) {\n return res.status(400).json({\n success: false,\n message: info.message,\n });\n }\n req.logIn(user, (err) => {\n if (err) {\n return next(err);\n }\n return res.status(200).json({\n success : true,\n channelName : req.user.channelName,\n channelClaimId: req.user.channelClaimId,\n shortChannelId: req.user.shortChannelId,\n });\n });\n })(req, res, next);\n};\n\nmodule.exports = login;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/auth/login.js","const logout = (req, res) => {\n req.logout();\n res.status(200).json({success: true, message: 'you successfully logged out'});\n};\n\nmodule.exports = logout;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/auth/logout.js","const user = (req, res) => {\n if (req.user) {\n res.status(200).json({success: true, data: req.user});\n } else {\n res.status(401).json({success: false, message: 'user is not logged in'});\n }\n};\n\nmodule.exports = user;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/auth/user.js","const channelAvailability = require('./channelAvailability');\nconst channelClaims = require('./channelClaims');\nconst channelData = require('./channelData');\nconst channelShortId = require('./channelShortId');\nconst claimAvailability = require('./claimAvailability');\nconst claimData = require('./claimData');\nconst claimGet = require('./claimGet');\nconst claimLongId = require('./claimLongId');\nconst claimPublish = require('./claimPublish');\nconst claimResolve = require('./claimResolve');\nconst claimShortId = require('./claimShortId');\nconst claimList = require('./claimList');\nconst fileAvailability = require('./fileAvailability');\n\nconst multipartMiddleware = require('helpers/multipartMiddleware');\n\nmodule.exports = (app) => {\n // channel routes\n app.get('/api/channel/availability/:name', channelAvailability);\n app.get('/api/channel/short-id/:longId/:name', channelShortId);\n app.get('/api/channel/data/:channelName/:channelClaimId', channelData);\n app.get('/api/channel/claims/:channelName/:channelClaimId/:page', channelClaims);\n // claim routes\n app.get('/api/claim/list/:name', claimList);\n app.get('/api/claim/get/:name/:claimId', claimGet);\n app.get('/api/claim/availability/:name', claimAvailability);\n app.get('/api/claim/resolve/:name/:claimId', claimResolve);\n app.post('/api/claim/publish', multipartMiddleware, claimPublish);\n app.get('/api/claim/short-id/:longId/:name', claimShortId);\n app.post('/api/claim/long-id', claimLongId);\n app.get('/api/claim/data/:claimName/:claimId', claimData);\n // file routes\n app.get('/api/file/availability/:name/:claimId', fileAvailability);\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/index.js","const { checkChannelAvailability } = require('controllers/publishController.js');\nconst { sendGATimingEvent } = require('helpers/googleAnalytics.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\n\n/*\n\n route to check whether site has published to a channel\n\n*/\n\nconst channelAvailability = ({ ip, originalUrl, params: { name } }, res) => {\n const gaStartTime = Date.now();\n checkChannelAvailability(name)\n .then(availableName => {\n res.status(200).json(availableName);\n sendGATimingEvent('end-to-end', 'claim name availability', name, gaStartTime, Date.now());\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = channelAvailability;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/channelAvailability.js","module.exports = require(\"fs\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"fs\"\n// module id = 64\n// module chunks = 0","const { getChannelClaims } = require('controllers/serveController.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\n\nconst NO_CHANNEL = 'NO_CHANNEL';\n\n/*\n\n route to get all claims for channel\n\n*/\n\nconst channelClaims = ({ ip, originalUrl, body, params }, res) => {\n const channelName = params.channelName;\n let channelClaimId = params.channelClaimId;\n if (channelClaimId === 'none') channelClaimId = null;\n const page = params.page;\n getChannelClaims(channelName, channelClaimId, page)\n .then(data => {\n if (data === NO_CHANNEL) {\n return res.status(404).json({success: false, message: 'No matching channel was found'});\n }\n res.status(200).json({success: true, data});\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = channelClaims;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/channelClaims.js","const CLAIMS_PER_PAGE = 12;\n\nmodule.exports = {\n returnPaginatedChannelClaims (channelName, longChannelClaimId, claims, page) {\n const totalPages = module.exports.determineTotalPages(claims);\n const paginationPage = module.exports.getPageFromQuery(page);\n const viewData = {\n channelName : channelName,\n longChannelClaimId: longChannelClaimId,\n claims : module.exports.extractPageFromClaims(claims, paginationPage),\n previousPage : module.exports.determinePreviousPage(paginationPage),\n currentPage : paginationPage,\n nextPage : module.exports.determineNextPage(totalPages, paginationPage),\n totalPages : totalPages,\n totalResults : module.exports.determineTotalClaims(claims),\n };\n return viewData;\n },\n getPageFromQuery (page) {\n if (page) {\n return parseInt(page);\n }\n return 1;\n },\n extractPageFromClaims (claims, pageNumber) {\n if (!claims) {\n return []; // if no claims, return this default\n }\n // logger.debug('claims is array?', Array.isArray(claims));\n // logger.debug(`pageNumber ${pageNumber} is number?`, Number.isInteger(pageNumber));\n const claimStartIndex = (pageNumber - 1) * CLAIMS_PER_PAGE;\n const claimEndIndex = claimStartIndex + CLAIMS_PER_PAGE;\n const pageOfClaims = claims.slice(claimStartIndex, claimEndIndex);\n return pageOfClaims;\n },\n determineTotalPages (claims) {\n if (!claims) {\n return 0;\n } else {\n const totalClaims = claims.length;\n if (totalClaims < CLAIMS_PER_PAGE) {\n return 1;\n }\n const fullPages = Math.floor(totalClaims / CLAIMS_PER_PAGE);\n const remainder = totalClaims % CLAIMS_PER_PAGE;\n if (remainder === 0) {\n return fullPages;\n }\n return fullPages + 1;\n }\n },\n determinePreviousPage (currentPage) {\n if (currentPage === 1) {\n return null;\n }\n return currentPage - 1;\n },\n determineNextPage (totalPages, currentPage) {\n if (currentPage === totalPages) {\n return null;\n }\n return currentPage + 1;\n },\n determineTotalClaims (claims) {\n if (!claims) {\n return 0;\n }\n return claims.length;\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/channelPagination.js","const { getChannelData } = require('controllers/serveController.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\n\nconst NO_CHANNEL = 'NO_CHANNEL';\n\n/*\n\n route to get data for a channel\n\n*/\n\nconst channelData = ({ ip, originalUrl, body, params }, res) => {\n const channelName = params.channelName;\n let channelClaimId = params.channelClaimId;\n if (channelClaimId === 'none') channelClaimId = null;\n getChannelData(channelName, channelClaimId, 0)\n .then(data => {\n if (data === NO_CHANNEL) {\n return res.status(404).json({success: false, message: 'No matching channel was found'});\n }\n res.status(200).json({success: true, data});\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = channelData;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/channelData.js","const { handleErrorResponse } = require('helpers/errorHandlers.js');\nconst db = require('models');\n\n/*\n\nroute to get a short channel id from long channel Id\n\n*/\n\nconst channelShortIdRoute = ({ ip, originalUrl, params }, res) => {\n db.Certificate.getShortChannelIdFromLongChannelId(params.longId, params.name)\n .then(shortId => {\n res.status(200).json(shortId);\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = channelShortIdRoute;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/channelShortId.js","const { claimNameIsAvailable } = require('controllers/publishController.js');\nconst { sendGATimingEvent } = require('helpers/googleAnalytics.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\n\n/*\n\n route to check whether this site published to a claim\n\n*/\n\nconst claimAvailability = ({ ip, originalUrl, params: { name } }, res) => {\n const gaStartTime = Date.now();\n claimNameIsAvailable(name)\n .then(result => {\n res.status(200).json(result);\n sendGATimingEvent('end-to-end', 'claim name availability', name, gaStartTime, Date.now());\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimAvailability;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimAvailability.js","const { handleErrorResponse } = require('helpers/errorHandlers.js');\nconst db = require('models');\n\n/*\n\n route to return data for a claim\n\n*/\n\nconst claimData = ({ ip, originalUrl, body, params }, res) => {\n const claimName = params.claimName;\n let claimId = params.claimId;\n if (claimId === 'none') claimId = null;\n db.Claim.resolveClaim(claimName, claimId)\n .then(claimInfo => {\n if (!claimInfo) {\n return res.status(404).json({success: false, message: 'No claim could be found'});\n }\n res.status(200).json({success: true, data: claimInfo});\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimData;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimData.js","const { getClaim } = require('helpers/lbryApi.js');\nconst { addGetResultsToFileData, createFileData } = require('../../helpers/publishHelpers.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\nconst db = require('models');\n\n/*\n\n route to get a claim\n\n*/\n\nconst claimGet = ({ ip, originalUrl, params }, res) => {\n const name = params.name;\n const claimId = params.claimId;\n // resolve the claim\n db.Claim.resolveClaim(name, claimId)\n .then(resolveResult => {\n // make sure a claim actually exists at that uri\n if (!resolveResult) {\n throw new Error('No matching uri found in Claim table');\n }\n let fileData = createFileData(resolveResult);\n // get the claim\n return Promise.all([fileData, getClaim(`${name}#${claimId}`)]);\n })\n .then(([ fileData, getResult ]) => {\n fileData = addGetResultsToFileData(fileData, getResult);\n return Promise.all([db.upsert(db.File, fileData, {name, claimId}, 'File'), getResult]);\n })\n .then(([ fileRecord, {message, completed} ]) => {\n res.status(200).json({ success: true, message, completed });\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimGet;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimGet.js","const { getClaimId } = require('controllers/serveController.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\n\nconst NO_CHANNEL = 'NO_CHANNEL';\nconst NO_CLAIM = 'NO_CLAIM';\n\n/*\n\n route to get a long claim id\n\n*/\n\nconst claimLongId = ({ ip, originalUrl, body, params }, res) => {\n const channelName = body.channelName;\n const channelClaimId = body.channelClaimId;\n const claimName = body.claimName;\n const claimId = body.claimId;\n getClaimId(channelName, channelClaimId, claimName, claimId)\n .then(result => {\n if (result === NO_CHANNEL) {\n return res.status(404).json({success: false, message: 'No matching channel could be found'});\n }\n if (result === NO_CLAIM) {\n return res.status(404).json({success: false, message: 'No matching claim id could be found'});\n }\n res.status(200).json({success: true, data: result});\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimLongId;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimLongId.js","const { createBasicPublishParams, createThumbnailPublishParams, parsePublishApiRequestBody, parsePublishApiRequestFiles } = require('helpers/publishHelpers.js');\nconst { claimNameIsAvailable, publish } = require('controllers/publishController.js');\nconst { authenticateUser } = require('auth/authentication.js');\nconst { sendGATimingEvent } = require('helpers/googleAnalytics.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\nconst { details: { host } } = require('siteConfig.js');\n\n/*\n\n route to publish a claim through the daemon\n\n*/\n\nconst claimPublish = ({ body, files, headers, ip, originalUrl, user }, res) => {\n // define variables\n let channelName, channelId, channelPassword, description, fileName, filePath, fileType, gaStartTime, license, name, nsfw, thumbnail, thumbnailFileName, thumbnailFilePath, thumbnailFileType, title;\n // record the start time of the request\n gaStartTime = Date.now();\n // validate the body and files of the request\n try {\n // validateApiPublishRequest(body, files);\n ({name, nsfw, license, title, description, thumbnail} = parsePublishApiRequestBody(body));\n ({fileName, filePath, fileType, thumbnailFileName, thumbnailFilePath, thumbnailFileType} = parsePublishApiRequestFiles(files));\n ({channelName, channelId, channelPassword} = body);\n } catch (error) {\n return res.status(400).json({success: false, message: error.message});\n }\n // check channel authorization\n Promise\n .all([\n authenticateUser(channelName, channelId, channelPassword, user),\n claimNameIsAvailable(name),\n createBasicPublishParams(filePath, name, title, description, license, nsfw, thumbnail),\n createThumbnailPublishParams(thumbnailFilePath, name, license, nsfw),\n ])\n .then(([{channelName, channelClaimId}, validatedClaimName, publishParams, thumbnailPublishParams]) => {\n // add channel details to the publish params\n if (channelName && channelClaimId) {\n publishParams['channel_name'] = channelName;\n publishParams['channel_id'] = channelClaimId;\n }\n // publish the thumbnail\n if (thumbnailPublishParams) {\n publish(thumbnailPublishParams, thumbnailFileName, thumbnailFileType);\n }\n // publish the asset\n return publish(publishParams, fileName, fileType);\n })\n .then(result => {\n res.status(200).json({\n success: true,\n message: 'publish completed successfully',\n data : {\n name,\n claimId: result.claim_id,\n url : `${host}/${result.claim_id}/${name}`,\n lbryTx : result,\n },\n });\n // record the publish end time and send to google analytics\n sendGATimingEvent('end-to-end', 'publish', fileType, gaStartTime, Date.now());\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimPublish;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimPublish.js","const db = require('models');\nconst logger = require('winston');\n\nmodule.exports = {\n authenticateUser (channelName, channelId, channelPassword, user) {\n // case: no channelName or channel Id are provided (anonymous), regardless of whether user token is provided\n if (!channelName && !channelId) {\n return {\n channelName : null,\n channelClaimId: null,\n };\n }\n // case: channelName or channel Id are provided with user token\n if (user) {\n if (channelName && channelName !== user.channelName) {\n throw new Error('the provided channel name does not match user credentials');\n }\n if (channelId && channelId !== user.channelClaimId) {\n throw new Error('the provided channel id does not match user credentials');\n }\n return {\n channelName : user.channelName,\n channelClaimId: user.channelClaimId,\n };\n }\n // case: channelName or channel Id are provided with password instead of user token\n if (!channelPassword) throw new Error('no channel password provided');\n return module.exports.authenticateChannelCredentials(channelName, channelId, channelPassword);\n },\n authenticateChannelCredentials (channelName, channelId, userPassword) {\n return new Promise((resolve, reject) => {\n // hoisted variables\n let channelData;\n // build the params for finding the channel\n let channelFindParams = {};\n if (channelName) channelFindParams['channelName'] = channelName;\n if (channelId) channelFindParams['channelClaimId'] = channelId;\n // find the channel\n db.Channel\n .findOne({\n where: channelFindParams,\n })\n .then(channel => {\n if (!channel) {\n logger.debug('no channel found');\n throw new Error('Authentication failed, you do not have access to that channel');\n }\n channelData = channel.get();\n logger.debug('channel data:', channelData);\n return db.User.findOne({\n where: { userName: channelData.channelName.substring(1) },\n });\n })\n .then(user => {\n if (!user) {\n logger.debug('no user found');\n throw new Error('Authentication failed, you do not have access to that channel');\n }\n return user.comparePassword(userPassword);\n })\n .then(isMatch => {\n if (!isMatch) {\n logger.debug('incorrect password');\n throw new Error('Authentication failed, you do not have access to that channel');\n }\n logger.debug('...password was a match...');\n resolve(channelData);\n })\n .catch(error => {\n reject(error);\n });\n });\n },\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/auth/authentication.js","const { resolveUri } = require('helpers/lbryApi.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\n\n/*\n\n route to run a resolve request on the daemon\n\n*/\n\nconst claimResolve = ({ headers, ip, originalUrl, params }, res) => {\n resolveUri(`${params.name}#${params.claimId}`)\n .then(resolvedUri => {\n res.status(200).json(resolvedUri);\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimResolve;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimResolve.js","const { handleErrorResponse } = require('helpers/errorHandlers.js');\nconst db = require('models');\n\n/*\n\n route to get a short claim id from long claim Id\n\n*/\n\nconst claimShortId = ({ ip, originalUrl, body, params }, res) => {\n db.Claim.getShortClaimIdFromLongClaimId(params.longId, params.name)\n .then(shortId => {\n res.status(200).json({success: true, data: shortId});\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimShortId;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimShortId.js","const { getClaimList } = require('helpers/lbryApi.js');\nconst { handleErrorResponse } = require('helpers/errorHandlers.js');\n\n/*\n\n route to get list of claims\n\n*/\n\nconst claimList = ({ ip, originalUrl, params }, res) => {\n getClaimList(params.name)\n .then(claimsList => {\n res.status(200).json(claimsList);\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = claimList;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/claimList.js","const { handleErrorResponse } = require('helpers/errorHandlers.js');\nconst db = require('models');\n\n/*\n\n route to see if asset is available locally\n\n*/\n\nconst fileAvailability = ({ ip, originalUrl, params }, res) => {\n const name = params.name;\n const claimId = params.claimId;\n db.File\n .findOne({\n where: {\n name,\n claimId,\n },\n })\n .then(result => {\n if (result) {\n return res.status(200).json({success: true, data: true});\n }\n res.status(200).json({success: true, data: false});\n })\n .catch(error => {\n handleErrorResponse(originalUrl, ip, error, res);\n });\n};\n\nmodule.exports = fileAvailability;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/api/fileAvailability.js","const multipart = require('connect-multiparty');\r\nconst { publishing: { uploadDirectory } } = require('siteConfig.js');\r\nconst multipartMiddleware = multipart({uploadDir: uploadDirectory});\r\n\r\nmodule.exports = multipartMiddleware;\r\n\n\n\n// WEBPACK FOOTER //\n// ./server/helpers/multipartMiddleware.js","module.exports = require(\"connect-multiparty\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"connect-multiparty\"\n// module id = 80\n// module chunks = 0","const handlePageRequest = require('./sendReactApp');\nconst handleEmbedRequest = require('./sendEmbedPage');\nconst redirect = require('./redirect');\n\nmodule.exports = (app) => {\n app.get('/', handlePageRequest);\n app.get('/login', handlePageRequest);\n app.get('/about', handlePageRequest);\n app.get('/trending', redirect('/popular'));\n app.get('/popular', handlePageRequest);\n app.get('/new', handlePageRequest);\n app.get('/embed/:claimId/:name', handleEmbedRequest); // route to send embedable video player (for twitter)\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/pages/index.js","const handlePageRender = require('helpers/handlePageRender.jsx');\n\nconst sendReactApp = (req, res) => {\n handlePageRender(req, res);\n};\n\nmodule.exports = sendReactApp;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/pages/sendReactApp.js","const { details: { host } } = require('siteConfig.js');\n\nconst sendEmbedPage = ({ params }, res) => {\n const claimId = params.claimId;\n const name = params.name;\n // get and render the content\n res.status(200).render('embed', { layout: 'embed', host, claimId, name });\n};\n\nmodule.exports = sendEmbedPage;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/pages/sendEmbedPage.js","const redirect = (route) => {\n return (req, res) => {\n res.status(301).redirect(route);\n };\n};\n\nmodule.exports = redirect;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/pages/redirect.js","const serveAssetByClaim = require('./serveAssetByClaim');\nconst serveAssetByIdentifierAndClaim = require('./serveAssetByIdentifierAndClaim');\n\nmodule.exports = (app, db) => {\n app.get('/:identifier/:claim', serveAssetByIdentifierAndClaim);\n app.get('/:claim', serveAssetByClaim);\n};\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/assets/index.js","const { sendGAServeEvent } = require('helpers/googleAnalytics');\nconst { determineResponseType, logRequestData, getClaimIdAndServeAsset } = require('helpers/serveHelpers.js');\nconst lbryUri = require('helpers/lbryUri.js');\nconst handleShowRender = require('helpers/handleShowRender.jsx');\nconst SERVE = 'SERVE';\n\n/*\n\n route to serve an asset or the react app via the claim name only\n\n*/\n\nconst serverAssetByClaim = (req, res) => {\n const { headers, ip, originalUrl, params } = req;\n // decide if this is a show request\n let hasFileExtension;\n try {\n ({ hasFileExtension } = lbryUri.parseModifier(params.claim));\n } catch (error) {\n return res.status(400).json({success: false, message: error.message});\n }\n let responseType = determineResponseType(hasFileExtension, headers);\n if (responseType !== SERVE) {\n return handleShowRender(req, res);\n }\n // handle serve request\n // send google analytics\n sendGAServeEvent(headers, ip, originalUrl);\n // parse the claim\n let claimName;\n try {\n ({claimName} = lbryUri.parseClaim(params.claim));\n } catch (error) {\n return res.status(400).json({success: false, message: error.message});\n }\n // log the request data for debugging\n logRequestData(responseType, claimName, null, null);\n // get the claim Id and then serve the asset\n getClaimIdAndServeAsset(null, null, claimName, null, originalUrl, ip, res);\n};\n\nmodule.exports = serverAssetByClaim;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/assets/serveAssetByClaim.js","module.exports = require(\"redux-saga\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"redux-saga\"\n// module id = 87\n// module chunks = 0","module.exports = require(\"redux-saga/effects\");\n\n\n//////////////////\n// WEBPACK FOOTER\n// external \"redux-saga/effects\"\n// module id = 88\n// module chunks = 0","const { sendGAServeEvent } = require('helpers/googleAnalytics');\nconst {\n determineResponseType,\n flipClaimNameAndIdForBackwardsCompatibility,\n logRequestData,\n getClaimIdAndServeAsset,\n} = require('helpers/serveHelpers.js');\nconst lbryUri = require('helpers/lbryUri.js');\nconst handleShowRender = require('helpers/handleShowRender.jsx');\n\nconst SERVE = 'SERVE';\n\n/*\n\n route to serve an asset or the react app via the claim name and an identifier\n\n*/\n\nconst serverAssetByIdentifierAndClaim = (req, res) => {\n const { headers, ip, originalUrl, params } = req;\n // decide if this is a show request\n let hasFileExtension;\n try {\n ({ hasFileExtension } = lbryUri.parseModifier(params.claim));\n } catch (error) {\n return res.status(400).json({success: false, message: error.message});\n }\n let responseType = determineResponseType(hasFileExtension, headers);\n if (responseType !== SERVE) {\n return handleShowRender(req, res);\n }\n // handle serve request\n // send google analytics\n sendGAServeEvent(headers, ip, originalUrl);\n // parse the claim\n let claimName;\n try {\n ({ claimName } = lbryUri.parseClaim(params.claim));\n } catch (error) {\n return res.status(400).json({success: false, message: error.message});\n }\n // parse the identifier\n let isChannel, channelName, channelClaimId, claimId;\n try {\n ({ isChannel, channelName, channelClaimId, claimId } = lbryUri.parseIdentifier(params.identifier));\n } catch (error) {\n return res.status(400).json({success: false, message: error.message});\n }\n if (!isChannel) {\n [claimId, claimName] = flipClaimNameAndIdForBackwardsCompatibility(claimId, claimName);\n }\n // log the request data for debugging\n logRequestData(responseType, claimName, channelName, claimId);\n // get the claim Id and then serve the asset\n getClaimIdAndServeAsset(channelName, channelClaimId, claimName, claimId, originalUrl, ip, res);\n};\n\nmodule.exports = serverAssetByIdentifierAndClaim;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/assets/serveAssetByIdentifierAndClaim.js","const handlePageRequest = require('./sendReactApp');\r\n\r\nmodule.exports = (app) => {\r\n app.get('*', handlePageRequest);\r\n};\r\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/fallback/index.js","const handlePageRender = require('helpers/handlePageRender.jsx');\n\nconst sendReactApp = (req, res) => {\n handlePageRender(req, res);\n};\n\nmodule.exports = sendReactApp;\n\n\n\n// WEBPACK FOOTER //\n// ./server/routes/fallback/sendReactApp.js"],"sourceRoot":""} \ No newline at end of file diff --git a/server/helpers/handlePageRender.jsx b/server/helpers/handlePageRender.jsx index e0cca3dc..ce9844bd 100644 --- a/server/helpers/handlePageRender.jsx +++ b/server/helpers/handlePageRender.jsx @@ -7,11 +7,16 @@ import { Reducers, GAListener, App } from 'spee.ch-components'; import renderFullPage from './renderFullPage.js'; import Helmet from 'react-helmet'; +const siteConfig = require('siteConfig.js'); + module.exports = (req, res) => { let context = {}; + // customize the reducer by passing in intial state configs + const customizedReducers = Reducers(siteConfig); + // create a new Redux store instance - const store = createStore(Reducers); + const store = createStore(customizedReducers); // render component to a string const html = renderToString( @@ -40,4 +45,6 @@ module.exports = (req, res) => { // send the rendered page back to the client res.send(renderFullPage(helmet, html, preloadedState)); + + console.log('hello from spee.ch handlePageRender.jsx'); }; diff --git a/server/helpers/handleShowRender.jsx b/server/helpers/handleShowRender.jsx index dfef14b1..5b6aa272 100644 --- a/server/helpers/handleShowRender.jsx +++ b/server/helpers/handleShowRender.jsx @@ -10,6 +10,10 @@ import { Reducers, GAListener, App, Sagas, Actions } from 'spee.ch-components'; import Helmet from 'react-helmet'; +// configure the reducers by passing initial state configs +const siteConfig = require('siteConfig.js'); +const customizedReducers = Reducers(siteConfig); + const returnSagaWithParams = (saga, params) => { return function * () { yield call(saga, params); @@ -24,7 +28,7 @@ module.exports = (req, res) => { const middleware = applyMiddleware(sagaMiddleware); // create a new Redux store instance - const store = createStore(Reducers, middleware); + const store = createStore(customizedReducers, middleware); // create saga const action = Actions.onHandleShowPageUri(req.params); @@ -60,4 +64,6 @@ module.exports = (req, res) => { // send the rendered page back to the client res.send(renderFullPage(helmet, html, preloadedState)); }); + + console.log('hello from spee.ch handleShowRender.jsx'); };