From 9d1eb7ae6af6ee458006481e5c233186060e646f Mon Sep 17 00:00:00 2001 From: Yann Stepienik Date: Sat, 11 Feb 2023 16:00:16 +0000 Subject: [PATCH] Init --- .bin/acorn | 1 + .bin/browserslist | 1 + .bin/browserslist-lint | 1 + .bin/envinfo | 1 + .bin/esbuild | 1 + .bin/import-local-fixture | 1 + .bin/jsesc | 1 + .bin/json5 | 1 + .bin/loose-envify | 1 + .bin/nanoid | 1 + .bin/node-which | 1 + .bin/resolve | 1 + .bin/rollup | 1 + .bin/semver | 1 + .bin/terser | 1 + .bin/tsc | 1 + .bin/tsserver | 1 + .bin/vite | 1 + .bin/webpack | 1 + .bin/webpack-cli | 1 + .gitignore | 10 +++ .gupm_rc.gs | 7 ++ build.gs | 5 ++ client/dist/assets/index-fb8d864c.js | 40 ++++++++++ client/dist/index.html | 14 ++++ client/index.html | 13 ++++ client/src/main.css | 0 client/src/main.tsx | 9 +++ generate-certificate.sh | 11 +++ gupm.json | 32 ++++++++ readme.md | 30 +++++++ self-signed-cert.ext | 4 + src/file/copy.go | 59 ++++++++++++++ src/file/delete.go | 31 ++++++++ src/file/get.go | 112 +++++++++++++++++++++++++++ src/file/list.go | 55 +++++++++++++ src/file/move.go | 39 ++++++++++ src/index.go | 106 +++++++++++++++++++++++++ src/user/login.go | 28 +++++++ src/user/register.go | 43 ++++++++++ src/utils/config.go | 53 +++++++++++++ src/utils/db.go | 54 +++++++++++++ src/utils/types.go | 29 +++++++ src/utils/utils.go | 38 +++++++++ src/utils/validator.go | 5 ++ vite.config.js | 9 +++ 46 files changed, 856 insertions(+) create mode 120000 .bin/acorn create mode 120000 .bin/browserslist create mode 120000 .bin/browserslist-lint create mode 120000 .bin/envinfo create mode 120000 .bin/esbuild create mode 120000 .bin/import-local-fixture create mode 120000 .bin/jsesc create mode 120000 .bin/json5 create mode 120000 .bin/loose-envify create mode 120000 .bin/nanoid create mode 120000 .bin/node-which create mode 120000 .bin/resolve create mode 120000 .bin/rollup create mode 120000 .bin/semver create mode 120000 .bin/terser create mode 120000 .bin/tsc create mode 120000 .bin/tsserver create mode 120000 .bin/vite create mode 120000 .bin/webpack create mode 120000 .bin/webpack-cli create mode 100644 .gitignore create mode 100644 .gupm_rc.gs create mode 100644 build.gs create mode 100644 client/dist/assets/index-fb8d864c.js create mode 100644 client/dist/index.html create mode 100644 client/index.html create mode 100644 client/src/main.css create mode 100644 client/src/main.tsx create mode 100644 generate-certificate.sh create mode 100644 gupm.json create mode 100644 readme.md create mode 100644 self-signed-cert.ext create mode 100644 src/file/copy.go create mode 100644 src/file/delete.go create mode 100644 src/file/get.go create mode 100644 src/file/list.go create mode 100644 src/file/move.go create mode 100644 src/index.go create mode 100644 src/user/login.go create mode 100644 src/user/register.go create mode 100644 src/utils/config.go create mode 100644 src/utils/db.go create mode 100644 src/utils/types.go create mode 100644 src/utils/utils.go create mode 100644 src/utils/validator.go create mode 100644 vite.config.js diff --git a/.bin/acorn b/.bin/acorn new file mode 120000 index 0000000..285f823 --- /dev/null +++ b/.bin/acorn @@ -0,0 +1 @@ +../node_modules/acorn/bin/acorn \ No newline at end of file diff --git a/.bin/browserslist b/.bin/browserslist new file mode 120000 index 0000000..45a546d --- /dev/null +++ b/.bin/browserslist @@ -0,0 +1 @@ +../node_modules/browserslist/cli.js \ No newline at end of file diff --git a/.bin/browserslist-lint b/.bin/browserslist-lint new file mode 120000 index 0000000..9d57b50 --- /dev/null +++ b/.bin/browserslist-lint @@ -0,0 +1 @@ +../node_modules/update-browserslist-db/cli.js \ No newline at end of file diff --git a/.bin/envinfo b/.bin/envinfo new file mode 120000 index 0000000..c7b4849 --- /dev/null +++ b/.bin/envinfo @@ -0,0 +1 @@ +../node_modules/envinfo/dist/cli.js \ No newline at end of file diff --git a/.bin/esbuild b/.bin/esbuild new file mode 120000 index 0000000..7c35f96 --- /dev/null +++ b/.bin/esbuild @@ -0,0 +1 @@ +../node_modules/esbuild/bin/esbuild \ No newline at end of file diff --git a/.bin/import-local-fixture b/.bin/import-local-fixture new file mode 120000 index 0000000..aafe379 --- /dev/null +++ b/.bin/import-local-fixture @@ -0,0 +1 @@ +../node_modules/import-local/fixtures/cli.js \ No newline at end of file diff --git a/.bin/jsesc b/.bin/jsesc new file mode 120000 index 0000000..9f8f1c6 --- /dev/null +++ b/.bin/jsesc @@ -0,0 +1 @@ +../node_modules/jsesc/bin/jsesc \ No newline at end of file diff --git a/.bin/json5 b/.bin/json5 new file mode 120000 index 0000000..402451b --- /dev/null +++ b/.bin/json5 @@ -0,0 +1 @@ +../node_modules/json5/lib/cli.js \ No newline at end of file diff --git a/.bin/loose-envify b/.bin/loose-envify new file mode 120000 index 0000000..c9810f8 --- /dev/null +++ b/.bin/loose-envify @@ -0,0 +1 @@ +../node_modules/loose-envify/cli.js \ No newline at end of file diff --git a/.bin/nanoid b/.bin/nanoid new file mode 120000 index 0000000..b947531 --- /dev/null +++ b/.bin/nanoid @@ -0,0 +1 @@ +../node_modules/nanoid/bin/nanoid.cjs \ No newline at end of file diff --git a/.bin/node-which b/.bin/node-which new file mode 120000 index 0000000..0974888 --- /dev/null +++ b/.bin/node-which @@ -0,0 +1 @@ +../node_modules/which/bin/node-which \ No newline at end of file diff --git a/.bin/resolve b/.bin/resolve new file mode 120000 index 0000000..a228729 --- /dev/null +++ b/.bin/resolve @@ -0,0 +1 @@ +../node_modules/resolve/bin/resolve \ No newline at end of file diff --git a/.bin/rollup b/.bin/rollup new file mode 120000 index 0000000..9375925 --- /dev/null +++ b/.bin/rollup @@ -0,0 +1 @@ +../node_modules/rollup/dist/bin/rollup \ No newline at end of file diff --git a/.bin/semver b/.bin/semver new file mode 120000 index 0000000..20ecd89 --- /dev/null +++ b/.bin/semver @@ -0,0 +1 @@ +../node_modules/semver/bin/semver.js \ No newline at end of file diff --git a/.bin/terser b/.bin/terser new file mode 120000 index 0000000..ad7cf43 --- /dev/null +++ b/.bin/terser @@ -0,0 +1 @@ +../node_modules/terser/bin/terser \ No newline at end of file diff --git a/.bin/tsc b/.bin/tsc new file mode 120000 index 0000000..99d9a25 --- /dev/null +++ b/.bin/tsc @@ -0,0 +1 @@ +../node_modules/typescript/bin/tsc \ No newline at end of file diff --git a/.bin/tsserver b/.bin/tsserver new file mode 120000 index 0000000..f32bd93 --- /dev/null +++ b/.bin/tsserver @@ -0,0 +1 @@ +../node_modules/typescript/bin/tsserver \ No newline at end of file diff --git a/.bin/vite b/.bin/vite new file mode 120000 index 0000000..471f7fd --- /dev/null +++ b/.bin/vite @@ -0,0 +1 @@ +../node_modules/vite/bin/vite.js \ No newline at end of file diff --git a/.bin/webpack b/.bin/webpack new file mode 120000 index 0000000..6a1e920 --- /dev/null +++ b/.bin/webpack @@ -0,0 +1 @@ +../node_modules/webpack/bin/webpack.js \ No newline at end of file diff --git a/.bin/webpack-cli b/.bin/webpack-cli new file mode 120000 index 0000000..168e080 --- /dev/null +++ b/.bin/webpack-cli @@ -0,0 +1 @@ +../node_modules/webpack-cli/bin/cli.js \ No newline at end of file diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..61122a4 --- /dev/null +++ b/.gitignore @@ -0,0 +1,10 @@ +go_modules +gupm_modules +node_modules +build +dev.gs +localcert.csr +localcert.crt +localcert.key +.vite +dev.json \ No newline at end of file diff --git a/.gupm_rc.gs b/.gupm_rc.gs new file mode 100644 index 0000000..a73a066 --- /dev/null +++ b/.gupm_rc.gs @@ -0,0 +1,7 @@ +// look up dependencies in local go_modules and gupm_modules directories +env("GOPATH", run("go", ["env", "GOROOT"]) + ":" + pwd() + "/go_modules" + ":" + pwd() + "/gupm_modules") + +// dev mode +env("MONGODB", readJsonFile("dev.json").MONGODB) +env("HTTP_PORT", 8080) +env("HTTPS_PORT", 8443) \ No newline at end of file diff --git a/build.gs b/build.gs new file mode 100644 index 0000000..bec40ba --- /dev/null +++ b/build.gs @@ -0,0 +1,5 @@ +removeFiles(["build"]); +var goArgs = ["build", "-o", "build/bin"] +goArgs = goArgs.concat(dir("src/*.go")) +exec("go", goArgs); +copyFiles("client/dist/", "build/static/") \ No newline at end of file diff --git a/client/dist/assets/index-fb8d864c.js b/client/dist/assets/index-fb8d864c.js new file mode 100644 index 0000000..e2dab06 --- /dev/null +++ b/client/dist/assets/index-fb8d864c.js @@ -0,0 +1,40 @@ +(function(){const n=document.createElement("link").relList;if(n&&n.supports&&n.supports("modulepreload"))return;for(const l of document.querySelectorAll('link[rel="modulepreload"]'))r(l);new MutationObserver(l=>{for(const u of l)if(u.type==="childList")for(const o of u.addedNodes)o.tagName==="LINK"&&o.rel==="modulepreload"&&r(o)}).observe(document,{childList:!0,subtree:!0});function t(l){const u={};return l.integrity&&(u.integrity=l.integrity),l.referrerpolicy&&(u.referrerPolicy=l.referrerpolicy),l.crossorigin==="use-credentials"?u.credentials="include":l.crossorigin==="anonymous"?u.credentials="omit":u.credentials="same-origin",u}function r(l){if(l.ep)return;l.ep=!0;const u=t(l);fetch(l.href,u)}})();function nc(e){return e&&e.__esModule&&Object.prototype.hasOwnProperty.call(e,"default")?e.default:e}var Wl={},tc={get exports(){return Wl},set exports(e){Wl=e}},br={},Lt={},rc={get exports(){return Lt},set exports(e){Lt=e}},L={};/** + * @license React + * react.production.min.js + * + * Copyright (c) Facebook, Inc. and its affiliates. + * + * This source code is licensed under the MIT license found in the + * LICENSE file in the root directory of this source tree. + */var Yt=Symbol.for("react.element"),lc=Symbol.for("react.portal"),uc=Symbol.for("react.fragment"),oc=Symbol.for("react.strict_mode"),ic=Symbol.for("react.profiler"),sc=Symbol.for("react.provider"),ac=Symbol.for("react.context"),cc=Symbol.for("react.forward_ref"),fc=Symbol.for("react.suspense"),dc=Symbol.for("react.memo"),pc=Symbol.for("react.lazy"),Fo=Symbol.iterator;function mc(e){return e===null||typeof e!="object"?null:(e=Fo&&e[Fo]||e["@@iterator"],typeof e=="function"?e:null)}var Ki={isMounted:function(){return!1},enqueueForceUpdate:function(){},enqueueReplaceState:function(){},enqueueSetState:function(){}},Yi=Object.assign,Xi={};function rt(e,n,t){this.props=e,this.context=n,this.refs=Xi,this.updater=t||Ki}rt.prototype.isReactComponent={};rt.prototype.setState=function(e,n){if(typeof e!="object"&&typeof e!="function"&&e!=null)throw Error("setState(...): takes an object of state variables to update or a function which returns an object of state variables.");this.updater.enqueueSetState(this,e,n,"setState")};rt.prototype.forceUpdate=function(e){this.updater.enqueueForceUpdate(this,e,"forceUpdate")};function Gi(){}Gi.prototype=rt.prototype;function Vu(e,n,t){this.props=e,this.context=n,this.refs=Xi,this.updater=t||Ki}var Au=Vu.prototype=new Gi;Au.constructor=Vu;Yi(Au,rt.prototype);Au.isPureReactComponent=!0;var jo=Array.isArray,Zi=Object.prototype.hasOwnProperty,Bu={current:null},Ji={key:!0,ref:!0,__self:!0,__source:!0};function qi(e,n,t){var r,l={},u=null,o=null;if(n!=null)for(r in n.ref!==void 0&&(o=n.ref),n.key!==void 0&&(u=""+n.key),n)Zi.call(n,r)&&!Ji.hasOwnProperty(r)&&(l[r]=n[r]);var i=arguments.length-2;if(i===1)l.children=t;else if(1>>1,X=C[H];if(0>>1;Hl(yl,z))hnl(bt,yl)?(C[H]=bt,C[hn]=z,H=hn):(C[H]=yl,C[vn]=z,H=vn);else if(hnl(bt,z))C[H]=bt,C[hn]=z,H=hn;else break e}}return N}function l(C,N){var z=C.sortIndex-N.sortIndex;return z!==0?z:C.id-N.id}if(typeof performance=="object"&&typeof performance.now=="function"){var u=performance;e.unstable_now=function(){return u.now()}}else{var o=Date,i=o.now();e.unstable_now=function(){return o.now()-i}}var s=[],c=[],v=1,m=null,p=3,g=!1,w=!1,k=!1,F=typeof setTimeout=="function"?setTimeout:null,f=typeof clearTimeout=="function"?clearTimeout:null,a=typeof setImmediate<"u"?setImmediate:null;typeof navigator<"u"&&navigator.scheduling!==void 0&&navigator.scheduling.isInputPending!==void 0&&navigator.scheduling.isInputPending.bind(navigator.scheduling);function d(C){for(var N=t(c);N!==null;){if(N.callback===null)r(c);else if(N.startTime<=C)r(c),N.sortIndex=N.expirationTime,n(s,N);else break;N=t(c)}}function h(C){if(k=!1,d(C),!w)if(t(s)!==null)w=!0,vl(E);else{var N=t(c);N!==null&&hl(h,N.startTime-C)}}function E(C,N){w=!1,k&&(k=!1,f(P),P=-1),g=!0;var z=p;try{for(d(N),m=t(s);m!==null&&(!(m.expirationTime>N)||C&&!xe());){var H=m.callback;if(typeof H=="function"){m.callback=null,p=m.priorityLevel;var X=H(m.expirationTime<=N);N=e.unstable_now(),typeof X=="function"?m.callback=X:m===t(s)&&r(s),d(N)}else r(s);m=t(s)}if(m!==null)var qt=!0;else{var vn=t(c);vn!==null&&hl(h,vn.startTime-N),qt=!1}return qt}finally{m=null,p=z,g=!1}}var _=!1,x=null,P=-1,B=5,T=-1;function xe(){return!(e.unstable_now()-TC||125H?(C.sortIndex=z,n(c,C),t(s)===null&&C===t(c)&&(k?(f(P),P=-1):k=!0,hl(h,z-H))):(C.sortIndex=X,n(s,C),w||g||(w=!0,vl(E))),C},e.unstable_shouldYield=xe,e.unstable_wrapCallback=function(C){var N=p;return function(){var z=p;p=N;try{return C.apply(this,arguments)}finally{p=z}}}})(es);(function(e){e.exports=es})(Nc);/** + * @license React + * react-dom.production.min.js + * + * Copyright (c) Facebook, Inc. and its affiliates. + * + * This source code is licensed under the MIT license found in the + * LICENSE file in the root directory of this source tree. + */var ns=Lt,he=Yl;function y(e){for(var n="https://reactjs.org/docs/error-decoder.html?invariant="+e,t=1;t"u"||typeof window.document>"u"||typeof window.document.createElement>"u"),Xl=Object.prototype.hasOwnProperty,zc=/^[:A-Z_a-z\u00C0-\u00D6\u00D8-\u00F6\u00F8-\u02FF\u0370-\u037D\u037F-\u1FFF\u200C-\u200D\u2070-\u218F\u2C00-\u2FEF\u3001-\uD7FF\uF900-\uFDCF\uFDF0-\uFFFD][:A-Z_a-z\u00C0-\u00D6\u00D8-\u00F6\u00F8-\u02FF\u0370-\u037D\u037F-\u1FFF\u200C-\u200D\u2070-\u218F\u2C00-\u2FEF\u3001-\uD7FF\uF900-\uFDCF\uFDF0-\uFFFD\-.0-9\u00B7\u0300-\u036F\u203F-\u2040]*$/,Vo={},Ao={};function Lc(e){return Xl.call(Ao,e)?!0:Xl.call(Vo,e)?!1:zc.test(e)?Ao[e]=!0:(Vo[e]=!0,!1)}function Tc(e,n,t,r){if(t!==null&&t.type===0)return!1;switch(typeof n){case"function":case"symbol":return!0;case"boolean":return r?!1:t!==null?!t.acceptsBooleans:(e=e.toLowerCase().slice(0,5),e!=="data-"&&e!=="aria-");default:return!1}}function Rc(e,n,t,r){if(n===null||typeof n>"u"||Tc(e,n,t,r))return!0;if(r)return!1;if(t!==null)switch(t.type){case 3:return!n;case 4:return n===!1;case 5:return isNaN(n);case 6:return isNaN(n)||1>n}return!1}function ie(e,n,t,r,l,u,o){this.acceptsBooleans=n===2||n===3||n===4,this.attributeName=r,this.attributeNamespace=l,this.mustUseProperty=t,this.propertyName=e,this.type=n,this.sanitizeURL=u,this.removeEmptyString=o}var b={};"children dangerouslySetInnerHTML defaultValue defaultChecked innerHTML suppressContentEditableWarning suppressHydrationWarning style".split(" ").forEach(function(e){b[e]=new ie(e,0,!1,e,null,!1,!1)});[["acceptCharset","accept-charset"],["className","class"],["htmlFor","for"],["httpEquiv","http-equiv"]].forEach(function(e){var n=e[0];b[n]=new ie(n,1,!1,e[1],null,!1,!1)});["contentEditable","draggable","spellCheck","value"].forEach(function(e){b[e]=new ie(e,2,!1,e.toLowerCase(),null,!1,!1)});["autoReverse","externalResourcesRequired","focusable","preserveAlpha"].forEach(function(e){b[e]=new ie(e,2,!1,e,null,!1,!1)});"allowFullScreen async autoFocus autoPlay controls default defer disabled disablePictureInPicture disableRemotePlayback formNoValidate hidden loop noModule noValidate open playsInline readOnly required reversed scoped seamless itemScope".split(" ").forEach(function(e){b[e]=new ie(e,3,!1,e.toLowerCase(),null,!1,!1)});["checked","multiple","muted","selected"].forEach(function(e){b[e]=new ie(e,3,!0,e,null,!1,!1)});["capture","download"].forEach(function(e){b[e]=new ie(e,4,!1,e,null,!1,!1)});["cols","rows","size","span"].forEach(function(e){b[e]=new ie(e,6,!1,e,null,!1,!1)});["rowSpan","start"].forEach(function(e){b[e]=new ie(e,5,!1,e.toLowerCase(),null,!1,!1)});var Wu=/[\-:]([a-z])/g;function Qu(e){return e[1].toUpperCase()}"accent-height alignment-baseline arabic-form baseline-shift cap-height clip-path clip-rule color-interpolation color-interpolation-filters color-profile color-rendering dominant-baseline enable-background fill-opacity fill-rule flood-color flood-opacity font-family font-size font-size-adjust font-stretch font-style font-variant font-weight glyph-name glyph-orientation-horizontal glyph-orientation-vertical horiz-adv-x horiz-origin-x image-rendering letter-spacing lighting-color marker-end marker-mid marker-start overline-position overline-thickness paint-order panose-1 pointer-events rendering-intent shape-rendering stop-color stop-opacity strikethrough-position strikethrough-thickness stroke-dasharray stroke-dashoffset stroke-linecap stroke-linejoin stroke-miterlimit stroke-opacity stroke-width text-anchor text-decoration text-rendering underline-position underline-thickness unicode-bidi unicode-range units-per-em v-alphabetic v-hanging v-ideographic v-mathematical vector-effect vert-adv-y vert-origin-x vert-origin-y word-spacing writing-mode xmlns:xlink x-height".split(" ").forEach(function(e){var n=e.replace(Wu,Qu);b[n]=new ie(n,1,!1,e,null,!1,!1)});"xlink:actuate xlink:arcrole xlink:role xlink:show xlink:title xlink:type".split(" ").forEach(function(e){var n=e.replace(Wu,Qu);b[n]=new ie(n,1,!1,e,"http://www.w3.org/1999/xlink",!1,!1)});["xml:base","xml:lang","xml:space"].forEach(function(e){var n=e.replace(Wu,Qu);b[n]=new ie(n,1,!1,e,"http://www.w3.org/XML/1998/namespace",!1,!1)});["tabIndex","crossOrigin"].forEach(function(e){b[e]=new ie(e,1,!1,e.toLowerCase(),null,!1,!1)});b.xlinkHref=new ie("xlinkHref",1,!1,"xlink:href","http://www.w3.org/1999/xlink",!0,!1);["src","href","action","formAction"].forEach(function(e){b[e]=new ie(e,1,!1,e.toLowerCase(),null,!0,!0)});function Ku(e,n,t,r){var l=b.hasOwnProperty(n)?b[n]:null;(l!==null?l.type!==0:r||!(2i||l[o]!==u[i]){var s=` +`+l[o].replace(" at new "," at ");return e.displayName&&s.includes("")&&(s=s.replace("",e.displayName)),s}while(1<=o&&0<=i);break}}}finally{kl=!1,Error.prepareStackTrace=t}return(e=e?e.displayName||e.name:"")?ht(e):""}function Oc(e){switch(e.tag){case 5:return ht(e.type);case 16:return ht("Lazy");case 13:return ht("Suspense");case 19:return ht("SuspenseList");case 0:case 2:case 15:return e=Sl(e.type,!1),e;case 11:return e=Sl(e.type.render,!1),e;case 1:return e=Sl(e.type,!0),e;default:return""}}function ql(e){if(e==null)return null;if(typeof e=="function")return e.displayName||e.name||null;if(typeof e=="string")return e;switch(e){case Mn:return"Fragment";case On:return"Portal";case Gl:return"Profiler";case Yu:return"StrictMode";case Zl:return"Suspense";case Jl:return"SuspenseList"}if(typeof e=="object")switch(e.$$typeof){case ls:return(e.displayName||"Context")+".Consumer";case rs:return(e._context.displayName||"Context")+".Provider";case Xu:var n=e.render;return e=e.displayName,e||(e=n.displayName||n.name||"",e=e!==""?"ForwardRef("+e+")":"ForwardRef"),e;case Gu:return n=e.displayName||null,n!==null?n:ql(e.type)||"Memo";case Ge:n=e._payload,e=e._init;try{return ql(e(n))}catch{}}return null}function Mc(e){var n=e.type;switch(e.tag){case 24:return"Cache";case 9:return(n.displayName||"Context")+".Consumer";case 10:return(n._context.displayName||"Context")+".Provider";case 18:return"DehydratedFragment";case 11:return e=n.render,e=e.displayName||e.name||"",n.displayName||(e!==""?"ForwardRef("+e+")":"ForwardRef");case 7:return"Fragment";case 5:return n;case 4:return"Portal";case 3:return"Root";case 6:return"Text";case 16:return ql(n);case 8:return n===Yu?"StrictMode":"Mode";case 22:return"Offscreen";case 12:return"Profiler";case 21:return"Scope";case 13:return"Suspense";case 19:return"SuspenseList";case 25:return"TracingMarker";case 1:case 0:case 17:case 2:case 14:case 15:if(typeof n=="function")return n.displayName||n.name||null;if(typeof n=="string")return n}return null}function cn(e){switch(typeof e){case"boolean":case"number":case"string":case"undefined":return e;case"object":return e;default:return""}}function os(e){var n=e.type;return(e=e.nodeName)&&e.toLowerCase()==="input"&&(n==="checkbox"||n==="radio")}function Dc(e){var n=os(e)?"checked":"value",t=Object.getOwnPropertyDescriptor(e.constructor.prototype,n),r=""+e[n];if(!e.hasOwnProperty(n)&&typeof t<"u"&&typeof t.get=="function"&&typeof t.set=="function"){var l=t.get,u=t.set;return Object.defineProperty(e,n,{configurable:!0,get:function(){return l.call(this)},set:function(o){r=""+o,u.call(this,o)}}),Object.defineProperty(e,n,{enumerable:t.enumerable}),{getValue:function(){return r},setValue:function(o){r=""+o},stopTracking:function(){e._valueTracker=null,delete e[n]}}}}function tr(e){e._valueTracker||(e._valueTracker=Dc(e))}function is(e){if(!e)return!1;var n=e._valueTracker;if(!n)return!0;var t=n.getValue(),r="";return e&&(r=os(e)?e.checked?"true":"false":e.value),e=r,e!==t?(n.setValue(e),!0):!1}function Lr(e){if(e=e||(typeof document<"u"?document:void 0),typeof e>"u")return null;try{return e.activeElement||e.body}catch{return e.body}}function bl(e,n){var t=n.checked;return V({},n,{defaultChecked:void 0,defaultValue:void 0,value:void 0,checked:t??e._wrapperState.initialChecked})}function Ho(e,n){var t=n.defaultValue==null?"":n.defaultValue,r=n.checked!=null?n.checked:n.defaultChecked;t=cn(n.value!=null?n.value:t),e._wrapperState={initialChecked:r,initialValue:t,controlled:n.type==="checkbox"||n.type==="radio"?n.checked!=null:n.value!=null}}function ss(e,n){n=n.checked,n!=null&&Ku(e,"checked",n,!1)}function eu(e,n){ss(e,n);var t=cn(n.value),r=n.type;if(t!=null)r==="number"?(t===0&&e.value===""||e.value!=t)&&(e.value=""+t):e.value!==""+t&&(e.value=""+t);else if(r==="submit"||r==="reset"){e.removeAttribute("value");return}n.hasOwnProperty("value")?nu(e,n.type,t):n.hasOwnProperty("defaultValue")&&nu(e,n.type,cn(n.defaultValue)),n.checked==null&&n.defaultChecked!=null&&(e.defaultChecked=!!n.defaultChecked)}function Wo(e,n,t){if(n.hasOwnProperty("value")||n.hasOwnProperty("defaultValue")){var r=n.type;if(!(r!=="submit"&&r!=="reset"||n.value!==void 0&&n.value!==null))return;n=""+e._wrapperState.initialValue,t||n===e.value||(e.value=n),e.defaultValue=n}t=e.name,t!==""&&(e.name=""),e.defaultChecked=!!e._wrapperState.initialChecked,t!==""&&(e.name=t)}function nu(e,n,t){(n!=="number"||Lr(e.ownerDocument)!==e)&&(t==null?e.defaultValue=""+e._wrapperState.initialValue:e.defaultValue!==""+t&&(e.defaultValue=""+t))}var yt=Array.isArray;function Wn(e,n,t,r){if(e=e.options,n){n={};for(var l=0;l"+n.valueOf().toString()+"",n=rr.firstChild;e.firstChild;)e.removeChild(e.firstChild);for(;n.firstChild;)e.appendChild(n.firstChild)}});function Rt(e,n){if(n){var t=e.firstChild;if(t&&t===e.lastChild&&t.nodeType===3){t.nodeValue=n;return}}e.textContent=n}var kt={animationIterationCount:!0,aspectRatio:!0,borderImageOutset:!0,borderImageSlice:!0,borderImageWidth:!0,boxFlex:!0,boxFlexGroup:!0,boxOrdinalGroup:!0,columnCount:!0,columns:!0,flex:!0,flexGrow:!0,flexPositive:!0,flexShrink:!0,flexNegative:!0,flexOrder:!0,gridArea:!0,gridRow:!0,gridRowEnd:!0,gridRowSpan:!0,gridRowStart:!0,gridColumn:!0,gridColumnEnd:!0,gridColumnSpan:!0,gridColumnStart:!0,fontWeight:!0,lineClamp:!0,lineHeight:!0,opacity:!0,order:!0,orphans:!0,tabSize:!0,widows:!0,zIndex:!0,zoom:!0,fillOpacity:!0,floodOpacity:!0,stopOpacity:!0,strokeDasharray:!0,strokeDashoffset:!0,strokeMiterlimit:!0,strokeOpacity:!0,strokeWidth:!0},Ic=["Webkit","ms","Moz","O"];Object.keys(kt).forEach(function(e){Ic.forEach(function(n){n=n+e.charAt(0).toUpperCase()+e.substring(1),kt[n]=kt[e]})});function ds(e,n,t){return n==null||typeof n=="boolean"||n===""?"":t||typeof n!="number"||n===0||kt.hasOwnProperty(e)&&kt[e]?(""+n).trim():n+"px"}function ps(e,n){e=e.style;for(var t in n)if(n.hasOwnProperty(t)){var r=t.indexOf("--")===0,l=ds(t,n[t],r);t==="float"&&(t="cssFloat"),r?e.setProperty(t,l):e[t]=l}}var Fc=V({menuitem:!0},{area:!0,base:!0,br:!0,col:!0,embed:!0,hr:!0,img:!0,input:!0,keygen:!0,link:!0,meta:!0,param:!0,source:!0,track:!0,wbr:!0});function lu(e,n){if(n){if(Fc[e]&&(n.children!=null||n.dangerouslySetInnerHTML!=null))throw Error(y(137,e));if(n.dangerouslySetInnerHTML!=null){if(n.children!=null)throw Error(y(60));if(typeof n.dangerouslySetInnerHTML!="object"||!("__html"in n.dangerouslySetInnerHTML))throw Error(y(61))}if(n.style!=null&&typeof n.style!="object")throw Error(y(62))}}function uu(e,n){if(e.indexOf("-")===-1)return typeof n.is=="string";switch(e){case"annotation-xml":case"color-profile":case"font-face":case"font-face-src":case"font-face-uri":case"font-face-format":case"font-face-name":case"missing-glyph":return!1;default:return!0}}var ou=null;function Zu(e){return e=e.target||e.srcElement||window,e.correspondingUseElement&&(e=e.correspondingUseElement),e.nodeType===3?e.parentNode:e}var iu=null,Qn=null,Kn=null;function Yo(e){if(e=Zt(e)){if(typeof iu!="function")throw Error(y(280));var n=e.stateNode;n&&(n=ll(n),iu(e.stateNode,e.type,n))}}function ms(e){Qn?Kn?Kn.push(e):Kn=[e]:Qn=e}function vs(){if(Qn){var e=Qn,n=Kn;if(Kn=Qn=null,Yo(e),n)for(e=0;e>>=0,e===0?32:31-(Yc(e)/Xc|0)|0}var lr=64,ur=4194304;function gt(e){switch(e&-e){case 1:return 1;case 2:return 2;case 4:return 4;case 8:return 8;case 16:return 16;case 32:return 32;case 64:case 128:case 256:case 512:case 1024:case 2048:case 4096:case 8192:case 16384:case 32768:case 65536:case 131072:case 262144:case 524288:case 1048576:case 2097152:return e&4194240;case 4194304:case 8388608:case 16777216:case 33554432:case 67108864:return e&130023424;case 134217728:return 134217728;case 268435456:return 268435456;case 536870912:return 536870912;case 1073741824:return 1073741824;default:return e}}function Mr(e,n){var t=e.pendingLanes;if(t===0)return 0;var r=0,l=e.suspendedLanes,u=e.pingedLanes,o=t&268435455;if(o!==0){var i=o&~l;i!==0?r=gt(i):(u&=o,u!==0&&(r=gt(u)))}else o=t&~l,o!==0?r=gt(o):u!==0&&(r=gt(u));if(r===0)return 0;if(n!==0&&n!==r&&!(n&l)&&(l=r&-r,u=n&-n,l>=u||l===16&&(u&4194240)!==0))return n;if(r&4&&(r|=t&16),n=e.entangledLanes,n!==0)for(e=e.entanglements,n&=r;0t;t++)n.push(e);return n}function Xt(e,n,t){e.pendingLanes|=n,n!==536870912&&(e.suspendedLanes=0,e.pingedLanes=0),e=e.eventTimes,n=31-Te(n),e[n]=t}function qc(e,n){var t=e.pendingLanes&~n;e.pendingLanes=n,e.suspendedLanes=0,e.pingedLanes=0,e.expiredLanes&=n,e.mutableReadLanes&=n,e.entangledLanes&=n,n=e.entanglements;var r=e.eventTimes;for(e=e.expirationTimes;0=Et),ti=String.fromCharCode(32),ri=!1;function Is(e,n){switch(e){case"keyup":return Nf.indexOf(n.keyCode)!==-1;case"keydown":return n.keyCode!==229;case"keypress":case"mousedown":case"focusout":return!0;default:return!1}}function Fs(e){return e=e.detail,typeof e=="object"&&"data"in e?e.data:null}var Dn=!1;function Lf(e,n){switch(e){case"compositionend":return Fs(n);case"keypress":return n.which!==32?null:(ri=!0,ti);case"textInput":return e=n.data,e===ti&&ri?null:e;default:return null}}function Tf(e,n){if(Dn)return e==="compositionend"||!lo&&Is(e,n)?(e=Ms(),kr=no=be=null,Dn=!1,e):null;switch(e){case"paste":return null;case"keypress":if(!(n.ctrlKey||n.altKey||n.metaKey)||n.ctrlKey&&n.altKey){if(n.char&&1=n)return{node:t,offset:n-e};e=r}e:{for(;t;){if(t.nextSibling){t=t.nextSibling;break e}t=t.parentNode}t=void 0}t=ii(t)}}function Vs(e,n){return e&&n?e===n?!0:e&&e.nodeType===3?!1:n&&n.nodeType===3?Vs(e,n.parentNode):"contains"in e?e.contains(n):e.compareDocumentPosition?!!(e.compareDocumentPosition(n)&16):!1:!1}function As(){for(var e=window,n=Lr();n instanceof e.HTMLIFrameElement;){try{var t=typeof n.contentWindow.location.href=="string"}catch{t=!1}if(t)e=n.contentWindow;else break;n=Lr(e.document)}return n}function uo(e){var n=e&&e.nodeName&&e.nodeName.toLowerCase();return n&&(n==="input"&&(e.type==="text"||e.type==="search"||e.type==="tel"||e.type==="url"||e.type==="password")||n==="textarea"||e.contentEditable==="true")}function $f(e){var n=As(),t=e.focusedElem,r=e.selectionRange;if(n!==t&&t&&t.ownerDocument&&Vs(t.ownerDocument.documentElement,t)){if(r!==null&&uo(t)){if(n=r.start,e=r.end,e===void 0&&(e=n),"selectionStart"in t)t.selectionStart=n,t.selectionEnd=Math.min(e,t.value.length);else if(e=(n=t.ownerDocument||document)&&n.defaultView||window,e.getSelection){e=e.getSelection();var l=t.textContent.length,u=Math.min(r.start,l);r=r.end===void 0?u:Math.min(r.end,l),!e.extend&&u>r&&(l=r,r=u,u=l),l=si(t,u);var o=si(t,r);l&&o&&(e.rangeCount!==1||e.anchorNode!==l.node||e.anchorOffset!==l.offset||e.focusNode!==o.node||e.focusOffset!==o.offset)&&(n=n.createRange(),n.setStart(l.node,l.offset),e.removeAllRanges(),u>r?(e.addRange(n),e.extend(o.node,o.offset)):(n.setEnd(o.node,o.offset),e.addRange(n)))}}for(n=[],e=t;e=e.parentNode;)e.nodeType===1&&n.push({element:e,left:e.scrollLeft,top:e.scrollTop});for(typeof t.focus=="function"&&t.focus(),t=0;t=document.documentMode,In=null,pu=null,_t=null,mu=!1;function ai(e,n,t){var r=t.window===t?t.document:t.nodeType===9?t:t.ownerDocument;mu||In==null||In!==Lr(r)||(r=In,"selectionStart"in r&&uo(r)?r={start:r.selectionStart,end:r.selectionEnd}:(r=(r.ownerDocument&&r.ownerDocument.defaultView||window).getSelection(),r={anchorNode:r.anchorNode,anchorOffset:r.anchorOffset,focusNode:r.focusNode,focusOffset:r.focusOffset}),_t&&jt(_t,r)||(_t=r,r=Fr(pu,"onSelect"),0Un||(e.current=ku[Un],ku[Un]=null,Un--)}function M(e,n){Un++,ku[Un]=e.current,e.current=n}var fn={},re=pn(fn),ce=pn(!1),_n=fn;function Jn(e,n){var t=e.type.contextTypes;if(!t)return fn;var r=e.stateNode;if(r&&r.__reactInternalMemoizedUnmaskedChildContext===n)return r.__reactInternalMemoizedMaskedChildContext;var l={},u;for(u in t)l[u]=n[u];return r&&(e=e.stateNode,e.__reactInternalMemoizedUnmaskedChildContext=n,e.__reactInternalMemoizedMaskedChildContext=l),l}function fe(e){return e=e.childContextTypes,e!=null}function Ur(){I(ce),I(re)}function hi(e,n,t){if(re.current!==fn)throw Error(y(168));M(re,n),M(ce,t)}function Zs(e,n,t){var r=e.stateNode;if(n=n.childContextTypes,typeof r.getChildContext!="function")return t;r=r.getChildContext();for(var l in r)if(!(l in n))throw Error(y(108,Mc(e)||"Unknown",l));return V({},t,r)}function $r(e){return e=(e=e.stateNode)&&e.__reactInternalMemoizedMergedChildContext||fn,_n=re.current,M(re,e),M(ce,ce.current),!0}function yi(e,n,t){var r=e.stateNode;if(!r)throw Error(y(169));t?(e=Zs(e,n,_n),r.__reactInternalMemoizedMergedChildContext=e,I(ce),I(re),M(re,e)):I(ce),M(ce,t)}var $e=null,ul=!1,Il=!1;function Js(e){$e===null?$e=[e]:$e.push(e)}function Jf(e){ul=!0,Js(e)}function mn(){if(!Il&&$e!==null){Il=!0;var e=0,n=O;try{var t=$e;for(O=1;e>=o,l-=o,Ve=1<<32-Te(n)+l|t<P?(B=x,x=null):B=x.sibling;var T=p(f,x,d[P],h);if(T===null){x===null&&(x=B);break}e&&x&&T.alternate===null&&n(f,x),a=u(T,a,P),_===null?E=T:_.sibling=T,_=T,x=B}if(P===d.length)return t(f,x),j&&yn(f,P),E;if(x===null){for(;PP?(B=x,x=null):B=x.sibling;var xe=p(f,x,T.value,h);if(xe===null){x===null&&(x=B);break}e&&x&&xe.alternate===null&&n(f,x),a=u(xe,a,P),_===null?E=xe:_.sibling=xe,_=xe,x=B}if(T.done)return t(f,x),j&&yn(f,P),E;if(x===null){for(;!T.done;P++,T=d.next())T=m(f,T.value,h),T!==null&&(a=u(T,a,P),_===null?E=T:_.sibling=T,_=T);return j&&yn(f,P),E}for(x=r(f,x);!T.done;P++,T=d.next())T=g(x,f,P,T.value,h),T!==null&&(e&&T.alternate!==null&&x.delete(T.key===null?P:T.key),a=u(T,a,P),_===null?E=T:_.sibling=T,_=T);return e&&x.forEach(function(ot){return n(f,ot)}),j&&yn(f,P),E}function F(f,a,d,h){if(typeof d=="object"&&d!==null&&d.type===Mn&&d.key===null&&(d=d.props.children),typeof d=="object"&&d!==null){switch(d.$$typeof){case nr:e:{for(var E=d.key,_=a;_!==null;){if(_.key===E){if(E=d.type,E===Mn){if(_.tag===7){t(f,_.sibling),a=l(_,d.props.children),a.return=f,f=a;break e}}else if(_.elementType===E||typeof E=="object"&&E!==null&&E.$$typeof===Ge&&_i(E)===_.type){t(f,_.sibling),a=l(_,d.props),a.ref=pt(f,_,d),a.return=f,f=a;break e}t(f,_);break}else n(f,_);_=_.sibling}d.type===Mn?(a=Cn(d.props.children,f.mode,h,d.key),a.return=f,f=a):(h=zr(d.type,d.key,d.props,null,f.mode,h),h.ref=pt(f,a,d),h.return=f,f=h)}return o(f);case On:e:{for(_=d.key;a!==null;){if(a.key===_)if(a.tag===4&&a.stateNode.containerInfo===d.containerInfo&&a.stateNode.implementation===d.implementation){t(f,a.sibling),a=l(a,d.children||[]),a.return=f,f=a;break e}else{t(f,a);break}else n(f,a);a=a.sibling}a=Hl(d,f.mode,h),a.return=f,f=a}return o(f);case Ge:return _=d._init,F(f,a,_(d._payload),h)}if(yt(d))return w(f,a,d,h);if(st(d))return k(f,a,d,h);dr(f,d)}return typeof d=="string"&&d!==""||typeof d=="number"?(d=""+d,a!==null&&a.tag===6?(t(f,a.sibling),a=l(a,d),a.return=f,f=a):(t(f,a),a=Bl(d,f.mode,h),a.return=f,f=a),o(f)):t(f,a)}return F}var bn=ua(!0),oa=ua(!1),Jt={},je=pn(Jt),At=pn(Jt),Bt=pn(Jt);function Sn(e){if(e===Jt)throw Error(y(174));return e}function vo(e,n){switch(M(Bt,n),M(At,e),M(je,Jt),e=n.nodeType,e){case 9:case 11:n=(n=n.documentElement)?n.namespaceURI:ru(null,"");break;default:e=e===8?n.parentNode:n,n=e.namespaceURI||null,e=e.tagName,n=ru(n,e)}I(je),M(je,n)}function et(){I(je),I(At),I(Bt)}function ia(e){Sn(Bt.current);var n=Sn(je.current),t=ru(n,e.type);n!==t&&(M(At,e),M(je,t))}function ho(e){At.current===e&&(I(je),I(At))}var U=pn(0);function Qr(e){for(var n=e;n!==null;){if(n.tag===13){var t=n.memoizedState;if(t!==null&&(t=t.dehydrated,t===null||t.data==="$?"||t.data==="$!"))return n}else if(n.tag===19&&n.memoizedProps.revealOrder!==void 0){if(n.flags&128)return n}else if(n.child!==null){n.child.return=n,n=n.child;continue}if(n===e)break;for(;n.sibling===null;){if(n.return===null||n.return===e)return null;n=n.return}n.sibling.return=n.return,n=n.sibling}return null}var Fl=[];function yo(){for(var e=0;et?t:4,e(!0);var r=jl.transition;jl.transition={};try{e(!1),n()}finally{O=t,jl.transition=r}}function Ca(){return _e().memoizedState}function nd(e,n,t){var r=sn(e);if(t={lane:r,action:t,hasEagerState:!1,eagerState:null,next:null},_a(e))xa(n,t);else if(t=na(e,n,t,r),t!==null){var l=ue();Re(t,e,r,l),Pa(t,n,r)}}function td(e,n,t){var r=sn(e),l={lane:r,action:t,hasEagerState:!1,eagerState:null,next:null};if(_a(e))xa(n,l);else{var u=e.alternate;if(e.lanes===0&&(u===null||u.lanes===0)&&(u=n.lastRenderedReducer,u!==null))try{var o=n.lastRenderedState,i=u(o,t);if(l.hasEagerState=!0,l.eagerState=i,Oe(i,o)){var s=n.interleaved;s===null?(l.next=l,po(n)):(l.next=s.next,s.next=l),n.interleaved=l;return}}catch{}finally{}t=na(e,n,l,r),t!==null&&(l=ue(),Re(t,e,r,l),Pa(t,n,r))}}function _a(e){var n=e.alternate;return e===$||n!==null&&n===$}function xa(e,n){xt=Kr=!0;var t=e.pending;t===null?n.next=n:(n.next=t.next,t.next=n),e.pending=n}function Pa(e,n,t){if(t&4194240){var r=n.lanes;r&=e.pendingLanes,t|=r,n.lanes=t,qu(e,t)}}var Yr={readContext:Ce,useCallback:ee,useContext:ee,useEffect:ee,useImperativeHandle:ee,useInsertionEffect:ee,useLayoutEffect:ee,useMemo:ee,useReducer:ee,useRef:ee,useState:ee,useDebugValue:ee,useDeferredValue:ee,useTransition:ee,useMutableSource:ee,useSyncExternalStore:ee,useId:ee,unstable_isNewReconciler:!1},rd={readContext:Ce,useCallback:function(e,n){return De().memoizedState=[e,n===void 0?null:n],e},useContext:Ce,useEffect:Pi,useImperativeHandle:function(e,n,t){return t=t!=null?t.concat([e]):null,_r(4194308,4,ga.bind(null,n,e),t)},useLayoutEffect:function(e,n){return _r(4194308,4,e,n)},useInsertionEffect:function(e,n){return _r(4,2,e,n)},useMemo:function(e,n){var t=De();return n=n===void 0?null:n,e=e(),t.memoizedState=[e,n],e},useReducer:function(e,n,t){var r=De();return n=t!==void 0?t(n):n,r.memoizedState=r.baseState=n,e={pending:null,interleaved:null,lanes:0,dispatch:null,lastRenderedReducer:e,lastRenderedState:n},r.queue=e,e=e.dispatch=nd.bind(null,$,e),[r.memoizedState,e]},useRef:function(e){var n=De();return e={current:e},n.memoizedState=e},useState:xi,useDebugValue:Eo,useDeferredValue:function(e){return De().memoizedState=e},useTransition:function(){var e=xi(!1),n=e[0];return e=ed.bind(null,e[1]),De().memoizedState=e,[n,e]},useMutableSource:function(){},useSyncExternalStore:function(e,n,t){var r=$,l=De();if(j){if(t===void 0)throw Error(y(407));t=t()}else{if(t=n(),Z===null)throw Error(y(349));Pn&30||ca(r,n,t)}l.memoizedState=t;var u={value:t,getSnapshot:n};return l.queue=u,Pi(da.bind(null,r,u,e),[e]),r.flags|=2048,Qt(9,fa.bind(null,r,u,t,n),void 0,null),t},useId:function(){var e=De(),n=Z.identifierPrefix;if(j){var t=Ae,r=Ve;t=(r&~(1<<32-Te(r)-1)).toString(32)+t,n=":"+n+"R"+t,t=Ht++,0<\/script>",e=e.removeChild(e.firstChild)):typeof r.is=="string"?e=o.createElement(t,{is:r.is}):(e=o.createElement(t),t==="select"&&(o=e,r.multiple?o.multiple=!0:r.size&&(o.size=r.size))):e=o.createElementNS(e,t),e[Ie]=n,e[Vt]=r,Ia(e,n,!1,!1),n.stateNode=e;e:{switch(o=uu(t,r),t){case"dialog":D("cancel",e),D("close",e),l=r;break;case"iframe":case"object":case"embed":D("load",e),l=r;break;case"video":case"audio":for(l=0;ltt&&(n.flags|=128,r=!0,mt(u,!1),n.lanes=4194304)}else{if(!r)if(e=Qr(o),e!==null){if(n.flags|=128,r=!0,t=e.updateQueue,t!==null&&(n.updateQueue=t,n.flags|=4),mt(u,!0),u.tail===null&&u.tailMode==="hidden"&&!o.alternate&&!j)return ne(n),null}else 2*W()-u.renderingStartTime>tt&&t!==1073741824&&(n.flags|=128,r=!0,mt(u,!1),n.lanes=4194304);u.isBackwards?(o.sibling=n.child,n.child=o):(t=u.last,t!==null?t.sibling=o:n.child=o,u.last=o)}return u.tail!==null?(n=u.tail,u.rendering=n,u.tail=n.sibling,u.renderingStartTime=W(),n.sibling=null,t=U.current,M(U,r?t&1|2:t&1),n):(ne(n),null);case 22:case 23:return zo(),r=n.memoizedState!==null,e!==null&&e.memoizedState!==null!==r&&(n.flags|=8192),r&&n.mode&1?pe&1073741824&&(ne(n),n.subtreeFlags&6&&(n.flags|=8192)):ne(n),null;case 24:return null;case 25:return null}throw Error(y(156,n.tag))}function fd(e,n){switch(io(n),n.tag){case 1:return fe(n.type)&&Ur(),e=n.flags,e&65536?(n.flags=e&-65537|128,n):null;case 3:return et(),I(ce),I(re),yo(),e=n.flags,e&65536&&!(e&128)?(n.flags=e&-65537|128,n):null;case 5:return ho(n),null;case 13:if(I(U),e=n.memoizedState,e!==null&&e.dehydrated!==null){if(n.alternate===null)throw Error(y(340));qn()}return e=n.flags,e&65536?(n.flags=e&-65537|128,n):null;case 19:return I(U),null;case 4:return et(),null;case 10:return fo(n.type._context),null;case 22:case 23:return zo(),null;case 24:return null;default:return null}}var mr=!1,te=!1,dd=typeof WeakSet=="function"?WeakSet:Set,S=null;function Bn(e,n){var t=e.ref;if(t!==null)if(typeof t=="function")try{t(null)}catch(r){A(e,n,r)}else t.current=null}function Ou(e,n,t){try{t()}catch(r){A(e,n,r)}}var Ii=!1;function pd(e,n){if(vu=Dr,e=As(),uo(e)){if("selectionStart"in e)var t={start:e.selectionStart,end:e.selectionEnd};else e:{t=(t=e.ownerDocument)&&t.defaultView||window;var r=t.getSelection&&t.getSelection();if(r&&r.rangeCount!==0){t=r.anchorNode;var l=r.anchorOffset,u=r.focusNode;r=r.focusOffset;try{t.nodeType,u.nodeType}catch{t=null;break e}var o=0,i=-1,s=-1,c=0,v=0,m=e,p=null;n:for(;;){for(var g;m!==t||l!==0&&m.nodeType!==3||(i=o+l),m!==u||r!==0&&m.nodeType!==3||(s=o+r),m.nodeType===3&&(o+=m.nodeValue.length),(g=m.firstChild)!==null;)p=m,m=g;for(;;){if(m===e)break n;if(p===t&&++c===l&&(i=o),p===u&&++v===r&&(s=o),(g=m.nextSibling)!==null)break;m=p,p=m.parentNode}m=g}t=i===-1||s===-1?null:{start:i,end:s}}else t=null}t=t||{start:0,end:0}}else t=null;for(hu={focusedElem:e,selectionRange:t},Dr=!1,S=n;S!==null;)if(n=S,e=n.child,(n.subtreeFlags&1028)!==0&&e!==null)e.return=n,S=e;else for(;S!==null;){n=S;try{var w=n.alternate;if(n.flags&1024)switch(n.tag){case 0:case 11:case 15:break;case 1:if(w!==null){var k=w.memoizedProps,F=w.memoizedState,f=n.stateNode,a=f.getSnapshotBeforeUpdate(n.elementType===n.type?k:Ne(n.type,k),F);f.__reactInternalSnapshotBeforeUpdate=a}break;case 3:var d=n.stateNode.containerInfo;d.nodeType===1?d.textContent="":d.nodeType===9&&d.documentElement&&d.removeChild(d.documentElement);break;case 5:case 6:case 4:case 17:break;default:throw Error(y(163))}}catch(h){A(n,n.return,h)}if(e=n.sibling,e!==null){e.return=n.return,S=e;break}S=n.return}return w=Ii,Ii=!1,w}function Pt(e,n,t){var r=n.updateQueue;if(r=r!==null?r.lastEffect:null,r!==null){var l=r=r.next;do{if((l.tag&e)===e){var u=l.destroy;l.destroy=void 0,u!==void 0&&Ou(n,t,u)}l=l.next}while(l!==r)}}function sl(e,n){if(n=n.updateQueue,n=n!==null?n.lastEffect:null,n!==null){var t=n=n.next;do{if((t.tag&e)===e){var r=t.create;t.destroy=r()}t=t.next}while(t!==n)}}function Mu(e){var n=e.ref;if(n!==null){var t=e.stateNode;switch(e.tag){case 5:e=t;break;default:e=t}typeof n=="function"?n(e):n.current=e}}function Ua(e){var n=e.alternate;n!==null&&(e.alternate=null,Ua(n)),e.child=null,e.deletions=null,e.sibling=null,e.tag===5&&(n=e.stateNode,n!==null&&(delete n[Ie],delete n[Vt],delete n[wu],delete n[Gf],delete n[Zf])),e.stateNode=null,e.return=null,e.dependencies=null,e.memoizedProps=null,e.memoizedState=null,e.pendingProps=null,e.stateNode=null,e.updateQueue=null}function $a(e){return e.tag===5||e.tag===3||e.tag===4}function Fi(e){e:for(;;){for(;e.sibling===null;){if(e.return===null||$a(e.return))return null;e=e.return}for(e.sibling.return=e.return,e=e.sibling;e.tag!==5&&e.tag!==6&&e.tag!==18;){if(e.flags&2||e.child===null||e.tag===4)continue e;e.child.return=e,e=e.child}if(!(e.flags&2))return e.stateNode}}function Du(e,n,t){var r=e.tag;if(r===5||r===6)e=e.stateNode,n?t.nodeType===8?t.parentNode.insertBefore(e,n):t.insertBefore(e,n):(t.nodeType===8?(n=t.parentNode,n.insertBefore(e,t)):(n=t,n.appendChild(e)),t=t._reactRootContainer,t!=null||n.onclick!==null||(n.onclick=jr));else if(r!==4&&(e=e.child,e!==null))for(Du(e,n,t),e=e.sibling;e!==null;)Du(e,n,t),e=e.sibling}function Iu(e,n,t){var r=e.tag;if(r===5||r===6)e=e.stateNode,n?t.insertBefore(e,n):t.appendChild(e);else if(r!==4&&(e=e.child,e!==null))for(Iu(e,n,t),e=e.sibling;e!==null;)Iu(e,n,t),e=e.sibling}var J=null,ze=!1;function Xe(e,n,t){for(t=t.child;t!==null;)Va(e,n,t),t=t.sibling}function Va(e,n,t){if(Fe&&typeof Fe.onCommitFiberUnmount=="function")try{Fe.onCommitFiberUnmount(el,t)}catch{}switch(t.tag){case 5:te||Bn(t,n);case 6:var r=J,l=ze;J=null,Xe(e,n,t),J=r,ze=l,J!==null&&(ze?(e=J,t=t.stateNode,e.nodeType===8?e.parentNode.removeChild(t):e.removeChild(t)):J.removeChild(t.stateNode));break;case 18:J!==null&&(ze?(e=J,t=t.stateNode,e.nodeType===8?Dl(e.parentNode,t):e.nodeType===1&&Dl(e,t),It(e)):Dl(J,t.stateNode));break;case 4:r=J,l=ze,J=t.stateNode.containerInfo,ze=!0,Xe(e,n,t),J=r,ze=l;break;case 0:case 11:case 14:case 15:if(!te&&(r=t.updateQueue,r!==null&&(r=r.lastEffect,r!==null))){l=r=r.next;do{var u=l,o=u.destroy;u=u.tag,o!==void 0&&(u&2||u&4)&&Ou(t,n,o),l=l.next}while(l!==r)}Xe(e,n,t);break;case 1:if(!te&&(Bn(t,n),r=t.stateNode,typeof r.componentWillUnmount=="function"))try{r.props=t.memoizedProps,r.state=t.memoizedState,r.componentWillUnmount()}catch(i){A(t,n,i)}Xe(e,n,t);break;case 21:Xe(e,n,t);break;case 22:t.mode&1?(te=(r=te)||t.memoizedState!==null,Xe(e,n,t),te=r):Xe(e,n,t);break;default:Xe(e,n,t)}}function ji(e){var n=e.updateQueue;if(n!==null){e.updateQueue=null;var t=e.stateNode;t===null&&(t=e.stateNode=new dd),n.forEach(function(r){var l=Ed.bind(null,e,r);t.has(r)||(t.add(r),r.then(l,l))})}}function Pe(e,n){var t=n.deletions;if(t!==null)for(var r=0;rl&&(l=o),r&=~u}if(r=l,r=W()-r,r=(120>r?120:480>r?480:1080>r?1080:1920>r?1920:3e3>r?3e3:4320>r?4320:1960*vd(r/1960))-r,10e?16:e,en===null)var r=!1;else{if(e=en,en=null,Zr=0,R&6)throw Error(y(331));var l=R;for(R|=4,S=e.current;S!==null;){var u=S,o=u.child;if(S.flags&16){var i=u.deletions;if(i!==null){for(var s=0;sW()-Po?En(e,0):xo|=t),de(e,n)}function Xa(e,n){n===0&&(e.mode&1?(n=ur,ur<<=1,!(ur&130023424)&&(ur=4194304)):n=1);var t=ue();e=Qe(e,n),e!==null&&(Xt(e,n,t),de(e,t))}function Sd(e){var n=e.memoizedState,t=0;n!==null&&(t=n.retryLane),Xa(e,t)}function Ed(e,n){var t=0;switch(e.tag){case 13:var r=e.stateNode,l=e.memoizedState;l!==null&&(t=l.retryLane);break;case 19:r=e.stateNode;break;default:throw Error(y(314))}r!==null&&r.delete(n),Xa(e,t)}var Ga;Ga=function(e,n,t){if(e!==null)if(e.memoizedProps!==n.pendingProps||ce.current)ae=!0;else{if(!(e.lanes&t)&&!(n.flags&128))return ae=!1,ad(e,n,t);ae=!!(e.flags&131072)}else ae=!1,j&&n.flags&1048576&&qs(n,Ar,n.index);switch(n.lanes=0,n.tag){case 2:var r=n.type;xr(e,n),e=n.pendingProps;var l=Jn(n,re.current);Xn(n,t),l=wo(null,n,r,e,l,t);var u=ko();return n.flags|=1,typeof l=="object"&&l!==null&&typeof l.render=="function"&&l.$$typeof===void 0?(n.tag=1,n.memoizedState=null,n.updateQueue=null,fe(r)?(u=!0,$r(n)):u=!1,n.memoizedState=l.state!==null&&l.state!==void 0?l.state:null,mo(n),l.updater=ol,n.stateNode=l,l._reactInternals=n,xu(n,r,e,t),n=zu(null,n,r,!0,u,t)):(n.tag=0,j&&u&&oo(n),le(null,n,l,t),n=n.child),n;case 16:r=n.elementType;e:{switch(xr(e,n),e=n.pendingProps,l=r._init,r=l(r._payload),n.type=r,l=n.tag=_d(r),e=Ne(r,e),l){case 0:n=Nu(null,n,r,e,t);break e;case 1:n=Oi(null,n,r,e,t);break e;case 11:n=Ti(null,n,r,e,t);break e;case 14:n=Ri(null,n,r,Ne(r.type,e),t);break e}throw Error(y(306,r,""))}return n;case 0:return r=n.type,l=n.pendingProps,l=n.elementType===r?l:Ne(r,l),Nu(e,n,r,l,t);case 1:return r=n.type,l=n.pendingProps,l=n.elementType===r?l:Ne(r,l),Oi(e,n,r,l,t);case 3:e:{if(Oa(n),e===null)throw Error(y(387));r=n.pendingProps,u=n.memoizedState,l=u.element,ta(e,n),Wr(n,r,null,t);var o=n.memoizedState;if(r=o.element,u.isDehydrated)if(u={element:r,isDehydrated:!1,cache:o.cache,pendingSuspenseBoundaries:o.pendingSuspenseBoundaries,transitions:o.transitions},n.updateQueue.baseState=u,n.memoizedState=u,n.flags&256){l=nt(Error(y(423)),n),n=Mi(e,n,r,t,l);break e}else if(r!==l){l=nt(Error(y(424)),n),n=Mi(e,n,r,t,l);break e}else for(me=ln(n.stateNode.containerInfo.firstChild),ve=n,j=!0,Le=null,t=oa(n,null,r,t),n.child=t;t;)t.flags=t.flags&-3|4096,t=t.sibling;else{if(qn(),r===l){n=Ke(e,n,t);break e}le(e,n,r,t)}n=n.child}return n;case 5:return ia(n),e===null&&Eu(n),r=n.type,l=n.pendingProps,u=e!==null?e.memoizedProps:null,o=l.children,yu(r,l)?o=null:u!==null&&yu(r,u)&&(n.flags|=32),Ra(e,n),le(e,n,o,t),n.child;case 6:return e===null&&Eu(n),null;case 13:return Ma(e,n,t);case 4:return vo(n,n.stateNode.containerInfo),r=n.pendingProps,e===null?n.child=bn(n,null,r,t):le(e,n,r,t),n.child;case 11:return r=n.type,l=n.pendingProps,l=n.elementType===r?l:Ne(r,l),Ti(e,n,r,l,t);case 7:return le(e,n,n.pendingProps,t),n.child;case 8:return le(e,n,n.pendingProps.children,t),n.child;case 12:return le(e,n,n.pendingProps.children,t),n.child;case 10:e:{if(r=n.type._context,l=n.pendingProps,u=n.memoizedProps,o=l.value,M(Br,r._currentValue),r._currentValue=o,u!==null)if(Oe(u.value,o)){if(u.children===l.children&&!ce.current){n=Ke(e,n,t);break e}}else for(u=n.child,u!==null&&(u.return=n);u!==null;){var i=u.dependencies;if(i!==null){o=u.child;for(var s=i.firstContext;s!==null;){if(s.context===r){if(u.tag===1){s=Be(-1,t&-t),s.tag=2;var c=u.updateQueue;if(c!==null){c=c.shared;var v=c.pending;v===null?s.next=s:(s.next=v.next,v.next=s),c.pending=s}}u.lanes|=t,s=u.alternate,s!==null&&(s.lanes|=t),Cu(u.return,t,n),i.lanes|=t;break}s=s.next}}else if(u.tag===10)o=u.type===n.type?null:u.child;else if(u.tag===18){if(o=u.return,o===null)throw Error(y(341));o.lanes|=t,i=o.alternate,i!==null&&(i.lanes|=t),Cu(o,t,n),o=u.sibling}else o=u.child;if(o!==null)o.return=u;else for(o=u;o!==null;){if(o===n){o=null;break}if(u=o.sibling,u!==null){u.return=o.return,o=u;break}o=o.return}u=o}le(e,n,l.children,t),n=n.child}return n;case 9:return l=n.type,r=n.pendingProps.children,Xn(n,t),l=Ce(l),r=r(l),n.flags|=1,le(e,n,r,t),n.child;case 14:return r=n.type,l=Ne(r,n.pendingProps),l=Ne(r.type,l),Ri(e,n,r,l,t);case 15:return La(e,n,n.type,n.pendingProps,t);case 17:return r=n.type,l=n.pendingProps,l=n.elementType===r?l:Ne(r,l),xr(e,n),n.tag=1,fe(r)?(e=!0,$r(n)):e=!1,Xn(n,t),la(n,r,l),xu(n,r,l,t),zu(null,n,r,!0,e,t);case 19:return Da(e,n,t);case 22:return Ta(e,n,t)}throw Error(y(156,n.tag))};function Za(e,n){return Es(e,n)}function Cd(e,n,t,r){this.tag=e,this.key=t,this.sibling=this.child=this.return=this.stateNode=this.type=this.elementType=null,this.index=0,this.ref=null,this.pendingProps=n,this.dependencies=this.memoizedState=this.updateQueue=this.memoizedProps=null,this.mode=r,this.subtreeFlags=this.flags=0,this.deletions=null,this.childLanes=this.lanes=0,this.alternate=null}function Se(e,n,t,r){return new Cd(e,n,t,r)}function To(e){return e=e.prototype,!(!e||!e.isReactComponent)}function _d(e){if(typeof e=="function")return To(e)?1:0;if(e!=null){if(e=e.$$typeof,e===Xu)return 11;if(e===Gu)return 14}return 2}function an(e,n){var t=e.alternate;return t===null?(t=Se(e.tag,n,e.key,e.mode),t.elementType=e.elementType,t.type=e.type,t.stateNode=e.stateNode,t.alternate=e,e.alternate=t):(t.pendingProps=n,t.type=e.type,t.flags=0,t.subtreeFlags=0,t.deletions=null),t.flags=e.flags&14680064,t.childLanes=e.childLanes,t.lanes=e.lanes,t.child=e.child,t.memoizedProps=e.memoizedProps,t.memoizedState=e.memoizedState,t.updateQueue=e.updateQueue,n=e.dependencies,t.dependencies=n===null?null:{lanes:n.lanes,firstContext:n.firstContext},t.sibling=e.sibling,t.index=e.index,t.ref=e.ref,t}function zr(e,n,t,r,l,u){var o=2;if(r=e,typeof e=="function")To(e)&&(o=1);else if(typeof e=="string")o=5;else e:switch(e){case Mn:return Cn(t.children,l,u,n);case Yu:o=8,l|=8;break;case Gl:return e=Se(12,t,n,l|2),e.elementType=Gl,e.lanes=u,e;case Zl:return e=Se(13,t,n,l),e.elementType=Zl,e.lanes=u,e;case Jl:return e=Se(19,t,n,l),e.elementType=Jl,e.lanes=u,e;case us:return cl(t,l,u,n);default:if(typeof e=="object"&&e!==null)switch(e.$$typeof){case rs:o=10;break e;case ls:o=9;break e;case Xu:o=11;break e;case Gu:o=14;break e;case Ge:o=16,r=null;break e}throw Error(y(130,e==null?e:typeof e,""))}return n=Se(o,t,n,l),n.elementType=e,n.type=r,n.lanes=u,n}function Cn(e,n,t,r){return e=Se(7,e,r,n),e.lanes=t,e}function cl(e,n,t,r){return e=Se(22,e,r,n),e.elementType=us,e.lanes=t,e.stateNode={isHidden:!1},e}function Bl(e,n,t){return e=Se(6,e,null,n),e.lanes=t,e}function Hl(e,n,t){return n=Se(4,e.children!==null?e.children:[],e.key,n),n.lanes=t,n.stateNode={containerInfo:e.containerInfo,pendingChildren:null,implementation:e.implementation},n}function xd(e,n,t,r,l){this.tag=n,this.containerInfo=e,this.finishedWork=this.pingCache=this.current=this.pendingChildren=null,this.timeoutHandle=-1,this.callbackNode=this.pendingContext=this.context=null,this.callbackPriority=0,this.eventTimes=Cl(0),this.expirationTimes=Cl(-1),this.entangledLanes=this.finishedLanes=this.mutableReadLanes=this.expiredLanes=this.pingedLanes=this.suspendedLanes=this.pendingLanes=0,this.entanglements=Cl(0),this.identifierPrefix=r,this.onRecoverableError=l,this.mutableSourceEagerHydrationData=null}function Ro(e,n,t,r,l,u,o,i,s){return e=new xd(e,n,t,i,s),n===1?(n=1,u===!0&&(n|=8)):n=0,u=Se(3,null,null,n),e.current=u,u.stateNode=e,u.memoizedState={element:r,isDehydrated:t,cache:null,transitions:null,pendingSuspenseBoundaries:null},mo(u),e}function Pd(e,n,t){var r=3"u"||typeof __REACT_DEVTOOLS_GLOBAL_HOOK__.checkDCE!="function"))try{__REACT_DEVTOOLS_GLOBAL_HOOK__.checkDCE(n)}catch(t){console.error(t)}}n(),e.exports=ye})(Pc);var Qi=Kl;Ql.createRoot=Qi.createRoot,Ql.hydrateRoot=Qi.hydrateRoot;Ql.createRoot(document.getElementById("root")).render($o(wc.StrictMode,{children:$o("div",{children:"Hello"})})); diff --git a/client/dist/index.html b/client/dist/index.html new file mode 100644 index 0000000..7cc03d8 --- /dev/null +++ b/client/dist/index.html @@ -0,0 +1,14 @@ + + + + + + + Vite App + + + +
+ + + diff --git a/client/index.html b/client/index.html new file mode 100644 index 0000000..38f3861 --- /dev/null +++ b/client/index.html @@ -0,0 +1,13 @@ + + + + + + + Vite App + + +
+ + + diff --git a/client/src/main.css b/client/src/main.css new file mode 100644 index 0000000..e69de29 diff --git a/client/src/main.tsx b/client/src/main.tsx new file mode 100644 index 0000000..60801e6 --- /dev/null +++ b/client/src/main.tsx @@ -0,0 +1,9 @@ +import React from 'react' +import ReactDOM from 'react-dom/client' +import './main.css' + +ReactDOM.createRoot(document.getElementById('root')!).render( + +
Hello
+
+) diff --git a/generate-certificate.sh b/generate-certificate.sh new file mode 100644 index 0000000..ac21cc1 --- /dev/null +++ b/generate-certificate.sh @@ -0,0 +1,11 @@ +#!/bin/ash +FILE_CERT_NAME=localcert +if [ -f "certificates/$FILE_CERT_NAME.crt" ] && [ -f "$FILE_CERT_NAME.key" ] ; then + echo "Cert and Key already exist" +else + echo "Cert and Key does not exist, trying to create new ones..." + apk update && apk add openssl && rm -rf /var/cache/apk/* + openssl req -new -subj "/C=US/ST=California/CN=localhost" \ + -newkey rsa:2048 -nodes -keyout "$FILE_CERT_NAME.key" -out "$FILE_CERT_NAME.csr" + openssl x509 -req -days 365 -in "$FILE_CERT_NAME.csr" -signkey "$FILE_CERT_NAME.key" -out "$FILE_CERT_NAME.crt" -extfile "self-signed-cert.ext" +fi \ No newline at end of file diff --git a/gupm.json b/gupm.json new file mode 100644 index 0000000..5ffe4cd --- /dev/null +++ b/gupm.json @@ -0,0 +1,32 @@ +{ + "author": "Yann Stepienik", + "cli": { + "aliases": { + "start": "build/bin", + "certificate": "sh generate-certificate.sh" + } + }, + "dependencies": { + "default": { + "go://github.com/go-playground/validator/v10": "master", + "go://github.com/gorilla/mux": "master", + "go://github.com/imdario/mergo": "master", + "go://github.com/joho/godotenv": "master", + "go://github.com/lib/pq": "master", + "go://github.com/pquerna/ffjson": "master", + "go://go.mongodb.org/mongo-driver": "master", + "go://gopkg.in/ffmt.v1": "v1.5.6", + "npm://@esbuild/linux-x64": "0.16.17", + "npm://@vitejs/plugin-react": "3.1.0", + "npm://react": "18.2.0", + "npm://react-dom": "18.2.0", + "npm://typescript": "4.9.5", + "npm://vite": "4.1.1" + }, + "defaultProvider": "go" + }, + "description": "MyData file server", + "license": "ISC", + "name": "myFileServer", + "wrapInstallFolder": "src" +} \ No newline at end of file diff --git a/readme.md b/readme.md new file mode 100644 index 0000000..826b5d7 --- /dev/null +++ b/readme.md @@ -0,0 +1,30 @@ +# GUCO Server + +blablabla ... + +# Installation + +blablabala ... + +# Build locally + +You need [GuPM](https://github.com/azukaar/GuPM) with the [provider-go](https://github.com/azukaar/GuPM-official#provider-go) plugin to run this project. + +``` +g make +``` + +# Run locally + +First create a file called dev.json with: + +```json +{ + "MONGODB": "your mongodb connection string" +} +``` + +``` +g build +g start +``` \ No newline at end of file diff --git a/self-signed-cert.ext b/self-signed-cert.ext new file mode 100644 index 0000000..cdead92 --- /dev/null +++ b/self-signed-cert.ext @@ -0,0 +1,4 @@ +subjectAltName = @alt_names +[alt_names] +DNS.1 = localhost +DNS.2 = https-server \ No newline at end of file diff --git a/src/file/copy.go b/src/file/copy.go new file mode 100644 index 0000000..39d4907 --- /dev/null +++ b/src/file/copy.go @@ -0,0 +1,59 @@ +package file + +import ( + "log" + "net/http" + "io" + "encoding/json" + "os" + "../utils" +) + +func copyFile(src, dst string) error { + in, err := os.Open(src) + if err != nil { + return err + } + defer in.Close() + + out, err := os.Create(dst) + if err != nil { + return err + } + defer out.Close() + + _, err = io.Copy(out, in) + if err != nil { + return err + } + return out.Close() +} + +func FileCopy(w http.ResponseWriter, req *http.Request) { + utils.SetHeaders(w) + + fullPath := req.URL.Query().Get("path") + if fullPath == "" { + log.Println("No path specified") + } + + filePath := utils.GetRealPath(fullPath) + + fullDestination := req.URL.Query().Get("destination") + if fullDestination == "" { + log.Println("No destination specified") + } + + destination := utils.GetRealPath(fullDestination) + + // copy file to destination + err := copyFile(filePath, destination) + if err != nil { + log.Fatal(err) + } + + // return json object + json.NewEncoder(w).Encode(map[string]interface{}{ + "Status": "OK", + }) +} \ No newline at end of file diff --git a/src/file/delete.go b/src/file/delete.go new file mode 100644 index 0000000..b246e9a --- /dev/null +++ b/src/file/delete.go @@ -0,0 +1,31 @@ +package file + +import ( + "log" + "net/http" + "encoding/json" + "os" + "../utils" +) + +func FileDelete(w http.ResponseWriter, req *http.Request) { + utils.SetHeaders(w) + + fullPath := req.URL.Query().Get("path") + if fullPath == "" { + log.Println("No path specified") + } + + filePath := utils.GetRealPath(fullPath) + + // delete file + err := os.Remove(filePath) + if err != nil { + log.Fatal(err) + } + + // return json object + json.NewEncoder(w).Encode(map[string]interface{}{ + "Status": "OK", + }) +} \ No newline at end of file diff --git a/src/file/get.go b/src/file/get.go new file mode 100644 index 0000000..61bb072 --- /dev/null +++ b/src/file/get.go @@ -0,0 +1,112 @@ +package file + +import ( + "log" + "net/http" + "io" + "strings" + "os" + // "bytes" + // "mime/multipart" + "bufio" + "strconv" + "../utils" +) + +func getExtension(path string) string { + return strings.Split(path, ".")[len(strings.Split(path, ".")) - 1] +} + +func getContentType(path string) string { + switch getExtension(path) { + case "html": + return "text/html" + case "css": + return "text/css" + case "js": + return "application/javascript" + case "png": + return "image/png" + case "jpg": + return "image/jpeg" + case "jpeg": + return "image/jpeg" + case "gif": + return "image/gif" + case "svg": + return "image/svg+xml" + case "mp4": + return "video/mp4" + case "mkv": + return "video/x-matroska" + case "webm": + return "video/webm" + case "mp3": + return "audio/mpeg" + case "wav": + return "audio/wav" + case "ogg": + return "audio/ogg" + default: + return "text/plain" + } +} + +func FileGet(w http.ResponseWriter, req *http.Request) { + utils.SetHeaders(w) + + fullPath := req.URL.Query().Get("path") + if fullPath == "" { + log.Println("No path specified") + } + + filePath := utils.GetRealPath(fullPath) + + file, err := os.Open(filePath) + if err != nil { + log.Fatal(err) + } + defer file.Close() + + /*fileBytes, err := ioutil.ReadAll(file) + if err != nil { + log.Fatal(err) + }*/ + + // set header content type depending on file type + w.Header().Set("Content-Type", getContentType(filePath)) + + // multipart file send + if getExtension(filePath) == "mp4" || getExtension(filePath) == "mkv" || getExtension(filePath) == "webm" { + w.Header().Set("Content-Disposition", "attachment; filename=" + filePath) + } + + // open stat file + stat, err := os.Stat(filePath) + + + buffer := bufio.NewReader(file) + + // set content-length + w.Header().Set("Content-Length", strconv.FormatInt(stat.Size(), 10)) + + //copy buffer to client + io.Copy(w, buffer) + + /* + // get file stats + fileStats, err := os.Stat(filePath) + + // return json object with metadata FileStat and content + json.NewEncoder(w).Encode(map[string]interface{}{ + "Metadata": FileStats{ + Name: fileStats.Name(), + Path: filePath, + Size: fileStats.Size(), + Mode: fileStats.Mode(), + ModTime: fileStats.ModTime(), + IsDir: fileStats.IsDir(), + }, + "Content": string(file), + })*/ +} \ No newline at end of file diff --git a/src/file/list.go b/src/file/list.go new file mode 100644 index 0000000..4af68a2 --- /dev/null +++ b/src/file/list.go @@ -0,0 +1,55 @@ +package file + +import ( + "log" + "net/http" + "io/ioutil" + "os" + "encoding/json" + "../utils" +) + +func FileList(w http.ResponseWriter, req *http.Request) { + utils.SetHeaders(w) + + fullPath := req.URL.Query().Get("path") + if fullPath == "" { + log.Println("No path specified") + } + + filePath := utils.GetRealPath(fullPath) + + files, err := ioutil.ReadDir(filePath) + if err != nil { + log.Fatal(err) + } + + // get folder stats + folderStats, err := os.Stat(filePath) + + // add file FileStats to json object + var fileStats [](utils.FileStats) + for _, file := range files { + fileStats = append(fileStats, utils.FileStats{ + Name: file.Name(), + Path: fullPath + "/" + file.Name(), + Size: file.Size(), + Mode: file.Mode(), + ModTime: file.ModTime(), + IsDir: file.IsDir(), + }) + } + + // return json object + // return json object with metadata FileStat and content + json.NewEncoder(w).Encode(map[string]interface{}{ + "Metadata": utils.FileStats{ + Name: folderStats.Name(), + Size: folderStats.Size(), + Mode: folderStats.Mode(), + ModTime: folderStats.ModTime(), + IsDir: folderStats.IsDir(), + }, + "Content": fileStats, + }) +} \ No newline at end of file diff --git a/src/file/move.go b/src/file/move.go new file mode 100644 index 0000000..47737fc --- /dev/null +++ b/src/file/move.go @@ -0,0 +1,39 @@ +package file + +import ( + "log" + "net/http" + "encoding/json" + "os" + "../utils" +) + +func FileMove(w http.ResponseWriter, req *http.Request) { + utils.SetHeaders(w) + + fullPath := req.URL.Query().Get("path") + if fullPath == "" { + log.Println("No path specified") + } + + filePath := utils.GetRealPath(fullPath) + + fullDestination := req.URL.Query().Get("destination") + if fullDestination == "" { + log.Println("No destination specified") + } + + destination := utils.GetRealPath(fullDestination) + + // copy file to destination + + err := os.Rename(filePath, destination) + if err != nil { + log.Fatal(err) + } + + // return json object + json.NewEncoder(w).Encode(map[string]interface{}{ + "Status": "OK", + }) +} \ No newline at end of file diff --git a/src/index.go b/src/index.go new file mode 100644 index 0000000..c2aec85 --- /dev/null +++ b/src/index.go @@ -0,0 +1,106 @@ +package main + +import ( + "net/http" + "./utils" + "./file" + "./user" + "github.com/gorilla/mux" + "log" + "os" + "strings" +) + +var tlsCert = "localcert.crt" +var tlsKey= "localcert.key" +var serverPortHTTP = os.Getenv("HTTP_PORT") +var serverPortHTTPS = os.Getenv("HTTPS_PORT") + +func startHTTPServer(router *mux.Router) { + log.Println("Listening to HTTP on :" + serverPortHTTP) + + err := http.ListenAndServe("0.0.0.0:" + serverPortHTTP, router) + + if err != nil { + log.Fatal(err) + } +} + +func startHTTPSServer(router *mux.Router) { + // redirect http to https + go (func () { + err := http.ListenAndServe("0.0.0.0:" + serverPortHTTP, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + // change port in host + if strings.HasSuffix(r.Host, ":" + serverPortHTTP) { + if serverPortHTTPS != "443" { + r.Host = r.Host[:len(r.Host)-len(":" + serverPortHTTP)] + ":" + serverPortHTTPS + } else { + r.Host = r.Host[:len(r.Host)-len(":" + serverPortHTTP)] + } + } + + http.Redirect(w, r, "https://"+r.Host+r.URL.String(), http.StatusMovedPermanently) + })) + if err != nil { + log.Fatal(err) + } + })() + + log.Println("Listening to HTTP on :" + serverPortHTTP) + log.Println("Listening to HTTPS on :" + serverPortHTTPS) + + // start https server + err := http.ListenAndServeTLS("0.0.0.0:" + serverPortHTTPS, tlsCert, tlsKey, router) + + if err != nil { + log.Fatal(err) + } +} + +func main() { + log.Println("Starting...") + + if serverPortHTTP == "" { + serverPortHTTP = "80" + } + + if serverPortHTTPS == "" { + serverPortHTTPS = "443" + } + + router := mux.NewRouter().StrictSlash(true) + + utils.DB() + defer utils.Disconnect() + + router.HandleFunc("/health", func(w http.ResponseWriter, r *http.Request) { + w.WriteHeader(http.StatusOK) + w.Write([]byte("OK")) + }) + + router.PathPrefix("/static/").Handler(http.StripPrefix("/static/", http.FileServer(http.Dir("static")))) + + router.HandleFunc("/file/list", file.FileList) + router.HandleFunc("/file/get", file.FileGet) + router.HandleFunc("/file/delete", file.FileDelete) + router.HandleFunc("/file/copy", file.FileCopy) + router.HandleFunc("/file/move", file.FileMove) + + router.HandleFunc("/user/login", user.UserLogin) + // router.HandleFunc("/user/register", user.UserRegister) + // router.HandleFunc("/user/edit", ) + // router.HandleFunc("/user/delete", ) + + // router.HandleFunc("/config/get", ) + // router.HandleFunc("/config/set", ) + + // router.HandleFunc("/db", ) + + if utils.FileExists(tlsCert) && utils.FileExists(tlsKey) { + log.Println("TLS certificate found, starting HTTPS servers and redirecting HTTP to HTTPS") + startHTTPSServer(router) + } else { + log.Println("No TLS certificate found, starting HTTP server only") + startHTTPServer(router) + } +} \ No newline at end of file diff --git a/src/user/login.go b/src/user/login.go new file mode 100644 index 0000000..37f6e55 --- /dev/null +++ b/src/user/login.go @@ -0,0 +1,28 @@ +package user + +import ( + "net/http" + "log" + "encoding/json" + + "../utils" +) + +func UserLogin(w http.ResponseWriter, req *http.Request) { + utils.SetHeaders(w) + + if(req.Method == "POST") { + nickname := req.FormValue("nickname") + password := req.FormValue("password") + + log.Println("UserLogin: nickname: " + nickname) + log.Println("UserLogin: password: " + password) // im just testing ok dont panic + + json.NewEncoder(w).Encode(map[string]interface{}{ + "Status": "OK", + }) + } else { + log.Println("UserLogin: Method not allowed" + req.Method) + http.Error(w, "Method not allowed", http.StatusMethodNotAllowed) + } +} \ No newline at end of file diff --git a/src/user/register.go b/src/user/register.go new file mode 100644 index 0000000..4a4ef63 --- /dev/null +++ b/src/user/register.go @@ -0,0 +1,43 @@ +package user + +import ( + "net/http" + "log" + // "io" + // "os" + "encoding/json" + + "../utils" +) + +func UserRegister(w http.ResponseWriter, req *http.Request) { + utils.SetHeaders(w) + + if(req.Method == "GET") { + // sedn form + } + if(req.Method == "POST") { + // check origin + + // check password strength + + user := &utils.User{ + Nickname: req.FormValue("nickname"), + Password: req.FormValue("password"), + Role: (utils.Role)(req.FormValue("role")), + } + + err := utils.Validate.Struct(user) + + if(err != nil) { + log.Fatal(err) + } + + req.ParseForm() + } + + // return json object + json.NewEncoder(w).Encode(map[string]interface{}{ + "Status": "OK", + }) +} \ No newline at end of file diff --git a/src/utils/config.go b/src/utils/config.go new file mode 100644 index 0000000..4ce2738 --- /dev/null +++ b/src/utils/config.go @@ -0,0 +1,53 @@ +package utils + +import ( + "context" + "log" + "go.mongodb.org/mongo-driver/mongo" + "go.mongodb.org/mongo-driver/bson" + "go.mongodb.org/mongo-driver/mongo/options" + + "github.com/imdario/mergo" +) + +type GucoConfiguration struct { + Test string + Caca string +} + +var defaultConfig = GucoConfiguration{ + Test: "test", + Caca: "prout", +} + +func GetConfigs() GucoConfiguration { + c := GetCollection("GUCO", "Configurations") + config := GucoConfiguration{} + err := c.FindOne(context.TODO(), bson.M{"_id": "GUCO"}).Decode(&config) + if err == mongo.ErrNoDocuments { + log.Println("Record does not exist") + } else if err != nil { + log.Fatal(err) + } + + mergo.Merge(&config, defaultConfig) + + return config; +} + +func SetConfig(config GucoConfiguration) { + currentConfig := GetConfigs() + + mergo.Merge(&config, currentConfig) + + c := GetCollection("GUCO", "Configurations") + + opts := options.Update().SetUpsert(true) + filter := bson.D{{"_id", "GUCO"}} + update := bson.D{{"$set", config}} + + _, err := c.UpdateOne(context.Background(), filter, update, opts) + if err != nil { + log.Fatal(err) + } +} \ No newline at end of file diff --git a/src/utils/db.go b/src/utils/db.go new file mode 100644 index 0000000..ec6a585 --- /dev/null +++ b/src/utils/db.go @@ -0,0 +1,54 @@ +package utils + +import ( + "context" + "log" + "os" + "go.mongodb.org/mongo-driver/mongo" + "go.mongodb.org/mongo-driver/mongo/options" + "go.mongodb.org/mongo-driver/mongo/readpref" +) + + +var client *mongo.Client + +func DB() { + log.Println("Connecting to the database...") + + uri := os.Getenv("MONGODB") + "/?retryWrites=true&w=majority" + + var err error + + client, err = mongo.Connect(context.TODO(), options.Client().ApplyURI(uri)) + if err != nil { + log.Fatal(err) + } + defer func() { + }() + + // Ping the primary + if err := client.Ping(context.TODO(), readpref.Primary()); err != nil { + log.Fatal(err) + } + + log.Println("Successfully connected to the database.") +} + +func Disconnect() { + if err := client.Disconnect(context.TODO()); err != nil { + log.Fatal(err) + } +} + +func GetCollection(applicationId string, collection string) *mongo.Collection { + name := os.Getenv("MONGODB_NAME"); if name == "" { + name = "GUCO" + } + log.Println("Getting collection " + applicationId + "_" + collection + " from database " + name) + c := client.Database(name).Collection(applicationId + "_" + collection) + return c +} + +// func query(q string) (*sql.Rows, error) { +// return db.Query(q) +// } \ No newline at end of file diff --git a/src/utils/types.go b/src/utils/types.go new file mode 100644 index 0000000..6f41b19 --- /dev/null +++ b/src/utils/types.go @@ -0,0 +1,29 @@ +package utils + +import ( + "os" + "time" +) + +type Role string + +const ( + GUEST string = "GUEST" + USER = "USER" + ADMIN = "ADMIN" +) + +type FileStats struct { + Name string + Path string + Size int64 + Mode os.FileMode + ModTime time.Time + IsDir bool +} + +type User struct { + Nickname string `validate:"required"` + Password string `validate:"required"` + Role Role `validate:"required"` +} \ No newline at end of file diff --git a/src/utils/utils.go b/src/utils/utils.go new file mode 100644 index 0000000..e82dbf5 --- /dev/null +++ b/src/utils/utils.go @@ -0,0 +1,38 @@ +package utils + +import ( + "strings" + "log" + "os" + "net/http" +) + +func GetRealPath(fullPath string) string { + var dataPathsObject = map[string]string{ + "data": "/mnt/d/", + "diskE": "/mnt/e/", + } + + path := dataPathsObject[strings.Split(fullPath, "/")[0]] + if path == "" { + log.Println("No path specified") + } + + return path + strings.Join(strings.Split(fullPath, "/")[1:], "/") +} + +func SetHeaders(w http.ResponseWriter) { + w.Header().Set("Content-Type", "application/json") + w.Header().Set("Access-Control-Allow-Origin", "*") + w.Header().Set("Access-Control-Allow-Methods", "POST, GET, OPTIONS, PUT, DELETE") + w.Header().Set("Access-Control-Allow-Headers", "Accept, Content-Type, Content-Length, Accept-Encoding, X-CSRF-Token, Authorization") +} + +func FileExists(path string) bool { + _, err := os.Stat(path) + if err == nil { + return true + } + log.Println(err) + return false +} \ No newline at end of file diff --git a/src/utils/validator.go b/src/utils/validator.go new file mode 100644 index 0000000..7547223 --- /dev/null +++ b/src/utils/validator.go @@ -0,0 +1,5 @@ +package utils + +import "github.com/go-playground/validator/v10" + +var Validate = validator.New() \ No newline at end of file diff --git a/vite.config.js b/vite.config.js new file mode 100644 index 0000000..744bb58 --- /dev/null +++ b/vite.config.js @@ -0,0 +1,9 @@ +import { defineConfig } from 'vite' +import react from '@vitejs/plugin-react' + +// https://vitejs.dev/config/ +export default defineConfig({ + plugins: [react()], + root: 'client', + outDir: 'static', +})