diff --git a/404.html b/404.html index e4664c4..142683c 100644 --- a/404.html +++ b/404.html @@ -9,13 +9,13 @@ - - + +
Skip to main content

Page Not Found

We could not find what you were looking for.

Please contact the owner of the site that linked you to the original URL and let them know their link is broken.

- - + + \ No newline at end of file diff --git a/assets/images/floating_point_representation-efc73a8c1b722a82b772c2cd7ee7ba99.webp b/assets/images/floating_point_representation-efc73a8c1b722a82b772c2cd7ee7ba99.webp new file mode 100644 index 0000000..a74faa8 Binary files /dev/null and b/assets/images/floating_point_representation-efc73a8c1b722a82b772c2cd7ee7ba99.webp differ diff --git a/assets/js/00b62936.719c05c2.js b/assets/js/00b62936.719c05c2.js new file mode 100644 index 0000000..d23283e --- /dev/null +++ b/assets/js/00b62936.719c05c2.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkfoomo=self.webpackChunkfoomo||[]).push([[6284],{2620:a=>{a.exports=JSON.parse('{"label":"golang","permalink":"/blog/tags/golang","allTagsPath":"/blog/tags","count":1}')}}]); \ No newline at end of file diff --git a/assets/js/23b2474b.b0be182d.js b/assets/js/23b2474b.b0be182d.js new file mode 100644 index 0000000..a0cbb21 --- /dev/null +++ b/assets/js/23b2474b.b0be182d.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkfoomo=self.webpackChunkfoomo||[]).push([[1364],{3905:(e,t,n)=>{n.d(t,{Zo:()=>s,kt:()=>f});var o=n(7294);function r(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function a(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var o=Object.getOwnPropertySymbols(e);t&&(o=o.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,o)}return n}function i(e){for(var t=1;t=0||(r[n]=e[n]);return r}(e,t);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);for(o=0;o=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(r[n]=e[n])}return r}var l=o.createContext({}),u=function(e){var t=o.useContext(l),n=t;return e&&(n="function"==typeof e?e(t):i(i({},t),e)),n},s=function(e){var t=u(e.components);return o.createElement(l.Provider,{value:t},e.children)},p="mdxType",m={inlineCode:"code",wrapper:function(e){var t=e.children;return o.createElement(o.Fragment,{},t)}},d=o.forwardRef((function(e,t){var n=e.components,r=e.mdxType,a=e.originalType,l=e.parentName,s=c(e,["components","mdxType","originalType","parentName"]),p=u(n),d=r,f=p["".concat(l,".").concat(d)]||p[d]||m[d]||a;return n?o.createElement(f,i(i({ref:t},s),{},{components:n})):o.createElement(f,i({ref:t},s))}));function f(e,t){var n=arguments,r=t&&t.mdxType;if("string"==typeof e||r){var a=n.length,i=new Array(a);i[0]=d;var c={};for(var l in t)hasOwnProperty.call(t,l)&&(c[l]=t[l]);c.originalType=e,c[p]="string"==typeof e?e:r,i[1]=c;for(var u=2;u{n.r(t),n.d(t,{assets:()=>l,contentTitle:()=>i,default:()=>m,frontMatter:()=>a,metadata:()=>c,toc:()=>u});var o=n(7462),r=(n(7294),n(3905));const a={slug:"accuracy-of-decimal-computations",authors:["patrickb"],tags:["golang","currency","decimal accuracy"]},i="Accuracy of decimal computations",c={permalink:"/blog/accuracy-of-decimal-computations",editUrl:"https://github.com/foomo/foomo-docs/tree/main/foomo/blog/2023-03-06-accuracy-of-decimal-computations/index.mdx",source:"@site/blog/2023-03-06-accuracy-of-decimal-computations/index.mdx",title:"Accuracy of decimal computations",description:"Intro",date:"2023-03-06T00:00:00.000Z",formattedDate:"March 6, 2023",tags:[{label:"golang",permalink:"/blog/tags/golang"},{label:"currency",permalink:"/blog/tags/currency"},{label:"decimal accuracy",permalink:"/blog/tags/decimal-accuracy"}],hasTruncateMarker:!1,authors:[{name:"Patrick Buchner",title:"MSc",url:"https://github.com/smartinov",imageURL:"https://github.com/uebriges.png",key:"patrickb"}],frontMatter:{slug:"accuracy-of-decimal-computations",authors:["patrickb"],tags:["golang","currency","decimal accuracy"]},nextItem:{title:"Why bundle size is important?",permalink:"/blog/why-bundle-size-is-important"}},l={authorsImageUrls:[void 0]},u=[{value:"Intro",id:"intro",level:2},{value:"Finite accuracy of representation",id:"finite-accuracy-of-representation",level:2},{value:"Money computations",id:"money-computations",level:2},{value:"Solution",id:"solution",level:2},{value:"Conclusio",id:"conclusio",level:2}],s={toc:u},p="wrapper";function m(e){let{components:t,...a}=e;return(0,r.kt)(p,(0,o.Z)({},s,a,{components:t,mdxType:"MDXLayout"}),(0,r.kt)("h2",{id:"intro"},"Intro"),(0,r.kt)("p",null,"Calculating with money can be tricky if not taken proper precautions. Some might be tempted to use float representation for calculating with currency values. That is problematic because of possible rounding errors."),(0,r.kt)("h2",{id:"finite-accuracy-of-representation"},"Finite accuracy of representation"),(0,r.kt)("p",null,"Floating points are represented like this"),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Floating point representation",src:n(880).Z,width:"810",height:"226"})),(0,r.kt)("p",null,"Not every number can be represented with a finite number of decimal places"),(0,r.kt)("p",null,"0.01 \u2014> 0.0000011001100110011\u2026"),(0,r.kt)("p",null,"Taking 17 places of the above results in 0.010000000000000001"),(0,r.kt)("p",null,"Consider the following code snipet that shows the missing accuracy"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-go"},"func main() {\n\n var n float64 = 0\n\n for i := 0; i < 1000; i++ {\n n += .01\n }\n\n fmt.Println(n)\n\n}\n")),(0,r.kt)("p",null,"Result: 9.999999999999831"),(0,r.kt)("h2",{id:"money-computations"},"Money computations"),(0,r.kt)("p",null,"They can't be done with floating-point as it would inevitably lead to rounding errors."),(0,r.kt)("p",null,"Even the following packages are problematic:"),(0,r.kt)("p",null,(0,r.kt)("a",{parentName:"p",href:"https://github.com/shopspring/decimal"},"github.com/shopspring/decimal")),(0,r.kt)("p",null,(0,r.kt)("a",{parentName:"p",href:"https://github.com/Rhymond/go-money"},"github.com/Rhymond/go-money")),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-go"},"a := decimal.NewFromInt(2)\nb := decimal.NewFromFloat(300.99)\nc := a.Mul(b)\nd := c.Div(decimal.NewFromInt(3))\n")),(0,r.kt)("h2",{id:"solution"},"Solution"),(0,r.kt)("p",null,"Use Int by representing money in cents:"),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},"10.99 -> 1099 (cents)"),(0,r.kt)("li",{parentName:"ul"},"10.9900 -> 109900 (4 digit tax)")),(0,r.kt)("h2",{id:"conclusio"},"Conclusio"),(0,r.kt)("p",null,"Division is a problem!"),(0,r.kt)("p",null,"1/3 - > 0.33333333\u2026\nCorrect way: 0.33, 0.33, 0.34"),(0,r.kt)("p",null,"When doing money calculations one should avoid division as it inevitably leads to loss of accuracy.\nWhen dividing make sure to round to cent and deal with diffs."),(0,r.kt)("p",null,"Division by 10^k is ok till we are inside of the range of the data type."))}m.isMDXComponent=!0},880:(e,t,n)=>{n.d(t,{Z:()=>o});const o=n.p+"assets/images/floating_point_representation-efc73a8c1b722a82b772c2cd7ee7ba99.webp"}}]); \ No newline at end of file diff --git a/assets/js/26ea2a44.571352e6.js b/assets/js/26ea2a44.571352e6.js deleted file mode 100644 index e4cce7d..0000000 --- a/assets/js/26ea2a44.571352e6.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkfoomo=self.webpackChunkfoomo||[]).push([[1435],{3905:(A,e,t)=>{t.d(e,{Zo:()=>g,kt:()=>d});var l=t(7294);function n(A,e,t){return e in A?Object.defineProperty(A,e,{value:t,enumerable:!0,configurable:!0,writable:!0}):A[e]=t,A}function a(A,e){var t=Object.keys(A);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(A);e&&(l=l.filter((function(e){return Object.getOwnPropertyDescriptor(A,e).enumerable}))),t.push.apply(t,l)}return t}function o(A){for(var e=1;e=0||(n[t]=A[t]);return n}(A,e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(A);for(l=0;l=0||Object.prototype.propertyIsEnumerable.call(A,t)&&(n[t]=A[t])}return n}var r=l.createContext({}),s=function(A){var e=l.useContext(r),t=e;return A&&(t="function"==typeof A?A(e):o(o({},e),A)),t},g=function(A){var e=s(A.components);return l.createElement(r.Provider,{value:e},A.children)},u="mdxType",p={inlineCode:"code",wrapper:function(A){var e=A.children;return l.createElement(l.Fragment,{},e)}},c=l.forwardRef((function(A,e){var t=A.components,n=A.mdxType,a=A.originalType,r=A.parentName,g=i(A,["components","mdxType","originalType","parentName"]),u=s(t),c=n,d=u["".concat(r,".").concat(c)]||u[c]||p[c]||a;return t?l.createElement(d,o(o({ref:e},g),{},{components:t})):l.createElement(d,o({ref:e},g))}));function d(A,e){var t=arguments,n=e&&e.mdxType;if("string"==typeof A||n){var a=t.length,o=new Array(a);o[0]=c;var i={};for(var r in e)hasOwnProperty.call(e,r)&&(i[r]=e[r]);i.originalType=A,i[u]="string"==typeof A?A:n,o[1]=i;for(var s=2;s{t.r(e),t.d(e,{assets:()=>r,contentTitle:()=>o,default:()=>p,frontMatter:()=>a,metadata:()=>i,toc:()=>s});var l=t(7462),n=(t(7294),t(3905));const a={slug:"why-bundle-size-is-important",authors:["nicola"],tags:["javascript","bundle","bundle size"]},o="Why bundle size is important?",i={permalink:"/blog/why-bundle-size-is-important",editUrl:"https://github.com/foomo/foomo-docs/tree/main/foomo/blog/2022-03-17-why-bundle-size-is-important/index.mdx",source:"@site/blog/2022-03-17-why-bundle-size-is-important/index.mdx",title:"Why bundle size is important?",description:"Intro",date:"2022-03-17T00:00:00.000Z",formattedDate:"March 17, 2022",tags:[{label:"javascript",permalink:"/blog/tags/javascript"},{label:"bundle",permalink:"/blog/tags/bundle"},{label:"bundle size",permalink:"/blog/tags/bundle-size"}],hasTruncateMarker:!1,authors:[{name:"Nicola Turcato",title:"Memelord brother",url:"https://github.com/nicolaturcato",imageURL:"https://github.com/nicolaturcato.png",key:"nicola"}],frontMatter:{slug:"why-bundle-size-is-important",authors:["nicola"],tags:["javascript","bundle","bundle size"]},nextItem:{title:"Prometheus Is Out Of Memory. Again.",permalink:"/blog/prometheus-cardinality-issues"}},r={authorsImageUrls:[void 0]},s=[{value:"Intro",id:"intro",level:2},{value:"Nobody likes waiting\u2026",id:"nobody-likes-waiting",level:2},{value:"What is a "bundle"?",id:"what-is-a-bundle",level:2},{value:"Performance implications",id:"performance-implications",level:2},{value:"What is the recommended bundle size?",id:"what-is-the-recommended-bundle-size",level:2},{value:"What do we do then?",id:"what-do-we-do-then",level:2},{value:"How to start decreasing the bundle size?",id:"how-to-start-decreasing-the-bundle-size",level:2},{value:"Breaking up the bundle...",id:"breaking-up-the-bundle",level:2},{value:"Which strategies can we adopt?",id:"which-strategies-can-we-adopt",level:2},{value:"Useful tools to help you reducing bundle size",id:"useful-tools-to-help-you-reducing-bundle-size",level:2},{value:"Conclusion",id:"conclusion",level:2}],g={toc:s},u="wrapper";function p(A){let{components:e,...a}=A;return(0,n.kt)(u,(0,l.Z)({},g,a,{components:e,mdxType:"MDXLayout"}),(0,n.kt)("h2",{id:"intro"},"Intro"),(0,n.kt)("p",null,"JavaScript is parsed, compiled and executed in the main thread of the browser. Which means that users have to wait for all of this before they can interact with the website."),(0,n.kt)("p",null,"Frontend performance optimization is critical because\xa0it accounts for around 80-90% of user response time (10-20% backend).\nSo when a user is waiting for a page to load, around 80-90% of the time is due to frontend related code and assets."),(0,n.kt)("h2",{id:"nobody-likes-waiting"},"Nobody likes waiting\u2026"),(0,n.kt)("p",null,"A\xa0study found\xa0that if a site takes longer than 4 seconds to load, up to 25% of users would abandon the site."),(0,n.kt)("p",null,"Sending large JavaScript payloads impacts the speed of your site significantly."),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Mazzarri",src:t(7017).Z,width:"410",height:"276"})),(0,n.kt)("h2",{id:"what-is-a-bundle"},'What is a "bundle"?'),(0,n.kt)("p",null,"Your frontend application needs a bunch of JS files to run. These files can be in the format of internal dependency like the JS files you have written yourself. But they can also be external dependencies you use to build your application."),(0,n.kt)("p",null,"JS bundling is\xa0an optimization technique to reduce the number of server requests for JavaScript files. Bundling accomplishes this by merging multiple JA files together into one file to reduce the number of page requests."),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Bundle Everywhere",src:t(6865).Z,width:"448",height:"245"})),(0,n.kt)("h2",{id:"performance-implications"},"Performance implications"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Time to transmit over the network"),": considering slow connections with some mobile devices, it's possible that your page will not be interactive until it loads.-> More bytes = longer download times"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"JS parse and compile time"),": more code you load, more the browser must parse.-> JS gets parsed and compiled on the main thread, when the main thread is busy, the page can't respond to user input"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"JS execution time"),": optimally you will only pack the code that you expect to execute. The more code you want to execute the longer it will take. It's possible that your page won't be interactive until some of this completes.-> JS is also executed on the main thread, if your page runs a lot of code before it's really needed, that also delays your Time to Interactive"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Memory consumption"),": everything fills up the space -> code itself, runtime variables, DOM elements created, etc.-> Pages appear slow when it consumes a lot of memory. Memory leaks can cause your page to freeze up completely!!")),(0,n.kt)("h2",{id:"what-is-the-recommended-bundle-size"},"What is the recommended bundle size?"),(0,n.kt)("p",null,"AS SMALL AS POSSIBLE! I experienced that is not really possible to give a precise answer because each application is different. Generally you want the resources on initial load to be as small as possible, so that you decrease the initial load time, and then load more resources as needed on the fly."),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Mr Chao",src:t(1262).Z,width:"406",height:"250"})),(0,n.kt)("h2",{id:"what-do-we-do-then"},"What do we do then?"),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Meh",src:t(4812).Z,width:"539",height:"399"})),(0,n.kt)("h2",{id:"how-to-start-decreasing-the-bundle-size"},"How to start decreasing the bundle size?"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Measure"),": first of all you want to measure. The first step is to use Lighthouse and try to understand the results. It will give you a couple of interesting metrics and some tips. Time to interactive (TTI)\xa0is a good reflection of your bundle size because your bundle needs to be evaluated entirely before a user can interact with your web app."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Analyze"),": Consists on analyzing the bundle in order to detect critical chunks. A useful tool is Webpack Bundle Analyzer.")),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Stonks",src:t(3400).Z,width:"309",height:"229"})),(0,n.kt)("h2",{id:"breaking-up-the-bundle"},"Breaking up the bundle..."),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Monitor network requests"),": These happens between our FCP and TTI. As the initial request for data often occurs when our components initially mount."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Reduce the total dom nodes"),": the less the page needs to render, the less time it takes."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Moving work off the main thread"),": By moving heavy computations to a web worker, the computation will be run on a separate thread, and not block the actual rendering of the page"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Caching"),": Even if not useful for users on first page landing, caching data, bundles, and assets can make subsequent visits way fast")),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Breaking Bad",src:t(1278).Z,width:"299",height:"230"})),(0,n.kt)("h2",{id:"which-strategies-can-we-adopt"},"Which strategies can we adopt?"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Minification and Dead Code Elimination"),": These processes are often summed up as\xa0minifying\xa0or\xa0uglifying."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Tree shaking"),": Tree shaking is dead code elimination on a project or library. Always try to use deps which support \u201ctree shaking\u201d, Bundlephobia could be your friend in this case."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Code Splitting and Lazy Loading"),": Code splitting consists on taking a collection of modules and remove them from the main JS bundle. Lazy loading means we can load this newly created bundle later on."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Replace/rewrite large dependencies"),": Consider replacing or rewriting libraries that are large in size where you might not need all of its functionalities (Moment.js for example)."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Feature module import"),": Check to see if you are using only a feature module of the library that can be imported alone without importing the whole library (Lodash for example).")),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Strategy",src:t(2235).Z,width:"307",height:"204"})),(0,n.kt)("h2",{id:"useful-tools-to-help-you-reducing-bundle-size"},"Useful tools to help you reducing bundle size"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Lighthouse"),": automated tool for improving the performance, quality, and correctness of your web apps"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Bundlephobia"),": Bundlephobia\xa0helps you find the performance impact of npm packages"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Webpack Bundle Analyzer"),": analyzes your bundle"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"VS Code"),": Import Cost plugin -> Display import/require package size in the editor")),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Tools",src:t(4941).Z,width:"345",height:"255"})),(0,n.kt)("h2",{id:"conclusion"},"Conclusion"),(0,n.kt)("p",null,"Performance cannot be stripped down to a single metric such as bundle size. It would be great!\nUnfortunately there is no single place to measure all of them.I think metrics like the\xa0Core Web Vitals and a general look at bundle size should be considered as a starting point.\nYou will cry... A lot... But don\u2019t give up!"),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"The End",src:t(6899).Z,width:"358",height:"260"})))}p.isMDXComponent=!0},1278:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},6865:(A,e,t)=>{t.d(e,{Z:()=>l});const l=t.p+"assets/images/bundle_everywhere-12122e37ab5ca6bcdd61061c4dc32fd4.webp"},7017:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},4812:(A,e,t)=>{t.d(e,{Z:()=>l});const l=t.p+"assets/images/meh-3b42aead7d2246467fd0101518403734.webp"},1262:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},3400:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},2235:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},6899:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},4941:(A,e,t)=>{t.d(e,{Z:()=>l});const l=t.p+"assets/images/tools-91886b41cefeef7cc35b5cd21c43d2ba.webp"}}]); \ No newline at end of file diff --git a/assets/js/26ea2a44.e95e0c70.js b/assets/js/26ea2a44.e95e0c70.js new file mode 100644 index 0000000..63deced --- /dev/null +++ b/assets/js/26ea2a44.e95e0c70.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkfoomo=self.webpackChunkfoomo||[]).push([[1435],{3905:(A,e,t)=>{t.d(e,{Zo:()=>g,kt:()=>d});var l=t(7294);function n(A,e,t){return e in A?Object.defineProperty(A,e,{value:t,enumerable:!0,configurable:!0,writable:!0}):A[e]=t,A}function a(A,e){var t=Object.keys(A);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(A);e&&(l=l.filter((function(e){return Object.getOwnPropertyDescriptor(A,e).enumerable}))),t.push.apply(t,l)}return t}function o(A){for(var e=1;e=0||(n[t]=A[t]);return n}(A,e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(A);for(l=0;l=0||Object.prototype.propertyIsEnumerable.call(A,t)&&(n[t]=A[t])}return n}var r=l.createContext({}),s=function(A){var e=l.useContext(r),t=e;return A&&(t="function"==typeof A?A(e):o(o({},e),A)),t},g=function(A){var e=s(A.components);return l.createElement(r.Provider,{value:e},A.children)},u="mdxType",c={inlineCode:"code",wrapper:function(A){var e=A.children;return l.createElement(l.Fragment,{},e)}},p=l.forwardRef((function(A,e){var t=A.components,n=A.mdxType,a=A.originalType,r=A.parentName,g=i(A,["components","mdxType","originalType","parentName"]),u=s(t),p=n,d=u["".concat(r,".").concat(p)]||u[p]||c[p]||a;return t?l.createElement(d,o(o({ref:e},g),{},{components:t})):l.createElement(d,o({ref:e},g))}));function d(A,e){var t=arguments,n=e&&e.mdxType;if("string"==typeof A||n){var a=t.length,o=new Array(a);o[0]=p;var i={};for(var r in e)hasOwnProperty.call(e,r)&&(i[r]=e[r]);i.originalType=A,i[u]="string"==typeof A?A:n,o[1]=i;for(var s=2;s{t.r(e),t.d(e,{assets:()=>r,contentTitle:()=>o,default:()=>c,frontMatter:()=>a,metadata:()=>i,toc:()=>s});var l=t(7462),n=(t(7294),t(3905));const a={slug:"why-bundle-size-is-important",authors:["nicola"],tags:["javascript","bundle","bundle size"]},o="Why bundle size is important?",i={permalink:"/blog/why-bundle-size-is-important",editUrl:"https://github.com/foomo/foomo-docs/tree/main/foomo/blog/2022-03-17-why-bundle-size-is-important/index.mdx",source:"@site/blog/2022-03-17-why-bundle-size-is-important/index.mdx",title:"Why bundle size is important?",description:"Intro",date:"2022-03-17T00:00:00.000Z",formattedDate:"March 17, 2022",tags:[{label:"javascript",permalink:"/blog/tags/javascript"},{label:"bundle",permalink:"/blog/tags/bundle"},{label:"bundle size",permalink:"/blog/tags/bundle-size"}],hasTruncateMarker:!1,authors:[{name:"Nicola Turcato",title:"Memelord brother",url:"https://github.com/nicolaturcato",imageURL:"https://github.com/nicolaturcato.png",key:"nicola"}],frontMatter:{slug:"why-bundle-size-is-important",authors:["nicola"],tags:["javascript","bundle","bundle size"]},prevItem:{title:"Accuracy of decimal computations",permalink:"/blog/accuracy-of-decimal-computations"},nextItem:{title:"Prometheus Is Out Of Memory. Again.",permalink:"/blog/prometheus-cardinality-issues"}},r={authorsImageUrls:[void 0]},s=[{value:"Intro",id:"intro",level:2},{value:"Nobody likes waiting\u2026",id:"nobody-likes-waiting",level:2},{value:"What is a "bundle"?",id:"what-is-a-bundle",level:2},{value:"Performance implications",id:"performance-implications",level:2},{value:"What is the recommended bundle size?",id:"what-is-the-recommended-bundle-size",level:2},{value:"What do we do then?",id:"what-do-we-do-then",level:2},{value:"How to start decreasing the bundle size?",id:"how-to-start-decreasing-the-bundle-size",level:2},{value:"Breaking up the bundle...",id:"breaking-up-the-bundle",level:2},{value:"Which strategies can we adopt?",id:"which-strategies-can-we-adopt",level:2},{value:"Useful tools to help you reducing bundle size",id:"useful-tools-to-help-you-reducing-bundle-size",level:2},{value:"Conclusion",id:"conclusion",level:2}],g={toc:s},u="wrapper";function c(A){let{components:e,...a}=A;return(0,n.kt)(u,(0,l.Z)({},g,a,{components:e,mdxType:"MDXLayout"}),(0,n.kt)("h2",{id:"intro"},"Intro"),(0,n.kt)("p",null,"JavaScript is parsed, compiled and executed in the main thread of the browser. Which means that users have to wait for all of this before they can interact with the website."),(0,n.kt)("p",null,"Frontend performance optimization is critical because\xa0it accounts for around 80-90% of user response time (10-20% backend).\nSo when a user is waiting for a page to load, around 80-90% of the time is due to frontend related code and assets."),(0,n.kt)("h2",{id:"nobody-likes-waiting"},"Nobody likes waiting\u2026"),(0,n.kt)("p",null,"A\xa0study found\xa0that if a site takes longer than 4 seconds to load, up to 25% of users would abandon the site."),(0,n.kt)("p",null,"Sending large JavaScript payloads impacts the speed of your site significantly."),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Mazzarri",src:t(7017).Z,width:"410",height:"276"})),(0,n.kt)("h2",{id:"what-is-a-bundle"},'What is a "bundle"?'),(0,n.kt)("p",null,"Your frontend application needs a bunch of JS files to run. These files can be in the format of internal dependency like the JS files you have written yourself. But they can also be external dependencies you use to build your application."),(0,n.kt)("p",null,"JS bundling is\xa0an optimization technique to reduce the number of server requests for JavaScript files. Bundling accomplishes this by merging multiple JA files together into one file to reduce the number of page requests."),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Bundle Everywhere",src:t(6865).Z,width:"448",height:"245"})),(0,n.kt)("h2",{id:"performance-implications"},"Performance implications"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Time to transmit over the network"),": considering slow connections with some mobile devices, it's possible that your page will not be interactive until it loads.-> More bytes = longer download times"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"JS parse and compile time"),": more code you load, more the browser must parse.-> JS gets parsed and compiled on the main thread, when the main thread is busy, the page can't respond to user input"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"JS execution time"),": optimally you will only pack the code that you expect to execute. The more code you want to execute the longer it will take. It's possible that your page won't be interactive until some of this completes.-> JS is also executed on the main thread, if your page runs a lot of code before it's really needed, that also delays your Time to Interactive"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Memory consumption"),": everything fills up the space -> code itself, runtime variables, DOM elements created, etc.-> Pages appear slow when it consumes a lot of memory. Memory leaks can cause your page to freeze up completely!!")),(0,n.kt)("h2",{id:"what-is-the-recommended-bundle-size"},"What is the recommended bundle size?"),(0,n.kt)("p",null,"AS SMALL AS POSSIBLE! I experienced that is not really possible to give a precise answer because each application is different. Generally you want the resources on initial load to be as small as possible, so that you decrease the initial load time, and then load more resources as needed on the fly."),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Mr Chao",src:t(1262).Z,width:"406",height:"250"})),(0,n.kt)("h2",{id:"what-do-we-do-then"},"What do we do then?"),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Meh",src:t(4812).Z,width:"539",height:"399"})),(0,n.kt)("h2",{id:"how-to-start-decreasing-the-bundle-size"},"How to start decreasing the bundle size?"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Measure"),": first of all you want to measure. The first step is to use Lighthouse and try to understand the results. It will give you a couple of interesting metrics and some tips. Time to interactive (TTI)\xa0is a good reflection of your bundle size because your bundle needs to be evaluated entirely before a user can interact with your web app."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Analyze"),": Consists on analyzing the bundle in order to detect critical chunks. A useful tool is Webpack Bundle Analyzer.")),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Stonks",src:t(3400).Z,width:"309",height:"229"})),(0,n.kt)("h2",{id:"breaking-up-the-bundle"},"Breaking up the bundle..."),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Monitor network requests"),": These happens between our FCP and TTI. As the initial request for data often occurs when our components initially mount."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Reduce the total dom nodes"),": the less the page needs to render, the less time it takes."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Moving work off the main thread"),": By moving heavy computations to a web worker, the computation will be run on a separate thread, and not block the actual rendering of the page"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Caching"),": Even if not useful for users on first page landing, caching data, bundles, and assets can make subsequent visits way fast")),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Breaking Bad",src:t(1278).Z,width:"299",height:"230"})),(0,n.kt)("h2",{id:"which-strategies-can-we-adopt"},"Which strategies can we adopt?"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Minification and Dead Code Elimination"),": These processes are often summed up as\xa0minifying\xa0or\xa0uglifying."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Tree shaking"),": Tree shaking is dead code elimination on a project or library. Always try to use deps which support \u201ctree shaking\u201d, Bundlephobia could be your friend in this case."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Code Splitting and Lazy Loading"),": Code splitting consists on taking a collection of modules and remove them from the main JS bundle. Lazy loading means we can load this newly created bundle later on."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Replace/rewrite large dependencies"),": Consider replacing or rewriting libraries that are large in size where you might not need all of its functionalities (Moment.js for example)."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Feature module import"),": Check to see if you are using only a feature module of the library that can be imported alone without importing the whole library (Lodash for example).")),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Strategy",src:t(2235).Z,width:"307",height:"204"})),(0,n.kt)("h2",{id:"useful-tools-to-help-you-reducing-bundle-size"},"Useful tools to help you reducing bundle size"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Lighthouse"),": automated tool for improving the performance, quality, and correctness of your web apps"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Bundlephobia"),": Bundlephobia\xa0helps you find the performance impact of npm packages"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Webpack Bundle Analyzer"),": analyzes your bundle"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"VS Code"),": Import Cost plugin -> Display import/require package size in the editor")),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Tools",src:t(4941).Z,width:"345",height:"255"})),(0,n.kt)("h2",{id:"conclusion"},"Conclusion"),(0,n.kt)("p",null,"Performance cannot be stripped down to a single metric such as bundle size. It would be great!\nUnfortunately there is no single place to measure all of them.I think metrics like the\xa0Core Web Vitals and a general look at bundle size should be considered as a starting point.\nYou will cry... A lot... But don\u2019t give up!"),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"The End",src:t(6899).Z,width:"358",height:"260"})))}c.isMDXComponent=!0},1278:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},6865:(A,e,t)=>{t.d(e,{Z:()=>l});const l=t.p+"assets/images/bundle_everywhere-12122e37ab5ca6bcdd61061c4dc32fd4.webp"},7017:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},4812:(A,e,t)=>{t.d(e,{Z:()=>l});const l=t.p+"assets/images/meh-3b42aead7d2246467fd0101518403734.webp"},1262:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},3400:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},2235:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},6899:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},4941:(A,e,t)=>{t.d(e,{Z:()=>l});const l=t.p+"assets/images/tools-91886b41cefeef7cc35b5cd21c43d2ba.webp"}}]); \ No newline at end of file diff --git a/assets/js/487ae42e.a0943f9c.js b/assets/js/487ae42e.a0943f9c.js new file mode 100644 index 0000000..d730d76 --- /dev/null +++ b/assets/js/487ae42e.a0943f9c.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkfoomo=self.webpackChunkfoomo||[]).push([[9851],{6920:a=>{a.exports=JSON.parse('{"label":"decimal accuracy","permalink":"/blog/tags/decimal-accuracy","allTagsPath":"/blog/tags","count":1}')}}]); \ No newline at end of file diff --git a/assets/js/6b9bd0c6.76e75916.js b/assets/js/6b9bd0c6.76e75916.js new file mode 100644 index 0000000..b285b49 --- /dev/null +++ b/assets/js/6b9bd0c6.76e75916.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkfoomo=self.webpackChunkfoomo||[]).push([[7609],{3905:(e,t,n)=>{n.d(t,{Zo:()=>s,kt:()=>f});var o=n(7294);function r(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function a(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var o=Object.getOwnPropertySymbols(e);t&&(o=o.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,o)}return n}function i(e){for(var t=1;t=0||(r[n]=e[n]);return r}(e,t);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);for(o=0;o=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(r[n]=e[n])}return r}var l=o.createContext({}),u=function(e){var t=o.useContext(l),n=t;return e&&(n="function"==typeof e?e(t):i(i({},t),e)),n},s=function(e){var t=u(e.components);return o.createElement(l.Provider,{value:t},e.children)},p="mdxType",m={inlineCode:"code",wrapper:function(e){var t=e.children;return o.createElement(o.Fragment,{},t)}},d=o.forwardRef((function(e,t){var n=e.components,r=e.mdxType,a=e.originalType,l=e.parentName,s=c(e,["components","mdxType","originalType","parentName"]),p=u(n),d=r,f=p["".concat(l,".").concat(d)]||p[d]||m[d]||a;return n?o.createElement(f,i(i({ref:t},s),{},{components:n})):o.createElement(f,i({ref:t},s))}));function f(e,t){var n=arguments,r=t&&t.mdxType;if("string"==typeof e||r){var a=n.length,i=new Array(a);i[0]=d;var c={};for(var l in t)hasOwnProperty.call(t,l)&&(c[l]=t[l]);c.originalType=e,c[p]="string"==typeof e?e:r,i[1]=c;for(var u=2;u{n.r(t),n.d(t,{assets:()=>l,contentTitle:()=>i,default:()=>m,frontMatter:()=>a,metadata:()=>c,toc:()=>u});var o=n(7462),r=(n(7294),n(3905));const a={slug:"accuracy-of-decimal-computations",authors:["patrickb"],tags:["golang","currency","decimal accuracy"]},i="Accuracy of decimal computations",c={permalink:"/blog/accuracy-of-decimal-computations",editUrl:"https://github.com/foomo/foomo-docs/tree/main/foomo/blog/2023-03-06-accuracy-of-decimal-computations/index.mdx",source:"@site/blog/2023-03-06-accuracy-of-decimal-computations/index.mdx",title:"Accuracy of decimal computations",description:"Intro",date:"2023-03-06T00:00:00.000Z",formattedDate:"March 6, 2023",tags:[{label:"golang",permalink:"/blog/tags/golang"},{label:"currency",permalink:"/blog/tags/currency"},{label:"decimal accuracy",permalink:"/blog/tags/decimal-accuracy"}],hasTruncateMarker:!1,authors:[{name:"Patrick Buchner",title:"MSc",url:"https://github.com/smartinov",imageURL:"https://github.com/uebriges.png",key:"patrickb"}],frontMatter:{slug:"accuracy-of-decimal-computations",authors:["patrickb"],tags:["golang","currency","decimal accuracy"]},nextItem:{title:"Why bundle size is important?",permalink:"/blog/why-bundle-size-is-important"}},l={authorsImageUrls:[void 0]},u=[{value:"Intro",id:"intro",level:2},{value:"Finite accuracy of representation",id:"finite-accuracy-of-representation",level:2},{value:"Money computations",id:"money-computations",level:2},{value:"Solution",id:"solution",level:2},{value:"Conclusio",id:"conclusio",level:2}],s={toc:u},p="wrapper";function m(e){let{components:t,...a}=e;return(0,r.kt)(p,(0,o.Z)({},s,a,{components:t,mdxType:"MDXLayout"}),(0,r.kt)("h2",{id:"intro"},"Intro"),(0,r.kt)("p",null,"Calculating with money can be tricky if not taken proper precautions. Some might be tempted to use float representation for calculating with currency values. That is problematic because of possible rounding errors."),(0,r.kt)("h2",{id:"finite-accuracy-of-representation"},"Finite accuracy of representation"),(0,r.kt)("p",null,"Floating points are represented like this"),(0,r.kt)("p",null,(0,r.kt)("img",{alt:"Floating point representation",src:n(880).Z,width:"810",height:"226"})),(0,r.kt)("p",null,"Not every number can be represented with a finite number of decimal places"),(0,r.kt)("p",null,"0.01 \u2014> 0.0000011001100110011\u2026"),(0,r.kt)("p",null,"Taking 17 places of the above results in 0.010000000000000001"),(0,r.kt)("p",null,"Consider the following code snipet that shows the missing accuracy"),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-go"},"func main() {\n\n var n float64 = 0\n\n for i := 0; i < 1000; i++ {\n n += .01\n }\n\n fmt.Println(n)\n\n}\n")),(0,r.kt)("p",null,"Result: 9.999999999999831"),(0,r.kt)("h2",{id:"money-computations"},"Money computations"),(0,r.kt)("p",null,"They can't be done with floating-point as it would inevitably lead to rounding errors."),(0,r.kt)("p",null,"Even the following packages are problematic:"),(0,r.kt)("p",null,(0,r.kt)("a",{parentName:"p",href:"https://github.com/shopspring/decimal"},"github.com/shopspring/decimal")),(0,r.kt)("p",null,(0,r.kt)("a",{parentName:"p",href:"https://github.com/Rhymond/go-money"},"github.com/Rhymond/go-money")),(0,r.kt)("pre",null,(0,r.kt)("code",{parentName:"pre",className:"language-go"},"a := decimal.NewFromInt(2)\nb := decimal.NewFromFloat(300.99)\nc := a.Mul(b)\nd := c.Div(decimal.NewFromInt(3))\n")),(0,r.kt)("h2",{id:"solution"},"Solution"),(0,r.kt)("p",null,"Use Int by representing money in cents:"),(0,r.kt)("ul",null,(0,r.kt)("li",{parentName:"ul"},"10.99 -> 1099 (cents)"),(0,r.kt)("li",{parentName:"ul"},"10.9900 -> 109900 (4 digit tax)")),(0,r.kt)("h2",{id:"conclusio"},"Conclusio"),(0,r.kt)("p",null,"Division is a problem!"),(0,r.kt)("p",null,"1/3 - > 0.33333333\u2026\nCorrect way: 0.33, 0.33, 0.34"),(0,r.kt)("p",null,"When doing money calculations one should avoid division as it inevitably leads to loss of accuracy.\nWhen dividing make sure to round to cent and deal with diffs."),(0,r.kt)("p",null,"Division by 10^k is ok till we are inside of the range of the data type."))}m.isMDXComponent=!0},880:(e,t,n)=>{n.d(t,{Z:()=>o});const o=n.p+"assets/images/floating_point_representation-efc73a8c1b722a82b772c2cd7ee7ba99.webp"}}]); \ No newline at end of file diff --git a/assets/js/814f3328.7b160b12.js b/assets/js/814f3328.7b160b12.js deleted file mode 100644 index 20cb355..0000000 --- a/assets/js/814f3328.7b160b12.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkfoomo=self.webpackChunkfoomo||[]).push([[2535],{5641:e=>{e.exports=JSON.parse('{"title":"Recent posts","items":[{"title":"Why bundle size is important?","permalink":"/blog/why-bundle-size-is-important"},{"title":"Prometheus Is Out Of Memory. Again.","permalink":"/blog/prometheus-cardinality-issues"},{"title":"The never ending search a search engine 2022-01 edition","permalink":"/blog/searching-for-search-engines"},{"title":"Impact of 3rd party scripts on performance","permalink":"/blog/impact-of-3rd-party-scripts-on-performance"},{"title":"debugging Go map races in k8s","permalink":"/blog/debugging-go-map-races-in-k8s"}]}')}}]); \ No newline at end of file diff --git a/assets/js/814f3328.f556ba25.js b/assets/js/814f3328.f556ba25.js new file mode 100644 index 0000000..41303ce --- /dev/null +++ b/assets/js/814f3328.f556ba25.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkfoomo=self.webpackChunkfoomo||[]).push([[2535],{5641:e=>{e.exports=JSON.parse('{"title":"Recent posts","items":[{"title":"Accuracy of decimal computations","permalink":"/blog/accuracy-of-decimal-computations"},{"title":"Why bundle size is important?","permalink":"/blog/why-bundle-size-is-important"},{"title":"Prometheus Is Out Of Memory. Again.","permalink":"/blog/prometheus-cardinality-issues"},{"title":"The never ending search a search engine 2022-01 edition","permalink":"/blog/searching-for-search-engines"},{"title":"Impact of 3rd party scripts on performance","permalink":"/blog/impact-of-3rd-party-scripts-on-performance"}]}')}}]); \ No newline at end of file diff --git a/assets/js/87b8e9c4.24bc7ea5.js b/assets/js/87b8e9c4.24bc7ea5.js deleted file mode 100644 index 1412ddf..0000000 --- a/assets/js/87b8e9c4.24bc7ea5.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkfoomo=self.webpackChunkfoomo||[]).push([[2423],{3905:(A,e,t)=>{t.d(e,{Zo:()=>g,kt:()=>d});var l=t(7294);function n(A,e,t){return e in A?Object.defineProperty(A,e,{value:t,enumerable:!0,configurable:!0,writable:!0}):A[e]=t,A}function a(A,e){var t=Object.keys(A);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(A);e&&(l=l.filter((function(e){return Object.getOwnPropertyDescriptor(A,e).enumerable}))),t.push.apply(t,l)}return t}function o(A){for(var e=1;e=0||(n[t]=A[t]);return n}(A,e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(A);for(l=0;l=0||Object.prototype.propertyIsEnumerable.call(A,t)&&(n[t]=A[t])}return n}var r=l.createContext({}),s=function(A){var e=l.useContext(r),t=e;return A&&(t="function"==typeof A?A(e):o(o({},e),A)),t},g=function(A){var e=s(A.components);return l.createElement(r.Provider,{value:e},A.children)},u="mdxType",p={inlineCode:"code",wrapper:function(A){var e=A.children;return l.createElement(l.Fragment,{},e)}},c=l.forwardRef((function(A,e){var t=A.components,n=A.mdxType,a=A.originalType,r=A.parentName,g=i(A,["components","mdxType","originalType","parentName"]),u=s(t),c=n,d=u["".concat(r,".").concat(c)]||u[c]||p[c]||a;return t?l.createElement(d,o(o({ref:e},g),{},{components:t})):l.createElement(d,o({ref:e},g))}));function d(A,e){var t=arguments,n=e&&e.mdxType;if("string"==typeof A||n){var a=t.length,o=new Array(a);o[0]=c;var i={};for(var r in e)hasOwnProperty.call(e,r)&&(i[r]=e[r]);i.originalType=A,i[u]="string"==typeof A?A:n,o[1]=i;for(var s=2;s{t.r(e),t.d(e,{assets:()=>r,contentTitle:()=>o,default:()=>p,frontMatter:()=>a,metadata:()=>i,toc:()=>s});var l=t(7462),n=(t(7294),t(3905));const a={slug:"why-bundle-size-is-important",authors:["nicola"],tags:["javascript","bundle","bundle size"]},o="Why bundle size is important?",i={permalink:"/blog/why-bundle-size-is-important",editUrl:"https://github.com/foomo/foomo-docs/tree/main/foomo/blog/2022-03-17-why-bundle-size-is-important/index.mdx",source:"@site/blog/2022-03-17-why-bundle-size-is-important/index.mdx",title:"Why bundle size is important?",description:"Intro",date:"2022-03-17T00:00:00.000Z",formattedDate:"March 17, 2022",tags:[{label:"javascript",permalink:"/blog/tags/javascript"},{label:"bundle",permalink:"/blog/tags/bundle"},{label:"bundle size",permalink:"/blog/tags/bundle-size"}],hasTruncateMarker:!1,authors:[{name:"Nicola Turcato",title:"Memelord brother",url:"https://github.com/nicolaturcato",imageURL:"https://github.com/nicolaturcato.png",key:"nicola"}],frontMatter:{slug:"why-bundle-size-is-important",authors:["nicola"],tags:["javascript","bundle","bundle size"]},nextItem:{title:"Prometheus Is Out Of Memory. Again.",permalink:"/blog/prometheus-cardinality-issues"}},r={authorsImageUrls:[void 0]},s=[{value:"Intro",id:"intro",level:2},{value:"Nobody likes waiting\u2026",id:"nobody-likes-waiting",level:2},{value:"What is a "bundle"?",id:"what-is-a-bundle",level:2},{value:"Performance implications",id:"performance-implications",level:2},{value:"What is the recommended bundle size?",id:"what-is-the-recommended-bundle-size",level:2},{value:"What do we do then?",id:"what-do-we-do-then",level:2},{value:"How to start decreasing the bundle size?",id:"how-to-start-decreasing-the-bundle-size",level:2},{value:"Breaking up the bundle...",id:"breaking-up-the-bundle",level:2},{value:"Which strategies can we adopt?",id:"which-strategies-can-we-adopt",level:2},{value:"Useful tools to help you reducing bundle size",id:"useful-tools-to-help-you-reducing-bundle-size",level:2},{value:"Conclusion",id:"conclusion",level:2}],g={toc:s},u="wrapper";function p(A){let{components:e,...a}=A;return(0,n.kt)(u,(0,l.Z)({},g,a,{components:e,mdxType:"MDXLayout"}),(0,n.kt)("h2",{id:"intro"},"Intro"),(0,n.kt)("p",null,"JavaScript is parsed, compiled and executed in the main thread of the browser. Which means that users have to wait for all of this before they can interact with the website."),(0,n.kt)("p",null,"Frontend performance optimization is critical because\xa0it accounts for around 80-90% of user response time (10-20% backend).\nSo when a user is waiting for a page to load, around 80-90% of the time is due to frontend related code and assets."),(0,n.kt)("h2",{id:"nobody-likes-waiting"},"Nobody likes waiting\u2026"),(0,n.kt)("p",null,"A\xa0study found\xa0that if a site takes longer than 4 seconds to load, up to 25% of users would abandon the site."),(0,n.kt)("p",null,"Sending large JavaScript payloads impacts the speed of your site significantly."),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Mazzarri",src:t(7017).Z,width:"410",height:"276"})),(0,n.kt)("h2",{id:"what-is-a-bundle"},'What is a "bundle"?'),(0,n.kt)("p",null,"Your frontend application needs a bunch of JS files to run. These files can be in the format of internal dependency like the JS files you have written yourself. But they can also be external dependencies you use to build your application."),(0,n.kt)("p",null,"JS bundling is\xa0an optimization technique to reduce the number of server requests for JavaScript files. Bundling accomplishes this by merging multiple JA files together into one file to reduce the number of page requests."),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Bundle Everywhere",src:t(6865).Z,width:"448",height:"245"})),(0,n.kt)("h2",{id:"performance-implications"},"Performance implications"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Time to transmit over the network"),": considering slow connections with some mobile devices, it's possible that your page will not be interactive until it loads.-> More bytes = longer download times"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"JS parse and compile time"),": more code you load, more the browser must parse.-> JS gets parsed and compiled on the main thread, when the main thread is busy, the page can't respond to user input"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"JS execution time"),": optimally you will only pack the code that you expect to execute. The more code you want to execute the longer it will take. It's possible that your page won't be interactive until some of this completes.-> JS is also executed on the main thread, if your page runs a lot of code before it's really needed, that also delays your Time to Interactive"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Memory consumption"),": everything fills up the space -> code itself, runtime variables, DOM elements created, etc.-> Pages appear slow when it consumes a lot of memory. Memory leaks can cause your page to freeze up completely!!")),(0,n.kt)("h2",{id:"what-is-the-recommended-bundle-size"},"What is the recommended bundle size?"),(0,n.kt)("p",null,"AS SMALL AS POSSIBLE! I experienced that is not really possible to give a precise answer because each application is different. Generally you want the resources on initial load to be as small as possible, so that you decrease the initial load time, and then load more resources as needed on the fly."),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Mr Chao",src:t(1262).Z,width:"406",height:"250"})),(0,n.kt)("h2",{id:"what-do-we-do-then"},"What do we do then?"),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Meh",src:t(4812).Z,width:"539",height:"399"})),(0,n.kt)("h2",{id:"how-to-start-decreasing-the-bundle-size"},"How to start decreasing the bundle size?"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Measure"),": first of all you want to measure. The first step is to use Lighthouse and try to understand the results. It will give you a couple of interesting metrics and some tips. Time to interactive (TTI)\xa0is a good reflection of your bundle size because your bundle needs to be evaluated entirely before a user can interact with your web app."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Analyze"),": Consists on analyzing the bundle in order to detect critical chunks. A useful tool is Webpack Bundle Analyzer.")),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Stonks",src:t(3400).Z,width:"309",height:"229"})),(0,n.kt)("h2",{id:"breaking-up-the-bundle"},"Breaking up the bundle..."),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Monitor network requests"),": These happens between our FCP and TTI. As the initial request for data often occurs when our components initially mount."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Reduce the total dom nodes"),": the less the page needs to render, the less time it takes."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Moving work off the main thread"),": By moving heavy computations to a web worker, the computation will be run on a separate thread, and not block the actual rendering of the page"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Caching"),": Even if not useful for users on first page landing, caching data, bundles, and assets can make subsequent visits way fast")),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Breaking Bad",src:t(1278).Z,width:"299",height:"230"})),(0,n.kt)("h2",{id:"which-strategies-can-we-adopt"},"Which strategies can we adopt?"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Minification and Dead Code Elimination"),": These processes are often summed up as\xa0minifying\xa0or\xa0uglifying."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Tree shaking"),": Tree shaking is dead code elimination on a project or library. Always try to use deps which support \u201ctree shaking\u201d, Bundlephobia could be your friend in this case."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Code Splitting and Lazy Loading"),": Code splitting consists on taking a collection of modules and remove them from the main JS bundle. Lazy loading means we can load this newly created bundle later on."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Replace/rewrite large dependencies"),": Consider replacing or rewriting libraries that are large in size where you might not need all of its functionalities (Moment.js for example)."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Feature module import"),": Check to see if you are using only a feature module of the library that can be imported alone without importing the whole library (Lodash for example).")),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Strategy",src:t(2235).Z,width:"307",height:"204"})),(0,n.kt)("h2",{id:"useful-tools-to-help-you-reducing-bundle-size"},"Useful tools to help you reducing bundle size"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Lighthouse"),": automated tool for improving the performance, quality, and correctness of your web apps"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Bundlephobia"),": Bundlephobia\xa0helps you find the performance impact of npm packages"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Webpack Bundle Analyzer"),": analyzes your bundle"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"VS Code"),": Import Cost plugin -> Display import/require package size in the editor")),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Tools",src:t(4941).Z,width:"345",height:"255"})),(0,n.kt)("h2",{id:"conclusion"},"Conclusion"),(0,n.kt)("p",null,"Performance cannot be stripped down to a single metric such as bundle size. It would be great!\nUnfortunately there is no single place to measure all of them.I think metrics like the\xa0Core Web Vitals and a general look at bundle size should be considered as a starting point.\nYou will cry... A lot... But don\u2019t give up!"),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"The End",src:t(6899).Z,width:"358",height:"260"})))}p.isMDXComponent=!0},1278:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},6865:(A,e,t)=>{t.d(e,{Z:()=>l});const l=t.p+"assets/images/bundle_everywhere-12122e37ab5ca6bcdd61061c4dc32fd4.webp"},7017:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},4812:(A,e,t)=>{t.d(e,{Z:()=>l});const l=t.p+"assets/images/meh-3b42aead7d2246467fd0101518403734.webp"},1262:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},3400:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},2235:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},6899:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},4941:(A,e,t)=>{t.d(e,{Z:()=>l});const l=t.p+"assets/images/tools-91886b41cefeef7cc35b5cd21c43d2ba.webp"}}]); \ No newline at end of file diff --git a/assets/js/87b8e9c4.408c1281.js b/assets/js/87b8e9c4.408c1281.js new file mode 100644 index 0000000..bd4cd5e --- /dev/null +++ b/assets/js/87b8e9c4.408c1281.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkfoomo=self.webpackChunkfoomo||[]).push([[2423],{3905:(A,e,t)=>{t.d(e,{Zo:()=>g,kt:()=>d});var l=t(7294);function n(A,e,t){return e in A?Object.defineProperty(A,e,{value:t,enumerable:!0,configurable:!0,writable:!0}):A[e]=t,A}function a(A,e){var t=Object.keys(A);if(Object.getOwnPropertySymbols){var l=Object.getOwnPropertySymbols(A);e&&(l=l.filter((function(e){return Object.getOwnPropertyDescriptor(A,e).enumerable}))),t.push.apply(t,l)}return t}function o(A){for(var e=1;e=0||(n[t]=A[t]);return n}(A,e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(A);for(l=0;l=0||Object.prototype.propertyIsEnumerable.call(A,t)&&(n[t]=A[t])}return n}var r=l.createContext({}),s=function(A){var e=l.useContext(r),t=e;return A&&(t="function"==typeof A?A(e):o(o({},e),A)),t},g=function(A){var e=s(A.components);return l.createElement(r.Provider,{value:e},A.children)},u="mdxType",c={inlineCode:"code",wrapper:function(A){var e=A.children;return l.createElement(l.Fragment,{},e)}},p=l.forwardRef((function(A,e){var t=A.components,n=A.mdxType,a=A.originalType,r=A.parentName,g=i(A,["components","mdxType","originalType","parentName"]),u=s(t),p=n,d=u["".concat(r,".").concat(p)]||u[p]||c[p]||a;return t?l.createElement(d,o(o({ref:e},g),{},{components:t})):l.createElement(d,o({ref:e},g))}));function d(A,e){var t=arguments,n=e&&e.mdxType;if("string"==typeof A||n){var a=t.length,o=new Array(a);o[0]=p;var i={};for(var r in e)hasOwnProperty.call(e,r)&&(i[r]=e[r]);i.originalType=A,i[u]="string"==typeof A?A:n,o[1]=i;for(var s=2;s{t.r(e),t.d(e,{assets:()=>r,contentTitle:()=>o,default:()=>c,frontMatter:()=>a,metadata:()=>i,toc:()=>s});var l=t(7462),n=(t(7294),t(3905));const a={slug:"why-bundle-size-is-important",authors:["nicola"],tags:["javascript","bundle","bundle size"]},o="Why bundle size is important?",i={permalink:"/blog/why-bundle-size-is-important",editUrl:"https://github.com/foomo/foomo-docs/tree/main/foomo/blog/2022-03-17-why-bundle-size-is-important/index.mdx",source:"@site/blog/2022-03-17-why-bundle-size-is-important/index.mdx",title:"Why bundle size is important?",description:"Intro",date:"2022-03-17T00:00:00.000Z",formattedDate:"March 17, 2022",tags:[{label:"javascript",permalink:"/blog/tags/javascript"},{label:"bundle",permalink:"/blog/tags/bundle"},{label:"bundle size",permalink:"/blog/tags/bundle-size"}],hasTruncateMarker:!1,authors:[{name:"Nicola Turcato",title:"Memelord brother",url:"https://github.com/nicolaturcato",imageURL:"https://github.com/nicolaturcato.png",key:"nicola"}],frontMatter:{slug:"why-bundle-size-is-important",authors:["nicola"],tags:["javascript","bundle","bundle size"]},prevItem:{title:"Accuracy of decimal computations",permalink:"/blog/accuracy-of-decimal-computations"},nextItem:{title:"Prometheus Is Out Of Memory. Again.",permalink:"/blog/prometheus-cardinality-issues"}},r={authorsImageUrls:[void 0]},s=[{value:"Intro",id:"intro",level:2},{value:"Nobody likes waiting\u2026",id:"nobody-likes-waiting",level:2},{value:"What is a "bundle"?",id:"what-is-a-bundle",level:2},{value:"Performance implications",id:"performance-implications",level:2},{value:"What is the recommended bundle size?",id:"what-is-the-recommended-bundle-size",level:2},{value:"What do we do then?",id:"what-do-we-do-then",level:2},{value:"How to start decreasing the bundle size?",id:"how-to-start-decreasing-the-bundle-size",level:2},{value:"Breaking up the bundle...",id:"breaking-up-the-bundle",level:2},{value:"Which strategies can we adopt?",id:"which-strategies-can-we-adopt",level:2},{value:"Useful tools to help you reducing bundle size",id:"useful-tools-to-help-you-reducing-bundle-size",level:2},{value:"Conclusion",id:"conclusion",level:2}],g={toc:s},u="wrapper";function c(A){let{components:e,...a}=A;return(0,n.kt)(u,(0,l.Z)({},g,a,{components:e,mdxType:"MDXLayout"}),(0,n.kt)("h2",{id:"intro"},"Intro"),(0,n.kt)("p",null,"JavaScript is parsed, compiled and executed in the main thread of the browser. Which means that users have to wait for all of this before they can interact with the website."),(0,n.kt)("p",null,"Frontend performance optimization is critical because\xa0it accounts for around 80-90% of user response time (10-20% backend).\nSo when a user is waiting for a page to load, around 80-90% of the time is due to frontend related code and assets."),(0,n.kt)("h2",{id:"nobody-likes-waiting"},"Nobody likes waiting\u2026"),(0,n.kt)("p",null,"A\xa0study found\xa0that if a site takes longer than 4 seconds to load, up to 25% of users would abandon the site."),(0,n.kt)("p",null,"Sending large JavaScript payloads impacts the speed of your site significantly."),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Mazzarri",src:t(7017).Z,width:"410",height:"276"})),(0,n.kt)("h2",{id:"what-is-a-bundle"},'What is a "bundle"?'),(0,n.kt)("p",null,"Your frontend application needs a bunch of JS files to run. These files can be in the format of internal dependency like the JS files you have written yourself. But they can also be external dependencies you use to build your application."),(0,n.kt)("p",null,"JS bundling is\xa0an optimization technique to reduce the number of server requests for JavaScript files. Bundling accomplishes this by merging multiple JA files together into one file to reduce the number of page requests."),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Bundle Everywhere",src:t(6865).Z,width:"448",height:"245"})),(0,n.kt)("h2",{id:"performance-implications"},"Performance implications"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Time to transmit over the network"),": considering slow connections with some mobile devices, it's possible that your page will not be interactive until it loads.-> More bytes = longer download times"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"JS parse and compile time"),": more code you load, more the browser must parse.-> JS gets parsed and compiled on the main thread, when the main thread is busy, the page can't respond to user input"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"JS execution time"),": optimally you will only pack the code that you expect to execute. The more code you want to execute the longer it will take. It's possible that your page won't be interactive until some of this completes.-> JS is also executed on the main thread, if your page runs a lot of code before it's really needed, that also delays your Time to Interactive"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Memory consumption"),": everything fills up the space -> code itself, runtime variables, DOM elements created, etc.-> Pages appear slow when it consumes a lot of memory. Memory leaks can cause your page to freeze up completely!!")),(0,n.kt)("h2",{id:"what-is-the-recommended-bundle-size"},"What is the recommended bundle size?"),(0,n.kt)("p",null,"AS SMALL AS POSSIBLE! I experienced that is not really possible to give a precise answer because each application is different. Generally you want the resources on initial load to be as small as possible, so that you decrease the initial load time, and then load more resources as needed on the fly."),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Mr Chao",src:t(1262).Z,width:"406",height:"250"})),(0,n.kt)("h2",{id:"what-do-we-do-then"},"What do we do then?"),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Meh",src:t(4812).Z,width:"539",height:"399"})),(0,n.kt)("h2",{id:"how-to-start-decreasing-the-bundle-size"},"How to start decreasing the bundle size?"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Measure"),": first of all you want to measure. The first step is to use Lighthouse and try to understand the results. It will give you a couple of interesting metrics and some tips. Time to interactive (TTI)\xa0is a good reflection of your bundle size because your bundle needs to be evaluated entirely before a user can interact with your web app."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Analyze"),": Consists on analyzing the bundle in order to detect critical chunks. A useful tool is Webpack Bundle Analyzer.")),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Stonks",src:t(3400).Z,width:"309",height:"229"})),(0,n.kt)("h2",{id:"breaking-up-the-bundle"},"Breaking up the bundle..."),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Monitor network requests"),": These happens between our FCP and TTI. As the initial request for data often occurs when our components initially mount."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Reduce the total dom nodes"),": the less the page needs to render, the less time it takes."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Moving work off the main thread"),": By moving heavy computations to a web worker, the computation will be run on a separate thread, and not block the actual rendering of the page"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Caching"),": Even if not useful for users on first page landing, caching data, bundles, and assets can make subsequent visits way fast")),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Breaking Bad",src:t(1278).Z,width:"299",height:"230"})),(0,n.kt)("h2",{id:"which-strategies-can-we-adopt"},"Which strategies can we adopt?"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Minification and Dead Code Elimination"),": These processes are often summed up as\xa0minifying\xa0or\xa0uglifying."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Tree shaking"),": Tree shaking is dead code elimination on a project or library. Always try to use deps which support \u201ctree shaking\u201d, Bundlephobia could be your friend in this case."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Code Splitting and Lazy Loading"),": Code splitting consists on taking a collection of modules and remove them from the main JS bundle. Lazy loading means we can load this newly created bundle later on."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Replace/rewrite large dependencies"),": Consider replacing or rewriting libraries that are large in size where you might not need all of its functionalities (Moment.js for example)."),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Feature module import"),": Check to see if you are using only a feature module of the library that can be imported alone without importing the whole library (Lodash for example).")),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Strategy",src:t(2235).Z,width:"307",height:"204"})),(0,n.kt)("h2",{id:"useful-tools-to-help-you-reducing-bundle-size"},"Useful tools to help you reducing bundle size"),(0,n.kt)("ul",null,(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Lighthouse"),": automated tool for improving the performance, quality, and correctness of your web apps"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Bundlephobia"),": Bundlephobia\xa0helps you find the performance impact of npm packages"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"Webpack Bundle Analyzer"),": analyzes your bundle"),(0,n.kt)("li",{parentName:"ul"},(0,n.kt)("strong",{parentName:"li"},"VS Code"),": Import Cost plugin -> Display import/require package size in the editor")),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"Tools",src:t(4941).Z,width:"345",height:"255"})),(0,n.kt)("h2",{id:"conclusion"},"Conclusion"),(0,n.kt)("p",null,"Performance cannot be stripped down to a single metric such as bundle size. It would be great!\nUnfortunately there is no single place to measure all of them.I think metrics like the\xa0Core Web Vitals and a general look at bundle size should be considered as a starting point.\nYou will cry... A lot... But don\u2019t give up!"),(0,n.kt)("p",null,(0,n.kt)("img",{alt:"The End",src:t(6899).Z,width:"358",height:"260"})))}c.isMDXComponent=!0},1278:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},6865:(A,e,t)=>{t.d(e,{Z:()=>l});const l=t.p+"assets/images/bundle_everywhere-12122e37ab5ca6bcdd61061c4dc32fd4.webp"},7017:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},4812:(A,e,t)=>{t.d(e,{Z:()=>l});const l=t.p+"assets/images/meh-3b42aead7d2246467fd0101518403734.webp"},1262:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},3400:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},2235:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},6899:(A,e,t)=>{t.d(e,{Z:()=>l});const l=""},4941:(A,e,t)=>{t.d(e,{Z:()=>l});const l=t.p+"assets/images/tools-91886b41cefeef7cc35b5cd21c43d2ba.webp"}}]); \ No newline at end of file diff --git a/assets/js/90c7e6e2.80970463.js b/assets/js/90c7e6e2.80970463.js new file mode 100644 index 0000000..d120b95 --- /dev/null +++ b/assets/js/90c7e6e2.80970463.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkfoomo=self.webpackChunkfoomo||[]).push([[1284],{9473:o=>{o.exports=JSON.parse('{"permalink":"/blog/tags/decimal-accuracy","page":1,"postsPerPage":10,"totalPages":1,"totalCount":1,"blogDescription":"Blog","blogTitle":"Blog"}')}}]); \ No newline at end of file diff --git a/assets/js/a20900f1.d8e2e996.js b/assets/js/a20900f1.d8e2e996.js new file mode 100644 index 0000000..aa80d0c --- /dev/null +++ b/assets/js/a20900f1.d8e2e996.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkfoomo=self.webpackChunkfoomo||[]).push([[6618],{2067:e=>{e.exports=JSON.parse('{"label":"currency","permalink":"/blog/tags/currency","allTagsPath":"/blog/tags","count":1}')}}]); \ No newline at end of file diff --git a/assets/js/a7023ddc.9e2b53c7.js b/assets/js/a7023ddc.9e2b53c7.js new file mode 100644 index 0000000..c7e3e3b --- /dev/null +++ b/assets/js/a7023ddc.9e2b53c7.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkfoomo=self.webpackChunkfoomo||[]).push([[1713],{3457:l=>{l.exports=JSON.parse('[{"label":"golang","permalink":"/blog/tags/golang","count":1},{"label":"currency","permalink":"/blog/tags/currency","count":1},{"label":"decimal accuracy","permalink":"/blog/tags/decimal-accuracy","count":1},{"label":"javascript","permalink":"/blog/tags/javascript","count":1},{"label":"bundle","permalink":"/blog/tags/bundle","count":1},{"label":"bundle size","permalink":"/blog/tags/bundle-size","count":1},{"label":"prometheus","permalink":"/blog/tags/prometheus","count":1},{"label":"cardinality","permalink":"/blog/tags/cardinality","count":1},{"label":"devops","permalink":"/blog/tags/devops","count":1},{"label":"ops","permalink":"/blog/tags/ops","count":1},{"label":"k8s","permalink":"/blog/tags/k-8-s","count":1},{"label":"oom","permalink":"/blog/tags/oom","count":1},{"label":"memory","permalink":"/blog/tags/memory","count":1},{"label":"search","permalink":"/blog/tags/search","count":1},{"label":"search-engine","permalink":"/blog/tags/search-engine","count":1},{"label":"backend","permalink":"/blog/tags/backend","count":2},{"label":"go","permalink":"/blog/tags/go","count":2},{"label":"frontend","permalink":"/blog/tags/frontend","count":1},{"label":"performance","permalink":"/blog/tags/performance","count":1},{"label":"debugging","permalink":"/blog/tags/debugging","count":1},{"label":"foomo","permalink":"/blog/tags/foomo","count":1}]')}}]); \ No newline at end of file diff --git a/assets/js/a7023ddc.a066be17.js b/assets/js/a7023ddc.a066be17.js deleted file mode 100644 index 0ef2154..0000000 --- a/assets/js/a7023ddc.a066be17.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkfoomo=self.webpackChunkfoomo||[]).push([[1713],{3457:l=>{l.exports=JSON.parse('[{"label":"javascript","permalink":"/blog/tags/javascript","count":1},{"label":"bundle","permalink":"/blog/tags/bundle","count":1},{"label":"bundle size","permalink":"/blog/tags/bundle-size","count":1},{"label":"prometheus","permalink":"/blog/tags/prometheus","count":1},{"label":"cardinality","permalink":"/blog/tags/cardinality","count":1},{"label":"devops","permalink":"/blog/tags/devops","count":1},{"label":"ops","permalink":"/blog/tags/ops","count":1},{"label":"k8s","permalink":"/blog/tags/k-8-s","count":1},{"label":"oom","permalink":"/blog/tags/oom","count":1},{"label":"memory","permalink":"/blog/tags/memory","count":1},{"label":"search","permalink":"/blog/tags/search","count":1},{"label":"search-engine","permalink":"/blog/tags/search-engine","count":1},{"label":"backend","permalink":"/blog/tags/backend","count":2},{"label":"go","permalink":"/blog/tags/go","count":2},{"label":"frontend","permalink":"/blog/tags/frontend","count":1},{"label":"performance","permalink":"/blog/tags/performance","count":1},{"label":"debugging","permalink":"/blog/tags/debugging","count":1},{"label":"foomo","permalink":"/blog/tags/foomo","count":1}]')}}]); \ No newline at end of file diff --git a/assets/js/b2b675dd.6a893e1a.js b/assets/js/b2b675dd.14f7daa6.js similarity index 73% rename from assets/js/b2b675dd.6a893e1a.js rename to assets/js/b2b675dd.14f7daa6.js index fc22bb0..57eff17 100644 --- a/assets/js/b2b675dd.6a893e1a.js +++ b/assets/js/b2b675dd.14f7daa6.js @@ -1 +1 @@ -"use strict";(self.webpackChunkfoomo=self.webpackChunkfoomo||[]).push([[533],{8017:o=>{o.exports=JSON.parse('{"permalink":"/blog","page":1,"postsPerPage":10,"totalPages":1,"totalCount":6,"blogDescription":"Blog","blogTitle":"Blog"}')}}]); \ No newline at end of file +"use strict";(self.webpackChunkfoomo=self.webpackChunkfoomo||[]).push([[533],{8017:o=>{o.exports=JSON.parse('{"permalink":"/blog","page":1,"postsPerPage":10,"totalPages":1,"totalCount":7,"blogDescription":"Blog","blogTitle":"Blog"}')}}]); \ No newline at end of file diff --git a/assets/js/b2f554cd.a67ccd00.js b/assets/js/b2f554cd.a67ccd00.js deleted file mode 100644 index f19b030..0000000 --- a/assets/js/b2f554cd.a67ccd00.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkfoomo=self.webpackChunkfoomo||[]).push([[1477],{10:e=>{e.exports=JSON.parse('{"blogPosts":[{"id":"why-bundle-size-is-important","metadata":{"permalink":"/blog/why-bundle-size-is-important","editUrl":"https://github.com/foomo/foomo-docs/tree/main/foomo/blog/2022-03-17-why-bundle-size-is-important/index.mdx","source":"@site/blog/2022-03-17-why-bundle-size-is-important/index.mdx","title":"Why bundle size is important?","description":"Intro","date":"2022-03-17T00:00:00.000Z","formattedDate":"March 17, 2022","tags":[{"label":"javascript","permalink":"/blog/tags/javascript"},{"label":"bundle","permalink":"/blog/tags/bundle"},{"label":"bundle size","permalink":"/blog/tags/bundle-size"}],"hasTruncateMarker":false,"authors":[{"name":"Nicola Turcato","title":"Memelord brother","url":"https://github.com/nicolaturcato","imageURL":"https://github.com/nicolaturcato.png","key":"nicola"}],"frontMatter":{"slug":"why-bundle-size-is-important","authors":["nicola"],"tags":["javascript","bundle","bundle size"]},"nextItem":{"title":"Prometheus Is Out Of Memory. Again.","permalink":"/blog/prometheus-cardinality-issues"}},"content":"## Intro\\n\\nJavaScript is parsed, compiled and executed in the main thread of the browser. Which means that users have to wait for all of this before they can interact with the website.\\n\\nFrontend performance optimization is critical because\xa0it accounts for around 80-90% of user response time (10-20% backend).\\nSo when a user is waiting for a page to load, around 80-90% of the time is due to frontend related code and assets.\\n\\n## Nobody likes waiting\u2026\\n\\nA\xa0study found\xa0that if a site takes longer than 4 seconds to load, up to 25% of users would abandon the site.\\n\\nSending large JavaScript payloads impacts the speed of your site significantly.\\n\\n![Mazzarri](mazzarri.webp)\\n\\n## What is a \\"bundle\\"?\\n\\nYour frontend application needs a bunch of JS files to run. These files can be in the format of internal dependency like the JS files you have written yourself. But they can also be external dependencies you use to build your application.\\n\\nJS bundling is\xa0an optimization technique to reduce the number of server requests for JavaScript files. Bundling accomplishes this by merging multiple JA files together into one file to reduce the number of page requests.\\n\\n![Bundle Everywhere](bundle_everywhere.webp)\\n\\n## Performance implications\\n\\n- **Time to transmit over the network**: considering slow connections with some mobile devices, it\'s possible that your page will not be interactive until it loads.-> More bytes = longer download times\\n- **JS parse and compile time**: more code you load, more the browser must parse.-> JS gets parsed and compiled on the main thread, when the main thread is busy, the page can\'t respond to user input\\n- **JS execution time**: optimally you will only pack the code that you expect to execute. The more code you want to execute the longer it will take. It\'s possible that your page won\'t be interactive until some of this completes.-> JS is also executed on the main thread, if your page runs a lot of code before it\'s really needed, that also delays your Time to Interactive\\n- **Memory consumption**: everything fills up the space -> code itself, runtime variables, DOM elements created, etc.-> Pages appear slow when it consumes a lot of memory. Memory leaks can cause your page to freeze up completely!!\\n\\n## What is the recommended bundle size?\\n\\nAS SMALL AS POSSIBLE! I experienced that is not really possible to give a precise answer because each application is different. Generally you want the resources on initial load to be as small as possible, so that you decrease the initial load time, and then load more resources as needed on the fly.\\n\\n![Mr Chao](mr_chao.webp)\\n\\n## What do we do then?\\n\\n![Meh](meh.webp)\\n\\n## How to start decreasing the bundle size?\\n\\n- **Measure**: first of all you want to measure. The first step is to use Lighthouse and try to understand the results. It will give you a couple of interesting metrics and some tips. Time to interactive (TTI)\xa0is a good reflection of your bundle size because your bundle needs to be evaluated entirely before a user can interact with your web app.\\n- **Analyze**: Consists on analyzing the bundle in order to detect critical chunks. A useful tool is Webpack Bundle Analyzer.\\n\\n![Stonks](stonks.webp)\\n\\n## Breaking up the bundle...\\n\\n- **Monitor network requests**: These happens between our FCP and TTI. As the initial request for data often occurs when our components initially mount.\\n- **Reduce the total dom nodes**: the less the page needs to render, the less time it takes.\\n- **Moving work off the main thread**: By moving heavy computations to a web worker, the computation will be run on a separate thread, and not block the actual rendering of the page\\n- **Caching**: Even if not useful for users on first page landing, caching data, bundles, and assets can make subsequent visits way fast\\n\\n![Breaking Bad](breaking_bad.webp)\\n\\n## Which strategies can we adopt?\\n\\n- **Minification and Dead Code Elimination**: These processes are often summed up as\xa0minifying\xa0or\xa0uglifying.\\n- **Tree shaking**: Tree shaking is dead code elimination on a project or library. Always try to use deps which support \u201ctree shaking\u201d, Bundlephobia could be your friend in this case.\\n- **Code Splitting and Lazy Loading**: Code splitting consists on taking a collection of modules and remove them from the main JS bundle. Lazy loading means we can load this newly created bundle later on.\\n- **Replace/rewrite large dependencies**: Consider replacing or rewriting libraries that are large in size where you might not need all of its functionalities (Moment.js for example).\\n- **Feature module import**: Check to see if you are using only a feature module of the library that can be imported alone without importing the whole library (Lodash for example).\\n\\n![Strategy](strategy.webp)\\n\\n## Useful tools to help you reducing bundle size\\n\\n- **Lighthouse**: automated tool for improving the performance, quality, and correctness of your web apps\\n- **Bundlephobia**: Bundlephobia\xa0helps you find the performance impact of npm packages\\n- **Webpack Bundle Analyzer**: analyzes your bundle\\n- **VS Code**: Import Cost plugin -> Display import/require package size in the editor\\n\\n![Tools](tools.webp)\\n\\n## Conclusion\\n\\nPerformance cannot be stripped down to a single metric such as bundle size. It would be great!\\nUnfortunately there is no single place to measure all of them.I think metrics like the\xa0Core Web Vitals and a general look at bundle size should be considered as a starting point.\\nYou will cry... A lot... But don\u2019t give up!\\n\\n![The End](the_end.webp)"},{"id":"prometheus-cardinality-issues","metadata":{"permalink":"/blog/prometheus-cardinality-issues","editUrl":"https://github.com/foomo/foomo-docs/tree/main/foomo/blog/2022-01-25-prometheus-cardinality-issues/index.mdx","source":"@site/blog/2022-01-25-prometheus-cardinality-issues/index.mdx","title":"Prometheus Is Out Of Memory. Again.","description":"The Annoyance","date":"2022-01-25T00:00:00.000Z","formattedDate":"January 25, 2022","tags":[{"label":"prometheus","permalink":"/blog/tags/prometheus"},{"label":"cardinality","permalink":"/blog/tags/cardinality"},{"label":"devops","permalink":"/blog/tags/devops"},{"label":"ops","permalink":"/blog/tags/ops"},{"label":"k8s","permalink":"/blog/tags/k-8-s"},{"label":"oom","permalink":"/blog/tags/oom"},{"label":"memory","permalink":"/blog/tags/memory"}],"hasTruncateMarker":false,"authors":[{"name":"Stefan Martinov","title":"Memelord","url":"https://github.com/smartinov","imageURL":"https://github.com/smartinov.png","key":"smartinov"}],"frontMatter":{"slug":"prometheus-cardinality-issues","authors":["smartinov"],"tags":["prometheus","cardinality","devops","ops","k8s","oom","memory"]},"prevItem":{"title":"Why bundle size is important?","permalink":"/blog/why-bundle-size-is-important"},"nextItem":{"title":"The never ending search a search engine 2022-01 edition","permalink":"/blog/searching-for-search-engines"}},"content":"## The Annoyance\\n\\nSo, we\'ve all been there. You go to your trusty grafana, search for some sweet metrics that you implemented and WHAM!\\nPrometheus returns us a 503, a trusty way of saying I\'m not ready, and I\'m probably going to die soon.\\nAnd since we\'re running in kubernetes I\'m going to die soon, again and again.\\nAnd you\'re getting reports from your colleagues that prometheus is not responding.\\nAnd you can\'t ignore them anymore.\\n\\n![Bummer.](bummer.webp)\\n\\n\\n## The Problem\\n\\nAll right, lets check what\'s happening to the little guy.\\n\\n```bash\\nkubectl get pods -n monitoring\\n\\nprometheus-prometheus-kube-prometheus-prometheus-0 1/2 Running 4 5m\\n```\\n\\nIt seems like it\'s stuck in the running state, where the container is not yet ready.\\nLet\'s describe the deployment, to check out what\'s happening.\\n\\n```yaml\\n State: Running \u2502\\n Started: Wed, 12 Jan 2022 15:12:49 +0100 \u2502\\n Last State: Terminated \u2502\\n Reason: OOMKilled \u2502\\n Exit Code: 137 \u2502\\n Started: Tue, 11 Jan 2022 17:14:41 +0100 \u2502\\n Finished: Wed, 12 Jan 2022 15:12:47 +0100 \u2502\\n```\\n\\nSo we see that the prometheus is in a running state waiting for the readiness probe to trigger, probably working on recovering from Write Ahead Log (WAL).\\nThis could be an issue where prometheus is recovering from an error, or a restart and does not have enough memory to write everything in the WAL.\\nWe could be running into an issue where we set the request/limits memory lower than the prometheus requires, and the kube scheduler keeps killing prometheus for wanting more memory.\\n\\nFor this case, we could give it more memory to work to see if it recovers. We should also analyze why the prometheus WAL is getting clogged up.\\n\\nIn essence, we want to check what has changed so that we suddenly have a high memory spike in our sweet, sweet environment.\\n\\n## The Source\\n\\n![Cardinality](cardinality.webp)\\n\\nA lot of prometheus issues revolve around cardinality.\\nMemory spikes that break your deployment? Cardinality.\\nPrometheus dragging its feet like it\'s Monday after the log4j (the second one ofc) zero day security breach? Cardinality.\\nNot getting that raise since you worked hard the past 16 years without wavering? You bet your ass it\'s cardinality.\\nSo, as you can see much of life\'s problems can be accredited to cardinality.\\n\\nIn short cardinality of your metrics is the combination of all label values per metric.\\nFor example, if our metric ```http_request_total``` had a label response code, and let\'s say we support 8 status codes, our cardinality starts off at 8.\\nFor good measure we want to record the HTTP verb for the request. We support ``GET POST PUT HEAD`` which would put the cardinality to 4\\\\*8=**32**.\\nNow, if someone adds a URL to the metric label (**!!VERY BAD IDEA!!**, but bare with me now) and we have 2 active pages, we\'d have a cardinality of 2\\\\*4\\\\*8=**64**.\\nBut, imagine someone starts scraping your website for potential vulnerabilities. Imagine all the URLs that will appear, most likely only once.\\n\\n```text\\nmywebsite.com/admin.php\\nmywebsite.com/wp/admin.php\\nmywebsite.com/?utm_source=GUID\\n...\\n```\\n\\nThis would blow up our cardinality to kingdom come. Like you will be out of memory faster than \\"[a new super awesome Javascript gamechanger framework](https://www.reddit.com/r/ProgrammerHumor/comments/a483yz/those_javascript_devs/)\\" is born.\\nOr to quote user [naveen17797](https://www.reddit.com/user/naveen17797/) _Scientists predict the number of js frameworks may exceed human population by 2020,at that point of time random string generators will be used to name those frameworks._\\n\\nThe point to this story is, be very mindful of how you use labels and cardinality in prometheus, since that will indeed have great impact on your prometheus performance.\\n\\n## The Solution\\n\\nSince this has never happened to me (never-ever) I found the following solution to be handy.\\nSince we can\'t get prometheus up and running to utilize PromQL to detect the potential issues, we have to find another way to detect high cardinality.\\nTherefore, we might want to get our hands dirty with some ```kubectl exec -it -n monitoring pods/prometheus-prometheus-kube-prometheus-prometheus-0 -- sh```, and run the prometheus ``tsdb`` analysis too.\\n```bash\\n/prometheus $ promtool tsdb analyze .\\n```\\n\\nWhich produced the result.\\n\\n```text\\n> Block ID: 01FT8E8YY4THHZ2S7C3G04GJMG\\n> Duration: 1h59m59.997s\\n> Series: 564171\\n> Label names: 285\\n> Postings (unique label pairs): 21139\\n> Postings entries (total label pairs): 6423664\\n>\\n> ...\\n>\\n> Highest cardinality metric names:\\n> 11340 haproxy_server_http_responses_total\\n> ...\\n```\\n\\nWe see the potential issue here, where the ``haproxy_server_http_responses_total`` metric is having a super-high cardinality which is growing.\\nWe need to deal with it, so that our prometheus instance can breathe again. In this particular case, the solution was updating the haproxy.\\n\\n... or burn it, up to you.\\n\\n![Flame Thrower](flame-thrower.webp)\\n\\n## The Further Reading\\n\\n1. [WAL Definition](https://github.com/prometheus/prometheus/blob/main/tsdb/docs/format/wal.md)\\n2. [WAL & Checkpoints](https://ganeshvernekar.com/blog/prometheus-tsdb-wal-and-checkpoint/)\\n3. [Using TSDB](https://www.robustperception.io/using-tsdb-analyze-to-investigate-churn-and-cardinality)\\n4. [Biggest Metrics](https://www.robustperception.io/which-are-my-biggest-metrics)\\n5. [Cardinality](https://www.robustperception.io/cardinality-is-key)"},{"id":"searching-for-search-engines","metadata":{"permalink":"/blog/searching-for-search-engines","editUrl":"https://github.com/foomo/foomo-docs/tree/main/foomo/blog/2022-01-20-searching-for-search-engines.mdx","source":"@site/blog/2022-01-20-searching-for-search-engines.mdx","title":"The never ending search a search engine 2022-01 edition","description":"While building this website and integrating https://docsearch.algolia.com and evaluating another solution by a large company in parallel I could not help to search github and the web for the current state of search engines and search related services.","date":"2022-01-20T00:00:00.000Z","formattedDate":"January 20, 2022","tags":[{"label":"search","permalink":"/blog/tags/search"},{"label":"search-engine","permalink":"/blog/tags/search-engine"},{"label":"backend","permalink":"/blog/tags/backend"},{"label":"go","permalink":"/blog/tags/go"}],"hasTruncateMarker":false,"authors":[{"name":"Jan Halfar","title":"foomo maintainer","url":"https://github.com/janhalfar","imageURL":"https://github.com/janhalfar.png","key":"jan"}],"frontMatter":{"slug":"searching-for-search-engines","authors":["jan"],"tags":["search","search-engine","backend","go"]},"prevItem":{"title":"Prometheus Is Out Of Memory. Again.","permalink":"/blog/prometheus-cardinality-issues"},"nextItem":{"title":"Impact of 3rd party scripts on performance","permalink":"/blog/impact-of-3rd-party-scripts-on-performance"}},"content":"While building this website and integrating https://docsearch.algolia.com and evaluating another solution by a large company in parallel I could not help to search github and the web for the current state of search engines and search related services.\\n\\nSince I had done the same thing about a year ago, I was surprised to see how quickly things are moving atm.\\n\\n## Algolia\\n\\nI was blown away by the quality of https://www.algolia.com and I wish it was open source, but I guess, we all have to make a living ;)\\n\\nTo see how awesome a web (search) interface can be check out https://www.lacoste.com/us/#query=red%20jackets%20for%20men \\n\\nApart from that the UI/UX of their backend tools is fantastic.\\n\\n## Elastic\\n\\nWhen it comes to https://www.elastic.com I am a bit nervous about the future of the licensing, despite the fact, that I understand their motivation. At the same time the https://opensearch.org does not seem to be an ampty threat.\\n\\n\\n## typesense.org\\n\\nI do not know, who was hiding under a rock, but I had not seen https://typesense.org before and they certainly have a bold claim: ___\\"The Open Source Algolia Alternative\\" / \\"The Easier To Use ElasticSearch Alternative\\"___ \\n\\nWhen looking at https://github.com/typesense/typesense/graphs/contributors it seems, that Kishore Nallan has been working on this for a while. Unfourtunately I do not really see a lot of external contributions, C++ does not seem to attract a lot of contribution.\\n\\n## MeiliSearch\\n\\nThis Rust project https://www.meilisearch.com/ seems to be picking up speed and is definetly on the test short list. It is a fresh codebase with siginficant open source contributions and certainly will attract new developers with Rust and a modern architecture.\\n\\n## Go eco system\\n\\nObviously we are very interested in Go powered software and there are a few notable projects. ATM I do not see anything elastic or algolia like, that would be really mature. \\n\\n\\n### bleve / bluge\\n\\n[Marty Schoch](https://github.com/mschoch) seems to be the man when it comes down to text indexing libraries in written in Go and bluge seems to be THE library, that is solid and modern, when implementing text search in your Go application.\\n\\nhttps://github.com/blevesearch/bleve\\nhttps://github.com/blugelabs/bluge // next iteration of bleve\\n\\n#### projects using bluge\\n\\nAll bleeding edge afaik atm - but definitely good places to look at bluge usage\\n\\nhttps://github.com/prabhatsharma/zinc \\nhttps://github.com/mosuka/phalanx\\n\\n### Look ma I made a vector database\\n\\nGotta take a look at this one - will report later\\n\\nhttps://github.com/semi-technologies/weaviate"},{"id":"impact-of-3rd-party-scripts-on-performance","metadata":{"permalink":"/blog/impact-of-3rd-party-scripts-on-performance","editUrl":"https://github.com/foomo/foomo-docs/tree/main/foomo/blog/2022-01-20-exploring-partytown/index.mdx","source":"@site/blog/2022-01-20-exploring-partytown/index.mdx","title":"Impact of 3rd party scripts on performance","description":"Issue with performance","date":"2022-01-20T00:00:00.000Z","formattedDate":"January 20, 2022","tags":[{"label":"frontend","permalink":"/blog/tags/frontend"},{"label":"performance","permalink":"/blog/tags/performance"}],"hasTruncateMarker":false,"authors":[{"name":"Marko Trebi\u017ean","title":"Frontend Dev","url":"https://github.com/themre","imageURL":"https://github.com/themre.png","key":"marko"}],"frontMatter":{"slug":"impact-of-3rd-party-scripts-on-performance","authors":["marko"],"tags":["frontend","performance"]},"prevItem":{"title":"The never ending search a search engine 2022-01 edition","permalink":"/blog/searching-for-search-engines"},"nextItem":{"title":"debugging Go map races in k8s","permalink":"/blog/debugging-go-map-races-in-k8s"}},"content":"## Issue with performance\\n\\nWhen building an ecommerce site or an application where performance is a great deal for the users, you need to keep your application fast and responsive. Frontend developers have already many use-cases when the UI becomes laggy and this increases when 3rd party scripts are being included, such as Google Tag Manager or various Live chats (e.g. Intercom).\\n\\nThis does not only influences the users when using the site but also Lighthouse score gets lower which also influences page rankings. So the most naive and easy way for this is to defer loading of such scripts but when you need to get all the data from the start of the application, such tactic is not an option. So what else can we do?\\n\\n## Partytown to the rescue\\n\\nDevelopers at BuilderIO created an library [Partytown](https://github.com/BuilderIO/partytown) that would allow relocating resources from 3rd party scripts off the main thread.\\nWe won\'t dive into specifics how it works, because they explain it nicely on their GitHub page.\\n\\nIn our stack we use [Next.js](https://nextjs.org/) React framework and we will go through the basic steps that will allow us to include Partytown for Google Tag Manager.\\n\\n### Setup\\n\\nPartytown script needs to be located inside our application and live on the same domain. Since we\'re using monorepo structure, we need to copy this script across all our frontend application. For that we used CopyPlugin webpack plugin in our Next.js config file:\\n\\n```javascript\\nconfig.plugins.push(\\n ...\\n new CopyPlugin({\\n patterns: [\\n {\\n // we copy script from node_modules partytown package to `~partytown` folder in our package that serves static files\\n from: path.join(path.dirname(require.resolve(\'@builder.io/partytown\')), \'lib\'),\\n // paths for SSR and client side rendering differ\\n to: path.join(`${isServer ? \'..\' : \'.\'}/static/assets/`, \'~partytown\'),\\n },\\n ],\\n })\\n );\\n```\\n\\nPartytown\'s requirement is that it needs to know what script should it load into own web worker. For that we set script type to `text/partytown`. This will prevent script to load on initial load.\\n\\nInside `_document.tsx` we add this:\\n```javascript\\n\\n ...\\n // include Partytown and set custom path due to multiple frontends\\n \\n // tag 3rd party script with partytown type\\n

Awesome list of software

A list of commercial and open source software we use, because it is awesome.


desktop software

devtools

IDEs

creative tools

cli utilities

k8s / docker

backup

software as a service

productivity / collaboration

CMS

- - + + \ No newline at end of file diff --git a/blog.html b/blog.html index b4e4b55..f8c2621 100644 --- a/blog.html +++ b/blog.html @@ -9,12 +9,14 @@ - - + +
-

Nicola Turcato

Intro

JavaScript is parsed, compiled and executed in the main thread of the browser. Which means that users have to wait for all of this before they can interact with the website.

Frontend performance optimization is critical because it accounts for around 80-90% of user response time (10-20% backend). +

Intro

Calculating with money can be tricky if not taken proper precautions. Some might be tempted to use float representation for calculating with currency values. That is problematic because of possible rounding errors.

Finite accuracy of representation

Floating points are represented like this

Floating point representation

Not every number can be represented with a finite number of decimal places

0.01 —> 0.0000011001100110011…

Taking 17 places of the above results in 0.010000000000000001

Consider the following code snipet that shows the missing accuracy

func main() {

var n float64 = 0

for i := 0; i < 1000; i++ {
n += .01
}

fmt.Println(n)

}

Result: 9.999999999999831

Money computations

They can't be done with floating-point as it would inevitably lead to rounding errors.

Even the following packages are problematic:

github.com/shopspring/decimal

github.com/Rhymond/go-money

a := decimal.NewFromInt(2)
b := decimal.NewFromFloat(300.99)
c := a.Mul(b)
d := c.Div(decimal.NewFromInt(3))

Solution

Use Int by representing money in cents:

  • 10.99 -> 1099 (cents)
  • 10.9900 -> 109900 (4 digit tax)

Conclusio

Division is a problem!

1/3 - > 0.33333333… +Correct way: 0.33, 0.33, 0.34

When doing money calculations one should avoid division as it inevitably leads to loss of accuracy. +When dividing make sure to round to cent and deal with diffs.

Division by 10^k is ok till we are inside of the range of the data type.

Nicola Turcato

Intro

JavaScript is parsed, compiled and executed in the main thread of the browser. Which means that users have to wait for all of this before they can interact with the website.

Frontend performance optimization is critical because it accounts for around 80-90% of user response time (10-20% backend). So when a user is waiting for a page to load, around 80-90% of the time is due to frontend related code and assets.

Nobody likes waiting…

A study found that if a site takes longer than 4 seconds to load, up to 25% of users would abandon the site.

Sending large JavaScript payloads impacts the speed of your site significantly.

Mazzarri

What is a "bundle"?

Your frontend application needs a bunch of JS files to run. These files can be in the format of internal dependency like the JS files you have written yourself. But they can also be external dependencies you use to build your application.

JS bundling is an optimization technique to reduce the number of server requests for JavaScript files. Bundling accomplishes this by merging multiple JA files together into one file to reduce the number of page requests.

Bundle Everywhere

Performance implications

  • Time to transmit over the network: considering slow connections with some mobile devices, it's possible that your page will not be interactive until it loads.-> More bytes = longer download times
  • JS parse and compile time: more code you load, more the browser must parse.-> JS gets parsed and compiled on the main thread, when the main thread is busy, the page can't respond to user input
  • JS execution time: optimally you will only pack the code that you expect to execute. The more code you want to execute the longer it will take. It's possible that your page won't be interactive until some of this completes.-> JS is also executed on the main thread, if your page runs a lot of code before it's really needed, that also delays your Time to Interactive
  • Memory consumption: everything fills up the space -> code itself, runtime variables, DOM elements created, etc.-> Pages appear slow when it consumes a lot of memory. Memory leaks can cause your page to freeze up completely!!

AS SMALL AS POSSIBLE! I experienced that is not really possible to give a precise answer because each application is different. Generally you want the resources on initial load to be as small as possible, so that you decrease the initial load time, and then load more resources as needed on the fly.

Mr Chao

What do we do then?

Meh

How to start decreasing the bundle size?

  • Measure: first of all you want to measure. The first step is to use Lighthouse and try to understand the results. It will give you a couple of interesting metrics and some tips. Time to interactive (TTI) is a good reflection of your bundle size because your bundle needs to be evaluated entirely before a user can interact with your web app.
  • Analyze: Consists on analyzing the bundle in order to detect critical chunks. A useful tool is Webpack Bundle Analyzer.

Stonks

Breaking up the bundle...

  • Monitor network requests: These happens between our FCP and TTI. As the initial request for data often occurs when our components initially mount.
  • Reduce the total dom nodes: the less the page needs to render, the less time it takes.
  • Moving work off the main thread: By moving heavy computations to a web worker, the computation will be run on a separate thread, and not block the actual rendering of the page
  • Caching: Even if not useful for users on first page landing, caching data, bundles, and assets can make subsequent visits way fast

Breaking Bad

Which strategies can we adopt?

  • Minification and Dead Code Elimination: These processes are often summed up as minifying or uglifying.
  • Tree shaking: Tree shaking is dead code elimination on a project or library. Always try to use deps which support “tree shaking”, Bundlephobia could be your friend in this case.
  • Code Splitting and Lazy Loading: Code splitting consists on taking a collection of modules and remove them from the main JS bundle. Lazy loading means we can load this newly created bundle later on.
  • Replace/rewrite large dependencies: Consider replacing or rewriting libraries that are large in size where you might not need all of its functionalities (Moment.js for example).
  • Feature module import: Check to see if you are using only a feature module of the library that can be imported alone without importing the whole library (Lodash for example).

Strategy

Useful tools to help you reducing bundle size

  • Lighthouse: automated tool for improving the performance, quality, and correctness of your web apps
  • Bundlephobia: Bundlephobia helps you find the performance impact of npm packages
  • Webpack Bundle Analyzer: analyzes your bundle
  • VS Code: Import Cost plugin -> Display import/require package size in the editor

Tools

Conclusion

Performance cannot be stripped down to a single metric such as bundle size. It would be great! Unfortunately there is no single place to measure all of them.I think metrics like the Core Web Vitals and a general look at bundle size should be considered as a starting point. You will cry... A lot... But don’t give up!

The End

The Annoyance

So, we've all been there. You go to your trusty grafana, search for some sweet metrics that you implemented and WHAM! @@ -40,7 +42,7 @@ We need to deal with it, so that our prometheus instance can breathe again. In t https://github.com/blugelabs/bluge // next iteration of bleve

projects using bluge

All bleeding edge afaik atm - but definitely good places to look at bluge usage

https://github.com/prabhatsharma/zinc https://github.com/mosuka/phalanx

Look ma I made a vector database

Gotta take a look at this one - will report later

https://github.com/semi-technologies/weaviate

Marko Trebižan

Issue with performance

When building an ecommerce site or an application where performance is a great deal for the users, you need to keep your application fast and responsive. Frontend developers have already many use-cases when the UI becomes laggy and this increases when 3rd party scripts are being included, such as Google Tag Manager or various Live chats (e.g. Intercom).

This does not only influences the users when using the site but also Lighthouse score gets lower which also influences page rankings. So the most naive and easy way for this is to defer loading of such scripts but when you need to get all the data from the start of the application, such tactic is not an option. So what else can we do?

Partytown to the rescue

Developers at BuilderIO created an library Partytown that would allow relocating resources from 3rd party scripts off the main thread. We won't dive into specifics how it works, because they explain it nicely on their GitHub page.

In our stack we use Next.js React framework and we will go through the basic steps that will allow us to include Partytown for Google Tag Manager.

Setup

Partytown script needs to be located inside our application and live on the same domain. Since we're using monorepo structure, we need to copy this script across all our frontend application. For that we used CopyPlugin webpack plugin in our Next.js config file:

config.plugins.push(
...
new CopyPlugin({
patterns: [
{
// we copy script from node_modules partytown package to `~partytown` folder in our package that serves static files
from: path.join(path.dirname(require.resolve('@builder.io/partytown')), 'lib'),
// paths for SSR and client side rendering differ
to: path.join(`${isServer ? '..' : '.'}/static/assets/`, '~partytown'),
},
],
})
);

Partytown's requirement is that it needs to know what script should it load into own web worker. For that we set script type to text/partytown. This will prevent script to load on initial load.

Inside _document.tsx we add this:

<Head>
...
// include Partytown and set custom path due to multiple frontends
<Partytown lib={`${addTrailingSlash(this.props.basePath)}_next/static/assets/~partytown/`} debug={false} />
// tag 3rd party script with partytown type
<script type="text/partytown" src={`https://www.googletagmanager.com/gtm.js?id=${id}`} />
...
</Head>

Results

So now, does it work? We used one of our large Ecommerce sites to test the landing Lighthouse score.

This was before adding Partytown:

Lighthouse before Partytown

Here you can see 2 critical things: almost 1s of total blocking time (TBT) and 9s of time to interactive (TTI).

After we added Partytown, we got this:

Lighthouse after Partytown

Time to interactive went from 9s to 6.1s which is almost 33% improvement and total blocking time was reduce by more than 50%! We were more than impressed how easy it was to improve our performances.

Side note: Both screenshots were compressed using Squoosh App.

Next steps

After successful testing of Partytown for Google Tag Manager script, we are more than interested in trying it out on our other scripts. One important topic will be to test Partytown with other service-worker related libraries and how to use them together.

Jan Halfar

A few years ago we abandoned the previous version of https://www.foomo.org as we did not want to maintain the old wordpress installation and the project was only living in README.md in the repos living under https://www.github.com/foomo .

As things have grown over time we have decided to re-launch a website / cross project documentation.

So welcome back and enjoy the view to the past:

blast from the past

- - + + \ No newline at end of file diff --git a/blog/accuracy-of-decimal-computations.html b/blog/accuracy-of-decimal-computations.html new file mode 100644 index 0000000..4bf894f --- /dev/null +++ b/blog/accuracy-of-decimal-computations.html @@ -0,0 +1,23 @@ + + + + + +Accuracy of decimal computations | foomo project docs + + + + + + + + + +
+

Accuracy of decimal computations

Intro

Calculating with money can be tricky if not taken proper precautions. Some might be tempted to use float representation for calculating with currency values. That is problematic because of possible rounding errors.

Finite accuracy of representation

Floating points are represented like this

Floating point representation

Not every number can be represented with a finite number of decimal places

0.01 —> 0.0000011001100110011…

Taking 17 places of the above results in 0.010000000000000001

Consider the following code snipet that shows the missing accuracy

func main() {

var n float64 = 0

for i := 0; i < 1000; i++ {
n += .01
}

fmt.Println(n)

}

Result: 9.999999999999831

Money computations

They can't be done with floating-point as it would inevitably lead to rounding errors.

Even the following packages are problematic:

github.com/shopspring/decimal

github.com/Rhymond/go-money

a := decimal.NewFromInt(2)
b := decimal.NewFromFloat(300.99)
c := a.Mul(b)
d := c.Div(decimal.NewFromInt(3))

Solution

Use Int by representing money in cents:

  • 10.99 -> 1099 (cents)
  • 10.9900 -> 109900 (4 digit tax)

Conclusio

Division is a problem!

1/3 - > 0.33333333… +Correct way: 0.33, 0.33, 0.34

When doing money calculations one should avoid division as it inevitably leads to loss of accuracy. +When dividing make sure to round to cent and deal with diffs.

Division by 10^k is ok till we are inside of the range of the data type.

+ + + + \ No newline at end of file diff --git a/blog/archive.html b/blog/archive.html index 315f19a..dd7ee8b 100644 --- a/blog/archive.html +++ b/blog/archive.html @@ -9,13 +9,13 @@ - - + + - - +
+ + \ No newline at end of file diff --git a/blog/atom.xml b/blog/atom.xml index 4c4cfb4..681fa2c 100644 --- a/blog/atom.xml +++ b/blog/atom.xml @@ -2,11 +2,28 @@ https://www.foomo.org/blog foomo project docs Blog - 2022-03-17T00:00:00.000Z + 2023-03-06T00:00:00.000Z https://github.com/jpmonette/feed foomo project docs Blog https://www.foomo.org/img/favicon.ico + + <![CDATA[Accuracy of decimal computations]]> + https://www.foomo.org/blog/accuracy-of-decimal-computations + + 2023-03-06T00:00:00.000Z + + Intro

Calculating with money can be tricky if not taken proper precautions. Some might be tempted to use float representation for calculating with currency values. That is problematic because of possible rounding errors.

Finite accuracy of representation

Floating points are represented like this

Floating point representation

Not every number can be represented with a finite number of decimal places

0.01 —> 0.0000011001100110011…

Taking 17 places of the above results in 0.010000000000000001

Consider the following code snipet that shows the missing accuracy

func main() {

var n float64 = 0

for i := 0; i < 1000; i++ {
n += .01
}

fmt.Println(n)

}

Result: 9.999999999999831

Money computations

They can't be done with floating-point as it would inevitably lead to rounding errors.

Even the following packages are problematic:

github.com/shopspring/decimal

github.com/Rhymond/go-money

a := decimal.NewFromInt(2)
b := decimal.NewFromFloat(300.99)
c := a.Mul(b)
d := c.Div(decimal.NewFromInt(3))

Solution

Use Int by representing money in cents:

  • 10.99 -> 1099 (cents)
  • 10.9900 -> 109900 (4 digit tax)

Conclusio

Division is a problem!

1/3 - > 0.33333333… +Correct way: 0.33, 0.33, 0.34

When doing money calculations one should avoid division as it inevitably leads to loss of accuracy. +When dividing make sure to round to cent and deal with diffs.

Division by 10^k is ok till we are inside of the range of the data type.

]]>
+ + Patrick Buchner + https://github.com/smartinov + + + + +
<![CDATA[Why bundle size is important?]]> https://www.foomo.org/blog/why-bundle-size-is-important diff --git a/blog/debugging-go-map-races-in-k8s.html b/blog/debugging-go-map-races-in-k8s.html index ce6ba0f..5d2c5bc 100644 --- a/blog/debugging-go-map-races-in-k8s.html +++ b/blog/debugging-go-map-races-in-k8s.html @@ -9,13 +9,13 @@ - - + + - - +
+ + \ No newline at end of file diff --git a/blog/impact-of-3rd-party-scripts-on-performance.html b/blog/impact-of-3rd-party-scripts-on-performance.html index eab499c..2f22075 100644 --- a/blog/impact-of-3rd-party-scripts-on-performance.html +++ b/blog/impact-of-3rd-party-scripts-on-performance.html @@ -9,14 +9,14 @@ - - + +
-

Impact of 3rd party scripts on performance

Marko Trebižan

Issue with performance

When building an ecommerce site or an application where performance is a great deal for the users, you need to keep your application fast and responsive. Frontend developers have already many use-cases when the UI becomes laggy and this increases when 3rd party scripts are being included, such as Google Tag Manager or various Live chats (e.g. Intercom).

This does not only influences the users when using the site but also Lighthouse score gets lower which also influences page rankings. So the most naive and easy way for this is to defer loading of such scripts but when you need to get all the data from the start of the application, such tactic is not an option. So what else can we do?

Partytown to the rescue

Developers at BuilderIO created an library Partytown that would allow relocating resources from 3rd party scripts off the main thread. +

Impact of 3rd party scripts on performance

Marko Trebižan

Issue with performance

When building an ecommerce site or an application where performance is a great deal for the users, you need to keep your application fast and responsive. Frontend developers have already many use-cases when the UI becomes laggy and this increases when 3rd party scripts are being included, such as Google Tag Manager or various Live chats (e.g. Intercom).

This does not only influences the users when using the site but also Lighthouse score gets lower which also influences page rankings. So the most naive and easy way for this is to defer loading of such scripts but when you need to get all the data from the start of the application, such tactic is not an option. So what else can we do?

Partytown to the rescue

Developers at BuilderIO created an library Partytown that would allow relocating resources from 3rd party scripts off the main thread. We won't dive into specifics how it works, because they explain it nicely on their GitHub page.

In our stack we use Next.js React framework and we will go through the basic steps that will allow us to include Partytown for Google Tag Manager.

Setup

Partytown script needs to be located inside our application and live on the same domain. Since we're using monorepo structure, we need to copy this script across all our frontend application. For that we used CopyPlugin webpack plugin in our Next.js config file:

config.plugins.push(
...
new CopyPlugin({
patterns: [
{
// we copy script from node_modules partytown package to `~partytown` folder in our package that serves static files
from: path.join(path.dirname(require.resolve('@builder.io/partytown')), 'lib'),
// paths for SSR and client side rendering differ
to: path.join(`${isServer ? '..' : '.'}/static/assets/`, '~partytown'),
},
],
})
);

Partytown's requirement is that it needs to know what script should it load into own web worker. For that we set script type to text/partytown. This will prevent script to load on initial load.

Inside _document.tsx we add this:

<Head>
...
// include Partytown and set custom path due to multiple frontends
<Partytown lib={`${addTrailingSlash(this.props.basePath)}_next/static/assets/~partytown/`} debug={false} />
// tag 3rd party script with partytown type
<script type="text/partytown" src={`https://www.googletagmanager.com/gtm.js?id=${id}`} />
...
</Head>

Results

So now, does it work? We used one of our large Ecommerce sites to test the landing Lighthouse score.

This was before adding Partytown:

Lighthouse before Partytown

Here you can see 2 critical things: almost 1s of total blocking time (TBT) and 9s of time to interactive (TTI).

After we added Partytown, we got this:

Lighthouse after Partytown

Time to interactive went from 9s to 6.1s which is almost 33% improvement and total blocking time was reduce by more than 50%! We were more than impressed how easy it was to improve our performances.

Side note: Both screenshots were compressed using Squoosh App.

Next steps

After successful testing of Partytown for Google Tag Manager script, we are more than interested in trying it out on our other scripts. One important topic will be to test Partytown with other service-worker related libraries and how to use them together.

- - + + \ No newline at end of file diff --git a/blog/prometheus-cardinality-issues.html b/blog/prometheus-cardinality-issues.html index 41e25f9..f58276b 100644 --- a/blog/prometheus-cardinality-issues.html +++ b/blog/prometheus-cardinality-issues.html @@ -9,12 +9,12 @@ - - + +
-

Prometheus Is Out Of Memory. Again.

The Annoyance

So, we've all been there. You go to your trusty grafana, search for some sweet metrics that you implemented and WHAM! +

- - + + \ No newline at end of file diff --git a/blog/rss.xml b/blog/rss.xml index 4d8830a..dac7ae0 100644 --- a/blog/rss.xml +++ b/blog/rss.xml @@ -4,10 +4,23 @@ foomo project docs Blog https://www.foomo.org/blog foomo project docs Blog - Thu, 17 Mar 2022 00:00:00 GMT + Mon, 06 Mar 2023 00:00:00 GMT https://validator.w3.org/feed/docs/rss2.html https://github.com/jpmonette/feed en + + <![CDATA[Accuracy of decimal computations]]> + https://www.foomo.org/blog/accuracy-of-decimal-computations + https://www.foomo.org/blog/accuracy-of-decimal-computations + Mon, 06 Mar 2023 00:00:00 GMT + + Intro

Calculating with money can be tricky if not taken proper precautions. Some might be tempted to use float representation for calculating with currency values. That is problematic because of possible rounding errors.

Finite accuracy of representation

Floating points are represented like this

Floating point representation

Not every number can be represented with a finite number of decimal places

0.01 —> 0.0000011001100110011…

Taking 17 places of the above results in 0.010000000000000001

Consider the following code snipet that shows the missing accuracy

func main() {

var n float64 = 0

for i := 0; i < 1000; i++ {
n += .01
}

fmt.Println(n)

}

Result: 9.999999999999831

Money computations

They can't be done with floating-point as it would inevitably lead to rounding errors.

Even the following packages are problematic:

github.com/shopspring/decimal

github.com/Rhymond/go-money

a := decimal.NewFromInt(2)
b := decimal.NewFromFloat(300.99)
c := a.Mul(b)
d := c.Div(decimal.NewFromInt(3))

Solution

Use Int by representing money in cents:

  • 10.99 -> 1099 (cents)
  • 10.9900 -> 109900 (4 digit tax)

Conclusio

Division is a problem!

1/3 - > 0.33333333… +Correct way: 0.33, 0.33, 0.34

When doing money calculations one should avoid division as it inevitably leads to loss of accuracy. +When dividing make sure to round to cent and deal with diffs.

Division by 10^k is ok till we are inside of the range of the data type.

]]>
+ golang + currency + decimal accuracy +
<![CDATA[Why bundle size is important?]]> https://www.foomo.org/blog/why-bundle-size-is-important diff --git a/blog/searching-for-search-engines.html b/blog/searching-for-search-engines.html index f14e892..f18d6cc 100644 --- a/blog/searching-for-search-engines.html +++ b/blog/searching-for-search-engines.html @@ -9,15 +9,15 @@ - - + +
-

The never ending search a search engine 2022-01 edition

Jan Halfar

While building this website and integrating https://docsearch.algolia.com and evaluating another solution by a large company in parallel I could not help to search github and the web for the current state of search engines and search related services.

Since I had done the same thing about a year ago, I was surprised to see how quickly things are moving atm.

Algolia

I was blown away by the quality of https://www.algolia.com and I wish it was open source, but I guess, we all have to make a living ;)

To see how awesome a web (search) interface can be check out https://www.lacoste.com/us/#query=red%20jackets%20for%20men

Apart from that the UI/UX of their backend tools is fantastic.

Elastic

When it comes to https://www.elastic.com I am a bit nervous about the future of the licensing, despite the fact, that I understand their motivation. At the same time the https://opensearch.org does not seem to be an ampty threat.

typesense.org

I do not know, who was hiding under a rock, but I had not seen https://typesense.org before and they certainly have a bold claim: "The Open Source Algolia Alternative" / "The Easier To Use ElasticSearch Alternative"

When looking at https://github.com/typesense/typesense/graphs/contributors it seems, that Kishore Nallan has been working on this for a while. Unfourtunately I do not really see a lot of external contributions, C++ does not seem to attract a lot of contribution.

MeiliSearch

This Rust project https://www.meilisearch.com/ seems to be picking up speed and is definetly on the test short list. It is a fresh codebase with siginficant open source contributions and certainly will attract new developers with Rust and a modern architecture.

Go eco system

Obviously we are very interested in Go powered software and there are a few notable projects. ATM I do not see anything elastic or algolia like, that would be really mature.

bleve / bluge

Marty Schoch seems to be the man when it comes down to text indexing libraries in written in Go and bluge seems to be THE library, that is solid and modern, when implementing text search in your Go application.

https://github.com/blevesearch/bleve +

The never ending search a search engine 2022-01 edition

Jan Halfar

While building this website and integrating https://docsearch.algolia.com and evaluating another solution by a large company in parallel I could not help to search github and the web for the current state of search engines and search related services.

Since I had done the same thing about a year ago, I was surprised to see how quickly things are moving atm.

Algolia

I was blown away by the quality of https://www.algolia.com and I wish it was open source, but I guess, we all have to make a living ;)

To see how awesome a web (search) interface can be check out https://www.lacoste.com/us/#query=red%20jackets%20for%20men

Apart from that the UI/UX of their backend tools is fantastic.

Elastic

When it comes to https://www.elastic.com I am a bit nervous about the future of the licensing, despite the fact, that I understand their motivation. At the same time the https://opensearch.org does not seem to be an ampty threat.

typesense.org

I do not know, who was hiding under a rock, but I had not seen https://typesense.org before and they certainly have a bold claim: "The Open Source Algolia Alternative" / "The Easier To Use ElasticSearch Alternative"

When looking at https://github.com/typesense/typesense/graphs/contributors it seems, that Kishore Nallan has been working on this for a while. Unfourtunately I do not really see a lot of external contributions, C++ does not seem to attract a lot of contribution.

MeiliSearch

This Rust project https://www.meilisearch.com/ seems to be picking up speed and is definetly on the test short list. It is a fresh codebase with siginficant open source contributions and certainly will attract new developers with Rust and a modern architecture.

Go eco system

Obviously we are very interested in Go powered software and there are a few notable projects. ATM I do not see anything elastic or algolia like, that would be really mature.

bleve / bluge

Marty Schoch seems to be the man when it comes down to text indexing libraries in written in Go and bluge seems to be THE library, that is solid and modern, when implementing text search in your Go application.

https://github.com/blevesearch/bleve https://github.com/blugelabs/bluge // next iteration of bleve

projects using bluge

All bleeding edge afaik atm - but definitely good places to look at bluge usage

https://github.com/prabhatsharma/zinc https://github.com/mosuka/phalanx

Look ma I made a vector database

Gotta take a look at this one - will report later

https://github.com/semi-technologies/weaviate

- - + + \ No newline at end of file diff --git a/blog/tags.html b/blog/tags.html index 6ecd10b..f64f811 100644 --- a/blog/tags.html +++ b/blog/tags.html @@ -9,13 +9,13 @@ - - + + - - +
+ + \ No newline at end of file diff --git a/blog/tags/backend.html b/blog/tags/backend.html index acd5cbb..3c4ae04 100644 --- a/blog/tags/backend.html +++ b/blog/tags/backend.html @@ -9,15 +9,15 @@ - - + +
-

2 posts tagged with "backend"

View All Tags

Jan Halfar

While building this website and integrating https://docsearch.algolia.com and evaluating another solution by a large company in parallel I could not help to search github and the web for the current state of search engines and search related services.

Since I had done the same thing about a year ago, I was surprised to see how quickly things are moving atm.

Algolia

I was blown away by the quality of https://www.algolia.com and I wish it was open source, but I guess, we all have to make a living ;)

To see how awesome a web (search) interface can be check out https://www.lacoste.com/us/#query=red%20jackets%20for%20men

Apart from that the UI/UX of their backend tools is fantastic.

Elastic

When it comes to https://www.elastic.com I am a bit nervous about the future of the licensing, despite the fact, that I understand their motivation. At the same time the https://opensearch.org does not seem to be an ampty threat.

typesense.org

I do not know, who was hiding under a rock, but I had not seen https://typesense.org before and they certainly have a bold claim: "The Open Source Algolia Alternative" / "The Easier To Use ElasticSearch Alternative"

When looking at https://github.com/typesense/typesense/graphs/contributors it seems, that Kishore Nallan has been working on this for a while. Unfourtunately I do not really see a lot of external contributions, C++ does not seem to attract a lot of contribution.

MeiliSearch

This Rust project https://www.meilisearch.com/ seems to be picking up speed and is definetly on the test short list. It is a fresh codebase with siginficant open source contributions and certainly will attract new developers with Rust and a modern architecture.

Go eco system

Obviously we are very interested in Go powered software and there are a few notable projects. ATM I do not see anything elastic or algolia like, that would be really mature.

bleve / bluge

Marty Schoch seems to be the man when it comes down to text indexing libraries in written in Go and bluge seems to be THE library, that is solid and modern, when implementing text search in your Go application.

https://github.com/blevesearch/bleve +

2 posts tagged with "backend"

View All Tags

Jan Halfar

While building this website and integrating https://docsearch.algolia.com and evaluating another solution by a large company in parallel I could not help to search github and the web for the current state of search engines and search related services.

Since I had done the same thing about a year ago, I was surprised to see how quickly things are moving atm.

Algolia

I was blown away by the quality of https://www.algolia.com and I wish it was open source, but I guess, we all have to make a living ;)

To see how awesome a web (search) interface can be check out https://www.lacoste.com/us/#query=red%20jackets%20for%20men

Apart from that the UI/UX of their backend tools is fantastic.

Elastic

When it comes to https://www.elastic.com I am a bit nervous about the future of the licensing, despite the fact, that I understand their motivation. At the same time the https://opensearch.org does not seem to be an ampty threat.

typesense.org

I do not know, who was hiding under a rock, but I had not seen https://typesense.org before and they certainly have a bold claim: "The Open Source Algolia Alternative" / "The Easier To Use ElasticSearch Alternative"

When looking at https://github.com/typesense/typesense/graphs/contributors it seems, that Kishore Nallan has been working on this for a while. Unfourtunately I do not really see a lot of external contributions, C++ does not seem to attract a lot of contribution.

MeiliSearch

This Rust project https://www.meilisearch.com/ seems to be picking up speed and is definetly on the test short list. It is a fresh codebase with siginficant open source contributions and certainly will attract new developers with Rust and a modern architecture.

Go eco system

Obviously we are very interested in Go powered software and there are a few notable projects. ATM I do not see anything elastic or algolia like, that would be really mature.

bleve / bluge

Marty Schoch seems to be the man when it comes down to text indexing libraries in written in Go and bluge seems to be THE library, that is solid and modern, when implementing text search in your Go application.

https://github.com/blevesearch/bleve https://github.com/blugelabs/bluge // next iteration of bleve

projects using bluge

All bleeding edge afaik atm - but definitely good places to look at bluge usage

https://github.com/prabhatsharma/zinc https://github.com/mosuka/phalanx

Look ma I made a vector database

Gotta take a look at this one - will report later

https://github.com/semi-technologies/weaviate

- - + + \ No newline at end of file diff --git a/blog/tags/bundle-size.html b/blog/tags/bundle-size.html index 7771f05..99a1853 100644 --- a/blog/tags/bundle-size.html +++ b/blog/tags/bundle-size.html @@ -9,16 +9,16 @@ - - + +
-

One post tagged with "bundle size"

View All Tags

Nicola Turcato

Intro

JavaScript is parsed, compiled and executed in the main thread of the browser. Which means that users have to wait for all of this before they can interact with the website.

Frontend performance optimization is critical because it accounts for around 80-90% of user response time (10-20% backend). +

One post tagged with "bundle size"

View All Tags

Nicola Turcato

Intro

JavaScript is parsed, compiled and executed in the main thread of the browser. Which means that users have to wait for all of this before they can interact with the website.

Frontend performance optimization is critical because it accounts for around 80-90% of user response time (10-20% backend). So when a user is waiting for a page to load, around 80-90% of the time is due to frontend related code and assets.

Nobody likes waiting…

A study found that if a site takes longer than 4 seconds to load, up to 25% of users would abandon the site.

Sending large JavaScript payloads impacts the speed of your site significantly.

Mazzarri

What is a "bundle"?

Your frontend application needs a bunch of JS files to run. These files can be in the format of internal dependency like the JS files you have written yourself. But they can also be external dependencies you use to build your application.

JS bundling is an optimization technique to reduce the number of server requests for JavaScript files. Bundling accomplishes this by merging multiple JA files together into one file to reduce the number of page requests.

Bundle Everywhere

Performance implications

  • Time to transmit over the network: considering slow connections with some mobile devices, it's possible that your page will not be interactive until it loads.-> More bytes = longer download times
  • JS parse and compile time: more code you load, more the browser must parse.-> JS gets parsed and compiled on the main thread, when the main thread is busy, the page can't respond to user input
  • JS execution time: optimally you will only pack the code that you expect to execute. The more code you want to execute the longer it will take. It's possible that your page won't be interactive until some of this completes.-> JS is also executed on the main thread, if your page runs a lot of code before it's really needed, that also delays your Time to Interactive
  • Memory consumption: everything fills up the space -> code itself, runtime variables, DOM elements created, etc.-> Pages appear slow when it consumes a lot of memory. Memory leaks can cause your page to freeze up completely!!

AS SMALL AS POSSIBLE! I experienced that is not really possible to give a precise answer because each application is different. Generally you want the resources on initial load to be as small as possible, so that you decrease the initial load time, and then load more resources as needed on the fly.

Mr Chao

What do we do then?

Meh

How to start decreasing the bundle size?

  • Measure: first of all you want to measure. The first step is to use Lighthouse and try to understand the results. It will give you a couple of interesting metrics and some tips. Time to interactive (TTI) is a good reflection of your bundle size because your bundle needs to be evaluated entirely before a user can interact with your web app.
  • Analyze: Consists on analyzing the bundle in order to detect critical chunks. A useful tool is Webpack Bundle Analyzer.

Stonks

Breaking up the bundle...

  • Monitor network requests: These happens between our FCP and TTI. As the initial request for data often occurs when our components initially mount.
  • Reduce the total dom nodes: the less the page needs to render, the less time it takes.
  • Moving work off the main thread: By moving heavy computations to a web worker, the computation will be run on a separate thread, and not block the actual rendering of the page
  • Caching: Even if not useful for users on first page landing, caching data, bundles, and assets can make subsequent visits way fast

Breaking Bad

Which strategies can we adopt?

  • Minification and Dead Code Elimination: These processes are often summed up as minifying or uglifying.
  • Tree shaking: Tree shaking is dead code elimination on a project or library. Always try to use deps which support “tree shaking”, Bundlephobia could be your friend in this case.
  • Code Splitting and Lazy Loading: Code splitting consists on taking a collection of modules and remove them from the main JS bundle. Lazy loading means we can load this newly created bundle later on.
  • Replace/rewrite large dependencies: Consider replacing or rewriting libraries that are large in size where you might not need all of its functionalities (Moment.js for example).
  • Feature module import: Check to see if you are using only a feature module of the library that can be imported alone without importing the whole library (Lodash for example).

Strategy

Useful tools to help you reducing bundle size

  • Lighthouse: automated tool for improving the performance, quality, and correctness of your web apps
  • Bundlephobia: Bundlephobia helps you find the performance impact of npm packages
  • Webpack Bundle Analyzer: analyzes your bundle
  • VS Code: Import Cost plugin -> Display import/require package size in the editor

Tools

Conclusion

Performance cannot be stripped down to a single metric such as bundle size. It would be great! Unfortunately there is no single place to measure all of them.I think metrics like the Core Web Vitals and a general look at bundle size should be considered as a starting point. You will cry... A lot... But don’t give up!

The End

- - + + \ No newline at end of file diff --git a/blog/tags/bundle.html b/blog/tags/bundle.html index 78f823e..0b6dfbc 100644 --- a/blog/tags/bundle.html +++ b/blog/tags/bundle.html @@ -9,16 +9,16 @@ - - + +
-

One post tagged with "bundle"

View All Tags

Nicola Turcato

Intro

JavaScript is parsed, compiled and executed in the main thread of the browser. Which means that users have to wait for all of this before they can interact with the website.

Frontend performance optimization is critical because it accounts for around 80-90% of user response time (10-20% backend). +

One post tagged with "bundle"

View All Tags

Nicola Turcato

Intro

JavaScript is parsed, compiled and executed in the main thread of the browser. Which means that users have to wait for all of this before they can interact with the website.

Frontend performance optimization is critical because it accounts for around 80-90% of user response time (10-20% backend). So when a user is waiting for a page to load, around 80-90% of the time is due to frontend related code and assets.

Nobody likes waiting…

A study found that if a site takes longer than 4 seconds to load, up to 25% of users would abandon the site.

Sending large JavaScript payloads impacts the speed of your site significantly.

Mazzarri

What is a "bundle"?

Your frontend application needs a bunch of JS files to run. These files can be in the format of internal dependency like the JS files you have written yourself. But they can also be external dependencies you use to build your application.

JS bundling is an optimization technique to reduce the number of server requests for JavaScript files. Bundling accomplishes this by merging multiple JA files together into one file to reduce the number of page requests.

Bundle Everywhere

Performance implications

  • Time to transmit over the network: considering slow connections with some mobile devices, it's possible that your page will not be interactive until it loads.-> More bytes = longer download times
  • JS parse and compile time: more code you load, more the browser must parse.-> JS gets parsed and compiled on the main thread, when the main thread is busy, the page can't respond to user input
  • JS execution time: optimally you will only pack the code that you expect to execute. The more code you want to execute the longer it will take. It's possible that your page won't be interactive until some of this completes.-> JS is also executed on the main thread, if your page runs a lot of code before it's really needed, that also delays your Time to Interactive
  • Memory consumption: everything fills up the space -> code itself, runtime variables, DOM elements created, etc.-> Pages appear slow when it consumes a lot of memory. Memory leaks can cause your page to freeze up completely!!

AS SMALL AS POSSIBLE! I experienced that is not really possible to give a precise answer because each application is different. Generally you want the resources on initial load to be as small as possible, so that you decrease the initial load time, and then load more resources as needed on the fly.

Mr Chao

What do we do then?

Meh

How to start decreasing the bundle size?

  • Measure: first of all you want to measure. The first step is to use Lighthouse and try to understand the results. It will give you a couple of interesting metrics and some tips. Time to interactive (TTI) is a good reflection of your bundle size because your bundle needs to be evaluated entirely before a user can interact with your web app.
  • Analyze: Consists on analyzing the bundle in order to detect critical chunks. A useful tool is Webpack Bundle Analyzer.

Stonks

Breaking up the bundle...

  • Monitor network requests: These happens between our FCP and TTI. As the initial request for data often occurs when our components initially mount.
  • Reduce the total dom nodes: the less the page needs to render, the less time it takes.
  • Moving work off the main thread: By moving heavy computations to a web worker, the computation will be run on a separate thread, and not block the actual rendering of the page
  • Caching: Even if not useful for users on first page landing, caching data, bundles, and assets can make subsequent visits way fast

Breaking Bad

Which strategies can we adopt?

  • Minification and Dead Code Elimination: These processes are often summed up as minifying or uglifying.
  • Tree shaking: Tree shaking is dead code elimination on a project or library. Always try to use deps which support “tree shaking”, Bundlephobia could be your friend in this case.
  • Code Splitting and Lazy Loading: Code splitting consists on taking a collection of modules and remove them from the main JS bundle. Lazy loading means we can load this newly created bundle later on.
  • Replace/rewrite large dependencies: Consider replacing or rewriting libraries that are large in size where you might not need all of its functionalities (Moment.js for example).
  • Feature module import: Check to see if you are using only a feature module of the library that can be imported alone without importing the whole library (Lodash for example).

Strategy

Useful tools to help you reducing bundle size

  • Lighthouse: automated tool for improving the performance, quality, and correctness of your web apps
  • Bundlephobia: Bundlephobia helps you find the performance impact of npm packages
  • Webpack Bundle Analyzer: analyzes your bundle
  • VS Code: Import Cost plugin -> Display import/require package size in the editor

Tools

Conclusion

Performance cannot be stripped down to a single metric such as bundle size. It would be great! Unfortunately there is no single place to measure all of them.I think metrics like the Core Web Vitals and a general look at bundle size should be considered as a starting point. You will cry... A lot... But don’t give up!

The End

- - + + \ No newline at end of file diff --git a/blog/tags/cardinality.html b/blog/tags/cardinality.html index 6a6e99e..2377672 100644 --- a/blog/tags/cardinality.html +++ b/blog/tags/cardinality.html @@ -9,12 +9,12 @@ - - + +
-

One post tagged with "cardinality"

View All Tags

The Annoyance

So, we've all been there. You go to your trusty grafana, search for some sweet metrics that you implemented and WHAM! +

- - + + \ No newline at end of file diff --git a/blog/tags/currency.html b/blog/tags/currency.html new file mode 100644 index 0000000..628c4ee --- /dev/null +++ b/blog/tags/currency.html @@ -0,0 +1,23 @@ + + + + + +One post tagged with "currency" | foomo project docs + + + + + + + + + +
+

One post tagged with "currency"

View All Tags

Intro

Calculating with money can be tricky if not taken proper precautions. Some might be tempted to use float representation for calculating with currency values. That is problematic because of possible rounding errors.

Finite accuracy of representation

Floating points are represented like this

Floating point representation

Not every number can be represented with a finite number of decimal places

0.01 —> 0.0000011001100110011…

Taking 17 places of the above results in 0.010000000000000001

Consider the following code snipet that shows the missing accuracy

func main() {

var n float64 = 0

for i := 0; i < 1000; i++ {
n += .01
}

fmt.Println(n)

}

Result: 9.999999999999831

Money computations

They can't be done with floating-point as it would inevitably lead to rounding errors.

Even the following packages are problematic:

github.com/shopspring/decimal

github.com/Rhymond/go-money

a := decimal.NewFromInt(2)
b := decimal.NewFromFloat(300.99)
c := a.Mul(b)
d := c.Div(decimal.NewFromInt(3))

Solution

Use Int by representing money in cents:

  • 10.99 -> 1099 (cents)
  • 10.9900 -> 109900 (4 digit tax)

Conclusio

Division is a problem!

1/3 - > 0.33333333… +Correct way: 0.33, 0.33, 0.34

When doing money calculations one should avoid division as it inevitably leads to loss of accuracy. +When dividing make sure to round to cent and deal with diffs.

Division by 10^k is ok till we are inside of the range of the data type.

+ + + + \ No newline at end of file diff --git a/blog/tags/debugging.html b/blog/tags/debugging.html index ca61e0a..edf1420 100644 --- a/blog/tags/debugging.html +++ b/blog/tags/debugging.html @@ -9,13 +9,13 @@ - - + + - - +
+ + \ No newline at end of file diff --git a/blog/tags/decimal-accuracy.html b/blog/tags/decimal-accuracy.html new file mode 100644 index 0000000..f5820bc --- /dev/null +++ b/blog/tags/decimal-accuracy.html @@ -0,0 +1,23 @@ + + + + + +One post tagged with "decimal accuracy" | foomo project docs + + + + + + + + + +
+

One post tagged with "decimal accuracy"

View All Tags

Intro

Calculating with money can be tricky if not taken proper precautions. Some might be tempted to use float representation for calculating with currency values. That is problematic because of possible rounding errors.

Finite accuracy of representation

Floating points are represented like this

Floating point representation

Not every number can be represented with a finite number of decimal places

0.01 —> 0.0000011001100110011…

Taking 17 places of the above results in 0.010000000000000001

Consider the following code snipet that shows the missing accuracy

func main() {

var n float64 = 0

for i := 0; i < 1000; i++ {
n += .01
}

fmt.Println(n)

}

Result: 9.999999999999831

Money computations

They can't be done with floating-point as it would inevitably lead to rounding errors.

Even the following packages are problematic:

github.com/shopspring/decimal

github.com/Rhymond/go-money

a := decimal.NewFromInt(2)
b := decimal.NewFromFloat(300.99)
c := a.Mul(b)
d := c.Div(decimal.NewFromInt(3))

Solution

Use Int by representing money in cents:

  • 10.99 -> 1099 (cents)
  • 10.9900 -> 109900 (4 digit tax)

Conclusio

Division is a problem!

1/3 - > 0.33333333… +Correct way: 0.33, 0.33, 0.34

When doing money calculations one should avoid division as it inevitably leads to loss of accuracy. +When dividing make sure to round to cent and deal with diffs.

Division by 10^k is ok till we are inside of the range of the data type.

+ + + + \ No newline at end of file diff --git a/blog/tags/devops.html b/blog/tags/devops.html index 9ace53d..5905348 100644 --- a/blog/tags/devops.html +++ b/blog/tags/devops.html @@ -9,12 +9,12 @@ - - + +
-

One post tagged with "devops"

View All Tags

The Annoyance

So, we've all been there. You go to your trusty grafana, search for some sweet metrics that you implemented and WHAM! +

- - + + \ No newline at end of file diff --git a/blog/tags/foomo.html b/blog/tags/foomo.html index d49df43..13be471 100644 --- a/blog/tags/foomo.html +++ b/blog/tags/foomo.html @@ -9,13 +9,13 @@ - - + +
-

One post tagged with "foomo"

View All Tags

Jan Halfar

A few years ago we abandoned the previous version of https://www.foomo.org as we did not want to maintain the old wordpress installation and the project was only living in README.md in the repos living under https://www.github.com/foomo .

As things have grown over time we have decided to re-launch a website / cross project documentation.

So welcome back and enjoy the view to the past:

blast from the past

- - +

One post tagged with "foomo"

View All Tags

Jan Halfar

A few years ago we abandoned the previous version of https://www.foomo.org as we did not want to maintain the old wordpress installation and the project was only living in README.md in the repos living under https://www.github.com/foomo .

As things have grown over time we have decided to re-launch a website / cross project documentation.

So welcome back and enjoy the view to the past:

blast from the past

+ + \ No newline at end of file diff --git a/blog/tags/frontend.html b/blog/tags/frontend.html index 6a1035d..2b39ff2 100644 --- a/blog/tags/frontend.html +++ b/blog/tags/frontend.html @@ -9,14 +9,14 @@ - - + +
-

One post tagged with "frontend"

View All Tags

Marko Trebižan

Issue with performance

When building an ecommerce site or an application where performance is a great deal for the users, you need to keep your application fast and responsive. Frontend developers have already many use-cases when the UI becomes laggy and this increases when 3rd party scripts are being included, such as Google Tag Manager or various Live chats (e.g. Intercom).

This does not only influences the users when using the site but also Lighthouse score gets lower which also influences page rankings. So the most naive and easy way for this is to defer loading of such scripts but when you need to get all the data from the start of the application, such tactic is not an option. So what else can we do?

Partytown to the rescue

Developers at BuilderIO created an library Partytown that would allow relocating resources from 3rd party scripts off the main thread. +

One post tagged with "frontend"

View All Tags

Marko Trebižan

Issue with performance

When building an ecommerce site or an application where performance is a great deal for the users, you need to keep your application fast and responsive. Frontend developers have already many use-cases when the UI becomes laggy and this increases when 3rd party scripts are being included, such as Google Tag Manager or various Live chats (e.g. Intercom).

This does not only influences the users when using the site but also Lighthouse score gets lower which also influences page rankings. So the most naive and easy way for this is to defer loading of such scripts but when you need to get all the data from the start of the application, such tactic is not an option. So what else can we do?

Partytown to the rescue

Developers at BuilderIO created an library Partytown that would allow relocating resources from 3rd party scripts off the main thread. We won't dive into specifics how it works, because they explain it nicely on their GitHub page.

In our stack we use Next.js React framework and we will go through the basic steps that will allow us to include Partytown for Google Tag Manager.

Setup

Partytown script needs to be located inside our application and live on the same domain. Since we're using monorepo structure, we need to copy this script across all our frontend application. For that we used CopyPlugin webpack plugin in our Next.js config file:

config.plugins.push(
...
new CopyPlugin({
patterns: [
{
// we copy script from node_modules partytown package to `~partytown` folder in our package that serves static files
from: path.join(path.dirname(require.resolve('@builder.io/partytown')), 'lib'),
// paths for SSR and client side rendering differ
to: path.join(`${isServer ? '..' : '.'}/static/assets/`, '~partytown'),
},
],
})
);

Partytown's requirement is that it needs to know what script should it load into own web worker. For that we set script type to text/partytown. This will prevent script to load on initial load.

Inside _document.tsx we add this:

<Head>
...
// include Partytown and set custom path due to multiple frontends
<Partytown lib={`${addTrailingSlash(this.props.basePath)}_next/static/assets/~partytown/`} debug={false} />
// tag 3rd party script with partytown type
<script type="text/partytown" src={`https://www.googletagmanager.com/gtm.js?id=${id}`} />
...
</Head>

Results

So now, does it work? We used one of our large Ecommerce sites to test the landing Lighthouse score.

This was before adding Partytown:

Lighthouse before Partytown

Here you can see 2 critical things: almost 1s of total blocking time (TBT) and 9s of time to interactive (TTI).

After we added Partytown, we got this:

Lighthouse after Partytown

Time to interactive went from 9s to 6.1s which is almost 33% improvement and total blocking time was reduce by more than 50%! We were more than impressed how easy it was to improve our performances.

Side note: Both screenshots were compressed using Squoosh App.

Next steps

After successful testing of Partytown for Google Tag Manager script, we are more than interested in trying it out on our other scripts. One important topic will be to test Partytown with other service-worker related libraries and how to use them together.

- - + + \ No newline at end of file diff --git a/blog/tags/go.html b/blog/tags/go.html index b8bc6c4..a21ea93 100644 --- a/blog/tags/go.html +++ b/blog/tags/go.html @@ -9,15 +9,15 @@ - - + +
-

2 posts tagged with "go"

View All Tags

Jan Halfar

While building this website and integrating https://docsearch.algolia.com and evaluating another solution by a large company in parallel I could not help to search github and the web for the current state of search engines and search related services.

Since I had done the same thing about a year ago, I was surprised to see how quickly things are moving atm.

Algolia

I was blown away by the quality of https://www.algolia.com and I wish it was open source, but I guess, we all have to make a living ;)

To see how awesome a web (search) interface can be check out https://www.lacoste.com/us/#query=red%20jackets%20for%20men

Apart from that the UI/UX of their backend tools is fantastic.

Elastic

When it comes to https://www.elastic.com I am a bit nervous about the future of the licensing, despite the fact, that I understand their motivation. At the same time the https://opensearch.org does not seem to be an ampty threat.

typesense.org

I do not know, who was hiding under a rock, but I had not seen https://typesense.org before and they certainly have a bold claim: "The Open Source Algolia Alternative" / "The Easier To Use ElasticSearch Alternative"

When looking at https://github.com/typesense/typesense/graphs/contributors it seems, that Kishore Nallan has been working on this for a while. Unfourtunately I do not really see a lot of external contributions, C++ does not seem to attract a lot of contribution.

MeiliSearch

This Rust project https://www.meilisearch.com/ seems to be picking up speed and is definetly on the test short list. It is a fresh codebase with siginficant open source contributions and certainly will attract new developers with Rust and a modern architecture.

Go eco system

Obviously we are very interested in Go powered software and there are a few notable projects. ATM I do not see anything elastic or algolia like, that would be really mature.

bleve / bluge

Marty Schoch seems to be the man when it comes down to text indexing libraries in written in Go and bluge seems to be THE library, that is solid and modern, when implementing text search in your Go application.

https://github.com/blevesearch/bleve +

2 posts tagged with "go"

View All Tags

Jan Halfar

While building this website and integrating https://docsearch.algolia.com and evaluating another solution by a large company in parallel I could not help to search github and the web for the current state of search engines and search related services.

Since I had done the same thing about a year ago, I was surprised to see how quickly things are moving atm.

Algolia

I was blown away by the quality of https://www.algolia.com and I wish it was open source, but I guess, we all have to make a living ;)

To see how awesome a web (search) interface can be check out https://www.lacoste.com/us/#query=red%20jackets%20for%20men

Apart from that the UI/UX of their backend tools is fantastic.

Elastic

When it comes to https://www.elastic.com I am a bit nervous about the future of the licensing, despite the fact, that I understand their motivation. At the same time the https://opensearch.org does not seem to be an ampty threat.

typesense.org

I do not know, who was hiding under a rock, but I had not seen https://typesense.org before and they certainly have a bold claim: "The Open Source Algolia Alternative" / "The Easier To Use ElasticSearch Alternative"

When looking at https://github.com/typesense/typesense/graphs/contributors it seems, that Kishore Nallan has been working on this for a while. Unfourtunately I do not really see a lot of external contributions, C++ does not seem to attract a lot of contribution.

MeiliSearch

This Rust project https://www.meilisearch.com/ seems to be picking up speed and is definetly on the test short list. It is a fresh codebase with siginficant open source contributions and certainly will attract new developers with Rust and a modern architecture.

Go eco system

Obviously we are very interested in Go powered software and there are a few notable projects. ATM I do not see anything elastic or algolia like, that would be really mature.

bleve / bluge

Marty Schoch seems to be the man when it comes down to text indexing libraries in written in Go and bluge seems to be THE library, that is solid and modern, when implementing text search in your Go application.

https://github.com/blevesearch/bleve https://github.com/blugelabs/bluge // next iteration of bleve

projects using bluge

All bleeding edge afaik atm - but definitely good places to look at bluge usage

https://github.com/prabhatsharma/zinc https://github.com/mosuka/phalanx

Look ma I made a vector database

Gotta take a look at this one - will report later

https://github.com/semi-technologies/weaviate

- - + + \ No newline at end of file diff --git a/blog/tags/golang.html b/blog/tags/golang.html new file mode 100644 index 0000000..b085f6b --- /dev/null +++ b/blog/tags/golang.html @@ -0,0 +1,23 @@ + + + + + +One post tagged with "golang" | foomo project docs + + + + + + + + + +
+

One post tagged with "golang"

View All Tags

Intro

Calculating with money can be tricky if not taken proper precautions. Some might be tempted to use float representation for calculating with currency values. That is problematic because of possible rounding errors.

Finite accuracy of representation

Floating points are represented like this

Floating point representation

Not every number can be represented with a finite number of decimal places

0.01 —> 0.0000011001100110011…

Taking 17 places of the above results in 0.010000000000000001

Consider the following code snipet that shows the missing accuracy

func main() {

var n float64 = 0

for i := 0; i < 1000; i++ {
n += .01
}

fmt.Println(n)

}

Result: 9.999999999999831

Money computations

They can't be done with floating-point as it would inevitably lead to rounding errors.

Even the following packages are problematic:

github.com/shopspring/decimal

github.com/Rhymond/go-money

a := decimal.NewFromInt(2)
b := decimal.NewFromFloat(300.99)
c := a.Mul(b)
d := c.Div(decimal.NewFromInt(3))

Solution

Use Int by representing money in cents:

  • 10.99 -> 1099 (cents)
  • 10.9900 -> 109900 (4 digit tax)

Conclusio

Division is a problem!

1/3 - > 0.33333333… +Correct way: 0.33, 0.33, 0.34

When doing money calculations one should avoid division as it inevitably leads to loss of accuracy. +When dividing make sure to round to cent and deal with diffs.

Division by 10^k is ok till we are inside of the range of the data type.

+ + + + \ No newline at end of file diff --git a/blog/tags/javascript.html b/blog/tags/javascript.html index 3febf39..01f638d 100644 --- a/blog/tags/javascript.html +++ b/blog/tags/javascript.html @@ -9,16 +9,16 @@ - - + +
-

One post tagged with "javascript"

View All Tags

Nicola Turcato

Intro

JavaScript is parsed, compiled and executed in the main thread of the browser. Which means that users have to wait for all of this before they can interact with the website.

Frontend performance optimization is critical because it accounts for around 80-90% of user response time (10-20% backend). +

One post tagged with "javascript"

View All Tags

Nicola Turcato

Intro

JavaScript is parsed, compiled and executed in the main thread of the browser. Which means that users have to wait for all of this before they can interact with the website.

Frontend performance optimization is critical because it accounts for around 80-90% of user response time (10-20% backend). So when a user is waiting for a page to load, around 80-90% of the time is due to frontend related code and assets.

Nobody likes waiting…

A study found that if a site takes longer than 4 seconds to load, up to 25% of users would abandon the site.

Sending large JavaScript payloads impacts the speed of your site significantly.

Mazzarri

What is a "bundle"?

Your frontend application needs a bunch of JS files to run. These files can be in the format of internal dependency like the JS files you have written yourself. But they can also be external dependencies you use to build your application.

JS bundling is an optimization technique to reduce the number of server requests for JavaScript files. Bundling accomplishes this by merging multiple JA files together into one file to reduce the number of page requests.

Bundle Everywhere

Performance implications

  • Time to transmit over the network: considering slow connections with some mobile devices, it's possible that your page will not be interactive until it loads.-> More bytes = longer download times
  • JS parse and compile time: more code you load, more the browser must parse.-> JS gets parsed and compiled on the main thread, when the main thread is busy, the page can't respond to user input
  • JS execution time: optimally you will only pack the code that you expect to execute. The more code you want to execute the longer it will take. It's possible that your page won't be interactive until some of this completes.-> JS is also executed on the main thread, if your page runs a lot of code before it's really needed, that also delays your Time to Interactive
  • Memory consumption: everything fills up the space -> code itself, runtime variables, DOM elements created, etc.-> Pages appear slow when it consumes a lot of memory. Memory leaks can cause your page to freeze up completely!!

AS SMALL AS POSSIBLE! I experienced that is not really possible to give a precise answer because each application is different. Generally you want the resources on initial load to be as small as possible, so that you decrease the initial load time, and then load more resources as needed on the fly.

Mr Chao

What do we do then?

Meh

How to start decreasing the bundle size?

  • Measure: first of all you want to measure. The first step is to use Lighthouse and try to understand the results. It will give you a couple of interesting metrics and some tips. Time to interactive (TTI) is a good reflection of your bundle size because your bundle needs to be evaluated entirely before a user can interact with your web app.
  • Analyze: Consists on analyzing the bundle in order to detect critical chunks. A useful tool is Webpack Bundle Analyzer.

Stonks

Breaking up the bundle...

  • Monitor network requests: These happens between our FCP and TTI. As the initial request for data often occurs when our components initially mount.
  • Reduce the total dom nodes: the less the page needs to render, the less time it takes.
  • Moving work off the main thread: By moving heavy computations to a web worker, the computation will be run on a separate thread, and not block the actual rendering of the page
  • Caching: Even if not useful for users on first page landing, caching data, bundles, and assets can make subsequent visits way fast

Breaking Bad

Which strategies can we adopt?

  • Minification and Dead Code Elimination: These processes are often summed up as minifying or uglifying.
  • Tree shaking: Tree shaking is dead code elimination on a project or library. Always try to use deps which support “tree shaking”, Bundlephobia could be your friend in this case.
  • Code Splitting and Lazy Loading: Code splitting consists on taking a collection of modules and remove them from the main JS bundle. Lazy loading means we can load this newly created bundle later on.
  • Replace/rewrite large dependencies: Consider replacing or rewriting libraries that are large in size where you might not need all of its functionalities (Moment.js for example).
  • Feature module import: Check to see if you are using only a feature module of the library that can be imported alone without importing the whole library (Lodash for example).

Strategy

Useful tools to help you reducing bundle size

  • Lighthouse: automated tool for improving the performance, quality, and correctness of your web apps
  • Bundlephobia: Bundlephobia helps you find the performance impact of npm packages
  • Webpack Bundle Analyzer: analyzes your bundle
  • VS Code: Import Cost plugin -> Display import/require package size in the editor

Tools

Conclusion

Performance cannot be stripped down to a single metric such as bundle size. It would be great! Unfortunately there is no single place to measure all of them.I think metrics like the Core Web Vitals and a general look at bundle size should be considered as a starting point. You will cry... A lot... But don’t give up!

The End

- - + + \ No newline at end of file diff --git a/blog/tags/k-8-s.html b/blog/tags/k-8-s.html index 07b4252..fb3d3b4 100644 --- a/blog/tags/k-8-s.html +++ b/blog/tags/k-8-s.html @@ -9,12 +9,12 @@ - - + +
-

One post tagged with "k8s"

View All Tags

The Annoyance

So, we've all been there. You go to your trusty grafana, search for some sweet metrics that you implemented and WHAM! +

- - + + \ No newline at end of file diff --git a/blog/tags/memory.html b/blog/tags/memory.html index 1dbe5c1..d285de9 100644 --- a/blog/tags/memory.html +++ b/blog/tags/memory.html @@ -9,12 +9,12 @@ - - + +
-

One post tagged with "memory"

View All Tags

The Annoyance

So, we've all been there. You go to your trusty grafana, search for some sweet metrics that you implemented and WHAM! +

- - + + \ No newline at end of file diff --git a/blog/tags/oom.html b/blog/tags/oom.html index ff5282a..dff81e7 100644 --- a/blog/tags/oom.html +++ b/blog/tags/oom.html @@ -9,12 +9,12 @@ - - + +
-

One post tagged with "oom"

View All Tags

The Annoyance

So, we've all been there. You go to your trusty grafana, search for some sweet metrics that you implemented and WHAM! +

- - + + \ No newline at end of file diff --git a/blog/tags/ops.html b/blog/tags/ops.html index eb0168c..c886123 100644 --- a/blog/tags/ops.html +++ b/blog/tags/ops.html @@ -9,12 +9,12 @@ - - + +
-

One post tagged with "ops"

View All Tags

The Annoyance

So, we've all been there. You go to your trusty grafana, search for some sweet metrics that you implemented and WHAM! +

- - + + \ No newline at end of file diff --git a/blog/tags/performance.html b/blog/tags/performance.html index a7abcc0..acff0f5 100644 --- a/blog/tags/performance.html +++ b/blog/tags/performance.html @@ -9,14 +9,14 @@ - - + +
-

One post tagged with "performance"

View All Tags

Marko Trebižan

Issue with performance

When building an ecommerce site or an application where performance is a great deal for the users, you need to keep your application fast and responsive. Frontend developers have already many use-cases when the UI becomes laggy and this increases when 3rd party scripts are being included, such as Google Tag Manager or various Live chats (e.g. Intercom).

This does not only influences the users when using the site but also Lighthouse score gets lower which also influences page rankings. So the most naive and easy way for this is to defer loading of such scripts but when you need to get all the data from the start of the application, such tactic is not an option. So what else can we do?

Partytown to the rescue

Developers at BuilderIO created an library Partytown that would allow relocating resources from 3rd party scripts off the main thread. +

One post tagged with "performance"

View All Tags

Marko Trebižan

Issue with performance

When building an ecommerce site or an application where performance is a great deal for the users, you need to keep your application fast and responsive. Frontend developers have already many use-cases when the UI becomes laggy and this increases when 3rd party scripts are being included, such as Google Tag Manager or various Live chats (e.g. Intercom).

This does not only influences the users when using the site but also Lighthouse score gets lower which also influences page rankings. So the most naive and easy way for this is to defer loading of such scripts but when you need to get all the data from the start of the application, such tactic is not an option. So what else can we do?

Partytown to the rescue

Developers at BuilderIO created an library Partytown that would allow relocating resources from 3rd party scripts off the main thread. We won't dive into specifics how it works, because they explain it nicely on their GitHub page.

In our stack we use Next.js React framework and we will go through the basic steps that will allow us to include Partytown for Google Tag Manager.

Setup

Partytown script needs to be located inside our application and live on the same domain. Since we're using monorepo structure, we need to copy this script across all our frontend application. For that we used CopyPlugin webpack plugin in our Next.js config file:

config.plugins.push(
...
new CopyPlugin({
patterns: [
{
// we copy script from node_modules partytown package to `~partytown` folder in our package that serves static files
from: path.join(path.dirname(require.resolve('@builder.io/partytown')), 'lib'),
// paths for SSR and client side rendering differ
to: path.join(`${isServer ? '..' : '.'}/static/assets/`, '~partytown'),
},
],
})
);

Partytown's requirement is that it needs to know what script should it load into own web worker. For that we set script type to text/partytown. This will prevent script to load on initial load.

Inside _document.tsx we add this:

<Head>
...
// include Partytown and set custom path due to multiple frontends
<Partytown lib={`${addTrailingSlash(this.props.basePath)}_next/static/assets/~partytown/`} debug={false} />
// tag 3rd party script with partytown type
<script type="text/partytown" src={`https://www.googletagmanager.com/gtm.js?id=${id}`} />
...
</Head>

Results

So now, does it work? We used one of our large Ecommerce sites to test the landing Lighthouse score.

This was before adding Partytown:

Lighthouse before Partytown

Here you can see 2 critical things: almost 1s of total blocking time (TBT) and 9s of time to interactive (TTI).

After we added Partytown, we got this:

Lighthouse after Partytown

Time to interactive went from 9s to 6.1s which is almost 33% improvement and total blocking time was reduce by more than 50%! We were more than impressed how easy it was to improve our performances.

Side note: Both screenshots were compressed using Squoosh App.

Next steps

After successful testing of Partytown for Google Tag Manager script, we are more than interested in trying it out on our other scripts. One important topic will be to test Partytown with other service-worker related libraries and how to use them together.

- - + + \ No newline at end of file diff --git a/blog/tags/prometheus.html b/blog/tags/prometheus.html index 27c2634..0b465bd 100644 --- a/blog/tags/prometheus.html +++ b/blog/tags/prometheus.html @@ -9,12 +9,12 @@ - - + +
-

One post tagged with "prometheus"

View All Tags

The Annoyance

So, we've all been there. You go to your trusty grafana, search for some sweet metrics that you implemented and WHAM! +

- - + + \ No newline at end of file diff --git a/blog/tags/search-engine.html b/blog/tags/search-engine.html index 85d384f..5867970 100644 --- a/blog/tags/search-engine.html +++ b/blog/tags/search-engine.html @@ -9,15 +9,15 @@ - - + +
-

One post tagged with "search-engine"

View All Tags

Jan Halfar

While building this website and integrating https://docsearch.algolia.com and evaluating another solution by a large company in parallel I could not help to search github and the web for the current state of search engines and search related services.

Since I had done the same thing about a year ago, I was surprised to see how quickly things are moving atm.

Algolia

I was blown away by the quality of https://www.algolia.com and I wish it was open source, but I guess, we all have to make a living ;)

To see how awesome a web (search) interface can be check out https://www.lacoste.com/us/#query=red%20jackets%20for%20men

Apart from that the UI/UX of their backend tools is fantastic.

Elastic

When it comes to https://www.elastic.com I am a bit nervous about the future of the licensing, despite the fact, that I understand their motivation. At the same time the https://opensearch.org does not seem to be an ampty threat.

typesense.org

I do not know, who was hiding under a rock, but I had not seen https://typesense.org before and they certainly have a bold claim: "The Open Source Algolia Alternative" / "The Easier To Use ElasticSearch Alternative"

When looking at https://github.com/typesense/typesense/graphs/contributors it seems, that Kishore Nallan has been working on this for a while. Unfourtunately I do not really see a lot of external contributions, C++ does not seem to attract a lot of contribution.

MeiliSearch

This Rust project https://www.meilisearch.com/ seems to be picking up speed and is definetly on the test short list. It is a fresh codebase with siginficant open source contributions and certainly will attract new developers with Rust and a modern architecture.

Go eco system

Obviously we are very interested in Go powered software and there are a few notable projects. ATM I do not see anything elastic or algolia like, that would be really mature.

bleve / bluge

Marty Schoch seems to be the man when it comes down to text indexing libraries in written in Go and bluge seems to be THE library, that is solid and modern, when implementing text search in your Go application.

https://github.com/blevesearch/bleve +

One post tagged with "search-engine"

View All Tags

Jan Halfar

While building this website and integrating https://docsearch.algolia.com and evaluating another solution by a large company in parallel I could not help to search github and the web for the current state of search engines and search related services.

Since I had done the same thing about a year ago, I was surprised to see how quickly things are moving atm.

Algolia

I was blown away by the quality of https://www.algolia.com and I wish it was open source, but I guess, we all have to make a living ;)

To see how awesome a web (search) interface can be check out https://www.lacoste.com/us/#query=red%20jackets%20for%20men

Apart from that the UI/UX of their backend tools is fantastic.

Elastic

When it comes to https://www.elastic.com I am a bit nervous about the future of the licensing, despite the fact, that I understand their motivation. At the same time the https://opensearch.org does not seem to be an ampty threat.

typesense.org

I do not know, who was hiding under a rock, but I had not seen https://typesense.org before and they certainly have a bold claim: "The Open Source Algolia Alternative" / "The Easier To Use ElasticSearch Alternative"

When looking at https://github.com/typesense/typesense/graphs/contributors it seems, that Kishore Nallan has been working on this for a while. Unfourtunately I do not really see a lot of external contributions, C++ does not seem to attract a lot of contribution.

MeiliSearch

This Rust project https://www.meilisearch.com/ seems to be picking up speed and is definetly on the test short list. It is a fresh codebase with siginficant open source contributions and certainly will attract new developers with Rust and a modern architecture.

Go eco system

Obviously we are very interested in Go powered software and there are a few notable projects. ATM I do not see anything elastic or algolia like, that would be really mature.

bleve / bluge

Marty Schoch seems to be the man when it comes down to text indexing libraries in written in Go and bluge seems to be THE library, that is solid and modern, when implementing text search in your Go application.

https://github.com/blevesearch/bleve https://github.com/blugelabs/bluge // next iteration of bleve

projects using bluge

All bleeding edge afaik atm - but definitely good places to look at bluge usage

https://github.com/prabhatsharma/zinc https://github.com/mosuka/phalanx

Look ma I made a vector database

Gotta take a look at this one - will report later

https://github.com/semi-technologies/weaviate

- - + + \ No newline at end of file diff --git a/blog/tags/search.html b/blog/tags/search.html index bc854df..1c8d968 100644 --- a/blog/tags/search.html +++ b/blog/tags/search.html @@ -9,15 +9,15 @@ - - + +
-

One post tagged with "search"

View All Tags

Jan Halfar

While building this website and integrating https://docsearch.algolia.com and evaluating another solution by a large company in parallel I could not help to search github and the web for the current state of search engines and search related services.

Since I had done the same thing about a year ago, I was surprised to see how quickly things are moving atm.

Algolia

I was blown away by the quality of https://www.algolia.com and I wish it was open source, but I guess, we all have to make a living ;)

To see how awesome a web (search) interface can be check out https://www.lacoste.com/us/#query=red%20jackets%20for%20men

Apart from that the UI/UX of their backend tools is fantastic.

Elastic

When it comes to https://www.elastic.com I am a bit nervous about the future of the licensing, despite the fact, that I understand their motivation. At the same time the https://opensearch.org does not seem to be an ampty threat.

typesense.org

I do not know, who was hiding under a rock, but I had not seen https://typesense.org before and they certainly have a bold claim: "The Open Source Algolia Alternative" / "The Easier To Use ElasticSearch Alternative"

When looking at https://github.com/typesense/typesense/graphs/contributors it seems, that Kishore Nallan has been working on this for a while. Unfourtunately I do not really see a lot of external contributions, C++ does not seem to attract a lot of contribution.

MeiliSearch

This Rust project https://www.meilisearch.com/ seems to be picking up speed and is definetly on the test short list. It is a fresh codebase with siginficant open source contributions and certainly will attract new developers with Rust and a modern architecture.

Go eco system

Obviously we are very interested in Go powered software and there are a few notable projects. ATM I do not see anything elastic or algolia like, that would be really mature.

bleve / bluge

Marty Schoch seems to be the man when it comes down to text indexing libraries in written in Go and bluge seems to be THE library, that is solid and modern, when implementing text search in your Go application.

https://github.com/blevesearch/bleve +

One post tagged with "search"

View All Tags

Jan Halfar

While building this website and integrating https://docsearch.algolia.com and evaluating another solution by a large company in parallel I could not help to search github and the web for the current state of search engines and search related services.

Since I had done the same thing about a year ago, I was surprised to see how quickly things are moving atm.

Algolia

I was blown away by the quality of https://www.algolia.com and I wish it was open source, but I guess, we all have to make a living ;)

To see how awesome a web (search) interface can be check out https://www.lacoste.com/us/#query=red%20jackets%20for%20men

Apart from that the UI/UX of their backend tools is fantastic.

Elastic

When it comes to https://www.elastic.com I am a bit nervous about the future of the licensing, despite the fact, that I understand their motivation. At the same time the https://opensearch.org does not seem to be an ampty threat.

typesense.org

I do not know, who was hiding under a rock, but I had not seen https://typesense.org before and they certainly have a bold claim: "The Open Source Algolia Alternative" / "The Easier To Use ElasticSearch Alternative"

When looking at https://github.com/typesense/typesense/graphs/contributors it seems, that Kishore Nallan has been working on this for a while. Unfourtunately I do not really see a lot of external contributions, C++ does not seem to attract a lot of contribution.

MeiliSearch

This Rust project https://www.meilisearch.com/ seems to be picking up speed and is definetly on the test short list. It is a fresh codebase with siginficant open source contributions and certainly will attract new developers with Rust and a modern architecture.

Go eco system

Obviously we are very interested in Go powered software and there are a few notable projects. ATM I do not see anything elastic or algolia like, that would be really mature.

bleve / bluge

Marty Schoch seems to be the man when it comes down to text indexing libraries in written in Go and bluge seems to be THE library, that is solid and modern, when implementing text search in your Go application.

https://github.com/blevesearch/bleve https://github.com/blugelabs/bluge // next iteration of bleve

projects using bluge

All bleeding edge afaik atm - but definitely good places to look at bluge usage

https://github.com/prabhatsharma/zinc https://github.com/mosuka/phalanx

Look ma I made a vector database

Gotta take a look at this one - will report later

https://github.com/semi-technologies/weaviate

- - + + \ No newline at end of file diff --git a/blog/welcome-back-2021.html b/blog/welcome-back-2021.html index 4a6ec0b..f0a7a18 100644 --- a/blog/welcome-back-2021.html +++ b/blog/welcome-back-2021.html @@ -9,13 +9,13 @@ - - + +
-

Relaunching foomo.org

Jan Halfar

A few years ago we abandoned the previous version of https://www.foomo.org as we did not want to maintain the old wordpress installation and the project was only living in README.md in the repos living under https://www.github.com/foomo .

As things have grown over time we have decided to re-launch a website / cross project documentation.

So welcome back and enjoy the view to the past:

blast from the past

- - +

Relaunching foomo.org

Jan Halfar

A few years ago we abandoned the previous version of https://www.foomo.org as we did not want to maintain the old wordpress installation and the project was only living in README.md in the repos living under https://www.github.com/foomo .

As things have grown over time we have decided to re-launch a website / cross project documentation.

So welcome back and enjoy the view to the past:

blast from the past

+ + \ No newline at end of file diff --git a/blog/why-bundle-size-is-important.html b/blog/why-bundle-size-is-important.html index 4da7d1d..25013a4 100644 --- a/blog/why-bundle-size-is-important.html +++ b/blog/why-bundle-size-is-important.html @@ -9,16 +9,16 @@ - - + +
-

Why bundle size is important?

Nicola Turcato

Intro

JavaScript is parsed, compiled and executed in the main thread of the browser. Which means that users have to wait for all of this before they can interact with the website.

Frontend performance optimization is critical because it accounts for around 80-90% of user response time (10-20% backend). +

Why bundle size is important?

Nicola Turcato

Intro

JavaScript is parsed, compiled and executed in the main thread of the browser. Which means that users have to wait for all of this before they can interact with the website.

Frontend performance optimization is critical because it accounts for around 80-90% of user response time (10-20% backend). So when a user is waiting for a page to load, around 80-90% of the time is due to frontend related code and assets.

Nobody likes waiting…

A study found that if a site takes longer than 4 seconds to load, up to 25% of users would abandon the site.

Sending large JavaScript payloads impacts the speed of your site significantly.

Mazzarri

What is a "bundle"?

Your frontend application needs a bunch of JS files to run. These files can be in the format of internal dependency like the JS files you have written yourself. But they can also be external dependencies you use to build your application.

JS bundling is an optimization technique to reduce the number of server requests for JavaScript files. Bundling accomplishes this by merging multiple JA files together into one file to reduce the number of page requests.

Bundle Everywhere

Performance implications

  • Time to transmit over the network: considering slow connections with some mobile devices, it's possible that your page will not be interactive until it loads.-> More bytes = longer download times
  • JS parse and compile time: more code you load, more the browser must parse.-> JS gets parsed and compiled on the main thread, when the main thread is busy, the page can't respond to user input
  • JS execution time: optimally you will only pack the code that you expect to execute. The more code you want to execute the longer it will take. It's possible that your page won't be interactive until some of this completes.-> JS is also executed on the main thread, if your page runs a lot of code before it's really needed, that also delays your Time to Interactive
  • Memory consumption: everything fills up the space -> code itself, runtime variables, DOM elements created, etc.-> Pages appear slow when it consumes a lot of memory. Memory leaks can cause your page to freeze up completely!!

AS SMALL AS POSSIBLE! I experienced that is not really possible to give a precise answer because each application is different. Generally you want the resources on initial load to be as small as possible, so that you decrease the initial load time, and then load more resources as needed on the fly.

Mr Chao

What do we do then?

Meh

How to start decreasing the bundle size?

  • Measure: first of all you want to measure. The first step is to use Lighthouse and try to understand the results. It will give you a couple of interesting metrics and some tips. Time to interactive (TTI) is a good reflection of your bundle size because your bundle needs to be evaluated entirely before a user can interact with your web app.
  • Analyze: Consists on analyzing the bundle in order to detect critical chunks. A useful tool is Webpack Bundle Analyzer.

Stonks

Breaking up the bundle...

  • Monitor network requests: These happens between our FCP and TTI. As the initial request for data often occurs when our components initially mount.
  • Reduce the total dom nodes: the less the page needs to render, the less time it takes.
  • Moving work off the main thread: By moving heavy computations to a web worker, the computation will be run on a separate thread, and not block the actual rendering of the page
  • Caching: Even if not useful for users on first page landing, caching data, bundles, and assets can make subsequent visits way fast

Breaking Bad

Which strategies can we adopt?

  • Minification and Dead Code Elimination: These processes are often summed up as minifying or uglifying.
  • Tree shaking: Tree shaking is dead code elimination on a project or library. Always try to use deps which support “tree shaking”, Bundlephobia could be your friend in this case.
  • Code Splitting and Lazy Loading: Code splitting consists on taking a collection of modules and remove them from the main JS bundle. Lazy loading means we can load this newly created bundle later on.
  • Replace/rewrite large dependencies: Consider replacing or rewriting libraries that are large in size where you might not need all of its functionalities (Moment.js for example).
  • Feature module import: Check to see if you are using only a feature module of the library that can be imported alone without importing the whole library (Lodash for example).

Strategy

Useful tools to help you reducing bundle size

  • Lighthouse: automated tool for improving the performance, quality, and correctness of your web apps
  • Bundlephobia: Bundlephobia helps you find the performance impact of npm packages
  • Webpack Bundle Analyzer: analyzes your bundle
  • VS Code: Import Cost plugin -> Display import/require package size in the editor

Tools

Conclusion

Performance cannot be stripped down to a single metric such as bundle size. It would be great! Unfortunately there is no single place to measure all of them.I think metrics like the Core Web Vitals and a general look at bundle size should be considered as a starting point. -You will cry... A lot... But don’t give up!

The End

- - +You will cry... A lot... But don’t give up!

The End

+ + \ No newline at end of file diff --git a/docs/backend/go-by-example/cli-applications.html b/docs/backend/go-by-example/cli-applications.html index 32d5cc4..39dea6c 100644 --- a/docs/backend/go-by-example/cli-applications.html +++ b/docs/backend/go-by-example/cli-applications.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/backend/go-by-example/context.html b/docs/backend/go-by-example/context.html index dc72858..41378a4 100644 --- a/docs/backend/go-by-example/context.html +++ b/docs/backend/go-by-example/context.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/backend/go-by-example/defer.html b/docs/backend/go-by-example/defer.html index cf57f64..839671c 100644 --- a/docs/backend/go-by-example/defer.html +++ b/docs/backend/go-by-example/defer.html @@ -9,8 +9,8 @@ - - + +
@@ -22,7 +22,7 @@ What would happen if we add the defer keyword before each of our prints?

It's good practice to use defer to close a response body right after you have opened it. In this way you make sure all resources that have been opened have been closed and in this way it can prevent some bugs. What you will see often is something like this:

func main() {
resp, err := http.Get("http://www.foomo.org")
if err != nil {
log.Fatal(err)
}
defer resp.Body.Close()
}

You open a resource, check if there is an error and after it use defer to close it before the function returns.

Another thing to keep in mind with defer is that it will execute with the value given to it at the time. To demonstrate:

func main() {
myVar := "hello"
defer fmt.Println(myVar)
myVar = "goodbye"
}

What do you think this prints? You might think it will print 'goodbye' because that is the last value of the variable before the main function returns. Surprisingly the value printed is 'hello', which is likely the result of a compiler optimization.

- - + + \ No newline at end of file diff --git a/docs/backend/go-by-example/embed.html b/docs/backend/go-by-example/embed.html index 01d08ae..b23f850 100644 --- a/docs/backend/go-by-example/embed.html +++ b/docs/backend/go-by-example/embed.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/backend/go-by-example/files.html b/docs/backend/go-by-example/files.html index 43b754b..e2e524d 100644 --- a/docs/backend/go-by-example/files.html +++ b/docs/backend/go-by-example/files.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/backend/go-by-example/goroutines-and-channels.html b/docs/backend/go-by-example/goroutines-and-channels.html index 732dd4d..d3917bc 100644 --- a/docs/backend/go-by-example/goroutines-and-channels.html +++ b/docs/backend/go-by-example/goroutines-and-channels.html @@ -9,8 +9,8 @@ - - + +
@@ -26,7 +26,7 @@ go into a waiting state once it has sent the data, until this data is read by th So until the data is read of the channel this go routine is blocked.

Buffered Channels

You can set a limit on the capacity of a channel by passing the limit to the make function:

func main() {
myChannel := make(chan string, 5)
}

To demonstrate how an unbuffered channel only allows for synchronous communication consider the following example:

... loading sources

You see that even though we made both go routines wait when the channel was read, the go routine using the buffered channel printed out all of the statements at once. And the go routine using the unbuffered channel had to wait each time until the channel was read out to be able to continue singing the greatest song of all time.

- - + + \ No newline at end of file diff --git a/docs/backend/go-by-example/http.html b/docs/backend/go-by-example/http.html index e42989b..7d40b51 100644 --- a/docs/backend/go-by-example/http.html +++ b/docs/backend/go-by-example/http.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/backend/go-by-example/interfaces.html b/docs/backend/go-by-example/interfaces.html index 21e9836..30f6f7e 100644 --- a/docs/backend/go-by-example/interfaces.html +++ b/docs/backend/go-by-example/interfaces.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/backend/go-by-example/map-racing.html b/docs/backend/go-by-example/map-racing.html index 53a101d..e5361cd 100644 --- a/docs/backend/go-by-example/map-racing.html +++ b/docs/backend/go-by-example/map-racing.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/backend/go-by-example/mongodb.html b/docs/backend/go-by-example/mongodb.html index 58e0f22..2cac394 100644 --- a/docs/backend/go-by-example/mongodb.html +++ b/docs/backend/go-by-example/mongodb.html @@ -9,14 +9,14 @@ - - + +

MongoDB

Make sure to use the official golang mongo driver:

go.mongodb.org/mongo-driver/mongo

This documentation is a simplified version of the tutorial on MongoDB's website

Set client connections

clientOptions := options.Client().ApplyURI(<MongoDB-database-URI>)

Connect with the database

client, err := mongo.Connect(context.TODO(), clientOptions)

Check connection

err = client.Ping(context.TODO(), nil)

A collection is a grouping of MongoDB documents. Documents within a collection can have different fields. A collection is the equivalent of a table in a relational database system. A collection exists within a single database

source: Geeks for Geeks https://www.geeksforgeeks.org/mongodb-database-collection-and-document/

JSON documents in MongoDB are stored in a binary representation called BSON (Binary-encoded JSON). Unlike other databases that store JSON data as simple strings and numbers, the BSON encoding extends the JSON representation to include additional types such as int, long, date, floating point, and decimal128. This makes it much easier for applications to reliably process, sort, and compare data. The Go Driver has two families of types for representing BSON data: The D types and the Raw types. The D family of types is used to concisely build BSON objects using native Go types:

  • D: A BSON document. This type should be used in situations where order matters, such as MongoDB commands.
  • M: An unordered map. It is the same as D, except it does not preserve order.
  • A: A BSON array.
  • E: A single element inside a D.

You use filters to get the specific data you ask for.

CRUD Operations

Insert one item

insertResult, err := collection.InsertOne(context.TODO(), <item>)

Insert Many

this takes a slice of objects

my-slice := []interface{}{item1, item2}
insertManyResult, err := collection.InsertMany(context.TODO(), my-slice)

Update a single document

Requires a filter document to match documents in the database and an update document to describe the update operation.

filter := bson.D{{"name", “item1”}}
update := bson.D{
{"$inc", bson.D{ // Increases var1 with 1
{“var1”, 1},
}},
}
updateResult, err := collection.UpdateOne(context.TODO(), filter, update)

Find a single document

Requires a filter document and a pointer to a value into which the result can be decoded, returns a single result which can be decoded into a value.

var item_result Item // value into which the result can be decoded
err = collection.FindOne(context.TODO(), filter).Decode(&result)

Find multiple documents

This method returns a Cursor. A Cursor provides a stream of documents through which you can iterate and decode one at a time. Once a Cursor has been exhausted, you should close the Cursor.

findOptions := options.Find()
var results []*Item
cur, err := collection.Find(context.TODO(), bson.D{{}}, findOptions)
for cur.Next(context.TODO()) {
var elem Item
err := cur.Decode(&elem)
if err != nil {
log.Fatal(err)
}
results = append(results, &elem)
}
cur.Close(context.TODO())

Delete documents

You can choose to use collection.DeleteOne() or collection.DeleteMany(), both take a filter document to match the documents in the collection.

deleteResult, err := collection.DeleteMany(context.TODO(), bson.D{{}})

Close connection

Best to keep connection open if you have to do multiple things. It does not make sense to open and close a connection if you have to query the database multiple times.

err = client.Disconnect(context.TODO())
- - + + \ No newline at end of file diff --git a/docs/backend/go-by-example/nil-maps.html b/docs/backend/go-by-example/nil-maps.html index bacca44..204871e 100644 --- a/docs/backend/go-by-example/nil-maps.html +++ b/docs/backend/go-by-example/nil-maps.html @@ -9,8 +9,8 @@ - - + +
@@ -22,7 +22,7 @@ without a problem to the other. Map1 is a nil map, because it has not b with the make statement you can add elements to this map later on in your program. Nil maps could for example be used for structs that have optional data. In that case we don't want to allocate them every time, only when the data is present.


While you can not add elements to a nil map, read operations work just fine. Try it out yourself in the following example:

... loading sources
- - + + \ No newline at end of file diff --git a/docs/backend/go-by-example/panic-and-recover.html b/docs/backend/go-by-example/panic-and-recover.html index 222913e..1347067 100644 --- a/docs/backend/go-by-example/panic-and-recover.html +++ b/docs/backend/go-by-example/panic-and-recover.html @@ -9,8 +9,8 @@ - - + +
@@ -19,7 +19,7 @@ So the order of execution is: Execute called function, then execute defer statements, and in case of a panic program execution will stop. So even if your application will panic defer statements are handled and therefore resources will be closed accordingly.

Panic and Recover

Try out the following example:

... loading sources
The function that panics will stop because it can no longer function so we don't see 'Is this printed?' printed. The recover function makes it possible to obtain the error in case a panic occurred, and leaves the decision how to proceed to the programmer. In this example, we decided to just log the error and continue to run. The function that panicked will stop execution at the panic, but the code that called it will continue to execute, which is why we see 'End' being printed.
- - + + \ No newline at end of file diff --git a/docs/backend/go-by-example/ranging.html b/docs/backend/go-by-example/ranging.html index d386bb0..8b941f6 100644 --- a/docs/backend/go-by-example/ranging.html +++ b/docs/backend/go-by-example/ranging.html @@ -9,14 +9,14 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/backend/go-by-example/reflection.html b/docs/backend/go-by-example/reflection.html index 478ff5d..bafea97 100644 --- a/docs/backend/go-by-example/reflection.html +++ b/docs/backend/go-by-example/reflection.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/backend/go-by-example/type-casting.html b/docs/backend/go-by-example/type-casting.html index 35569fb..2eb08d1 100644 --- a/docs/backend/go-by-example/type-casting.html +++ b/docs/backend/go-by-example/type-casting.html @@ -9,14 +9,14 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/backend/go-by-example/yaml-magic.html b/docs/backend/go-by-example/yaml-magic.html index 1d1d287..8515351 100644 --- a/docs/backend/go-by-example/yaml-magic.html +++ b/docs/backend/go-by-example/yaml-magic.html @@ -9,14 +9,14 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/backend/gograpple.html b/docs/backend/gograpple.html index 2ee57b7..fd2a78b 100644 --- a/docs/backend/gograpple.html +++ b/docs/backend/gograpple.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/backend/intro.html b/docs/backend/intro.html index 341fcb8..97b25e0 100644 --- a/docs/backend/intro.html +++ b/docs/backend/intro.html @@ -9,14 +9,14 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/backend/rtfm.html b/docs/backend/rtfm.html index b2eb499..7f83e67 100644 --- a/docs/backend/rtfm.html +++ b/docs/backend/rtfm.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/backend/setup.html b/docs/backend/setup.html index 0ef0cda..a548d0f 100644 --- a/docs/backend/setup.html +++ b/docs/backend/setup.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/devops/bookmarks.html b/docs/devops/bookmarks.html index 0dd5837..805765d 100644 --- a/docs/devops/bookmarks.html +++ b/docs/devops/bookmarks.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/devops/infrastructure/conventions.html b/docs/devops/infrastructure/conventions.html index 847cd9d..aba5311 100644 --- a/docs/devops/infrastructure/conventions.html +++ b/docs/devops/infrastructure/conventions.html @@ -9,13 +9,13 @@ - - + +

Conventions

When creating resources the most important thing is to be consistent.

Resource Naming Conventions

The following resource convention is preferred if no other exist in the project:

[customer*]-[project]-[resource*]-[env*]-[name*]-[region/location]-[index]

The fields with a * are required.

Legend:

customer:          bestbytes, foomo
project: hmd
reqource: s3, pql, vpc
env: prod, stage, dev, int, test
name: default, general, catalogue-backend
region/location: region for this resource if it's region specific
index: numbering of the resource, starting from 1 (3 numbers, leading zeros)

Examples:

# Foomo customer s3 bucket for prod environment with the described role (catalogue-backend/site-images)
foomo-s3-prod-NAME

- foomo-s3-prod-catalogue-backend
- foomo-s3-prod-site-images

# Foomo customer VPC for stage environment with the described role (default/mongodb)
foomo-vpc-stage-NAME

- foomo-vpc-stage-default
- foomo-vpc-stage-mongodb

# Bestbytes customer for HMD project, postgresql instance for stage environment and default usage
bestbytes-hmd-pql-stage-default

# Bestbytes customer VPN Tunnel #1 in the stage environment for externalcustomer with numbering
bestbytes-vpt-stage-externalcustomer-001
- - + + \ No newline at end of file diff --git a/docs/devops/intro.html b/docs/devops/intro.html index 9b0b0f6..ea02a3e 100644 --- a/docs/devops/intro.html +++ b/docs/devops/intro.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/devops/k8s.html b/docs/devops/k8s.html index 7d32ef8..c013ea7 100644 --- a/docs/devops/k8s.html +++ b/docs/devops/k8s.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/devops/kubernetes/spot-instance-node-pools.html b/docs/devops/kubernetes/spot-instance-node-pools.html index 3d53e8a..82d6a96 100644 --- a/docs/devops/kubernetes/spot-instance-node-pools.html +++ b/docs/devops/kubernetes/spot-instance-node-pools.html @@ -9,8 +9,8 @@ - - + +
@@ -18,7 +18,7 @@ What could happen is that nodes that are hosting the applications start terminating, and our application becomes unresponsive until the application is re-located to another node. To avoid that situation, for the instances we need to configure pod disruption budget on our helm deployments.

apiVersion: policy/v1
kind: PodDisruptionBudget
metadata:
name: application-name
spec:
maxAvailable: 50%
selector:
matchLabels:
app: application-name

For more details check out here

Setting Up Termination Handling

Setting Up K8s Cron Shutdown Cleanup

- - + + \ No newline at end of file diff --git a/docs/devops/monitoring/grafana.html b/docs/devops/monitoring/grafana.html index 153758a..0367633 100644 --- a/docs/devops/monitoring/grafana.html +++ b/docs/devops/monitoring/grafana.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/devops/monitoring/intro.html b/docs/devops/monitoring/intro.html index e84e8a6..a4ea10d 100644 --- a/docs/devops/monitoring/intro.html +++ b/docs/devops/monitoring/intro.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/devops/monitoring/jaeger.html b/docs/devops/monitoring/jaeger.html index eaa46e1..d25e8e5 100644 --- a/docs/devops/monitoring/jaeger.html +++ b/docs/devops/monitoring/jaeger.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/frontend/css.html b/docs/frontend/css.html index 9579bf6..2d0a531 100644 --- a/docs/frontend/css.html +++ b/docs/frontend/css.html @@ -9,13 +9,13 @@ - - + +
- - + + \ No newline at end of file diff --git a/docs/frontend/debugging_js.html b/docs/frontend/debugging_js.html index 22b1007..e6ba75b 100644 --- a/docs/frontend/debugging_js.html +++ b/docs/frontend/debugging_js.html @@ -9,13 +9,13 @@ - - + +

Debugging in JavaScript

Debugging experience is very important because it can save you hours of depressing search for a bug.

console logging

Probably 95% of devs nowadays still use console.log since it's the most convenient way of outputting something. There are also some nicer ways to output something, especially when having an array of object. In that case it's much nicer to use console.table(array) because it will create a nice table with all fields.

Sometimes a debugger in the browser won't create a breakpoint like you want it or you can't find a source file where you would want. A nice trick is to add a debugger keyword in your code which will instantly stop executing script on that place (but only if you have devtools open).

Bisecting

When you can't find what causes an error, a very common approach is doing bisection. This is simply removing half of the code and see if it still works or not. If it works, then the bug is in the removed code, otherwise start removing other half in existing code.

Chrome/Firefox devtools

Today's browsers have really good devtools and we can't imagine developing without them. And they all nice of helper tricks that ease our development and especially debugging.

... TODO: @themre

- - + + \ No newline at end of file diff --git a/docs/frontend/intro.html b/docs/frontend/intro.html index 14cbc07..ea837cf 100644 --- a/docs/frontend/intro.html +++ b/docs/frontend/intro.html @@ -9,13 +9,13 @@ - - + +

Frontend Guide

In this section you will find a guide on how to develop frontends with the foomo stack. Please also see the general guide.

Let me share a few thoughts on frontend development from our journey @bestbytes the company behind https://www.foomo.org .

Web development has been a wild ride since the beginning of the www in the mid 1990s. Today we take it as granted, that the default way to build applications is with web technologies.

Not too long ago this seemed absurd, since browsers were meant to share scientific research papers and therefore were extremely limited in their capabilities.

It was the era of experimentation, that turned the web into the application runtime, which it is today. Plugins like Macromedias flash inspired the rich APIs we can use today, when building web applications.

A little history

The current stack is the result of a more than 20 year journey.

Last millenium

The open source eco system was in its infancy, there was no github (is that really open source as it should be 🤔), there was not even git, there were no real web frameworks and agile methodologies were an exotic nerd topic.

Browsers would sometimes render things in a recognisable way and a table was the way to render your layout. Reliable interactivity could only result from a server round trip and calling JavaScript had a good chance to crash your browser.

Security in browsers was a complete nightmare, it was a time, when one tab could access anothers tabs dom 🤯

Early 2000s

As Microsoft had won the browser wars, we were stuck with IE 6 and like others we were building web applications with the limited resources we had. JavaScript frameworks were still

The plugin era / FLASH

As browsers were slow, buggy and inconsistent FLASH presented this amazing way to build highly interactive websites. It was the time of loaders, which people accepted, because amazing experiences awaited them: animated vector graphics (artists went wild ...), games, the first videos, client server communications, sockets and a canvas API.

Also ActionScript especially AS3 felt so superior to JavaScript at the time.

Flex

Flex used the capabilities of the flash player to provide a fantastic developer experience to build RIAs (rich internet applications). The resulting productivity, application performance and features were unparalled in the JavaScript eco system until long after the demise of the flash player.

After "thoughts on flash"

After the release of the iPhone and the beginning of the walled garden era flash was one of the initial victims. Just for the record, I definitely prefer an independent eco system, which the web is at least to a certain extent, but it certainly was not noble motives, that were driving Apple when pushing for HTML5 in their thoughts on flash.

HTML5

At its arrival HTML5 could not deliver on the many promises, that were made around it. Browser compatibilty was a nightmare, JavaScript performance was poor, the eco system was in its infancy.

As we left the flash eco system we stopped by RequireJS and were using BACKBONE.JS as we felt extremely burnt by Adobe, after they had dropped their whole eco system like a hot potatoe once Apple had declared the end of flash.

TypeScript and the wild framework years

Despite the fact, that we were very sceptic about trusting large vendors, we adopted TypeScript right after it was initially released and we have remained happy users of it until the present day.

The early years of the Node.js community were extremely wild, there was jokes about the number of frameworks released per day. The number of languages that compiled to JavaScript exploded and it just seemed impossible to keep up with the changing environment.

2020s

As we wake up today as frontend developers, we do not expect, that we will keep on doing things as we do them today until the end of our careers, but we at least see a chance, that the code base of a project will not be outdated with delivery of a MVP.

- - + + \ No newline at end of file diff --git a/docs/frontend/performance.html b/docs/frontend/performance.html index 1dfa780..61dd44b 100644 --- a/docs/frontend/performance.html +++ b/docs/frontend/performance.html @@ -9,14 +9,14 @@ - - + +

Performance

JS

JS is nowadays extremely fast and yet we have many performance issues. Here you can find few common mistakes that occur in JS that can decrease performance.

Extensive use of .map, .filter

Let's say you have a large list of objects and you would like to filter them and transform them in some form. Usually we do this:

const largeArray = [ .... ]
largeArray.map(obj => transformObj(obj)).filter(omitBadObject)

In the above case we first loop through whole set, transform it and then filter things out. Not only does this goes through all the items twice, but it also first time goes through all the items and then filters them. One optimization would be to first filter them and then transform them, but ideally we should just use a normal for loop or forEach where you go through items only once.

const finalArray = []
largeArray.forEach(obj => {
if (omitBadObject(obj)) {
finalArray.push(transformObj(obj))
}
})

This code will skip another loop of items.

- - + + \ No newline at end of file diff --git a/docs/frontend/pitfalls.html b/docs/frontend/pitfalls.html index 658bbab..44a938a 100644 --- a/docs/frontend/pitfalls.html +++ b/docs/frontend/pitfalls.html @@ -9,13 +9,13 @@ - - + +

Pitfalls

This section is very important to avoid dangerous bugs in your code. Of course each of our languages or libraries in our stack has many advantages, but they came with dangerous pitfalls.

JavaScript

This section is probably too small to write all the dangerous things in JavaScript, but we hope developers won't do some crazy things like concatenating numbers and strings and such sort of forbidden things will be catched by TS.

But nevertheless, very often we run into issues that are hard to debug.

Mutation

Mutation in JS is a very common problem and thus leads to dangerous bugs.

A very common mutation occurs in spreading. So if you spread an object and it's deeply nested, you need to know that this will only create a shallow copy. So that means that deeply nested fields will still be referenced by the original object. If you need to do a immutable instance we advice in using either klona or immer.

When doing sorting on the array like [1,2,3].sort(...), this does mutate the original array. Devs are used to create new array with .map or .filter, but this here is not the case. So be careful with that. Also do not forget that objects in the array will have a reference to objects in the original array and thus they also needs to be cloned.

Have in mind that doing immutability costs and sometimes it's better to mutate, but also know that in React if you don't create a new reference, setting state won't react. So always have in mind pros and cons of doing mutation.

typeof

typeof is used for checking type of some variables. So when checking if a variable is an array of an object, we try to use typeof. But we should be very careful since typeof works nicely on simple primitive types like number, string, boolean.

When you try to check whether a variable is an object or an array, you try to use typeof. But both are object. So in this case we should use Array.isArray(variable). But be careful, because also null is an object! If you want to deep dive into this, read here.

Numbers are also fun when it comes to NaN. If you check isNaN with typeof, it's actually a number! So for this use-case you need to use Number.isNaN(myNum).

- - + + \ No newline at end of file diff --git a/docs/frontend/rtfm.html b/docs/frontend/rtfm.html index 768a1c5..7025cff 100644 --- a/docs/frontend/rtfm.html +++ b/docs/frontend/rtfm.html @@ -9,13 +9,13 @@ - - + +

External documentation resources

Runtime / browser

In general the Mozilla Developer Network - MDN is a great resource, if you want to know, what your browser is capable of.

Once you found out, how great an API is, the question to answer will be "can I use" feature X. https://caniuse.com/ is a great resource to find the answers, when trying to understand, if a browser API / feature is ready to use for your audience.

JavaScript

JavaScript historically has a very bad reputation. Please enjoy the following classic piece of infotainment: https://www.destroyallsoftware.com/talks/wat

As JavaScript has been massively iterated on, the runtimes hav ecome crazy fast and the language keeps on evolving.

The (mostly functional) subset of the language, that is being used in modern JavaScript / TypeScript projects to is highly expressive and reads well.

Please take the time and learn about

  • functional programming
  • variable scope
  • modern JavaScript syntax
    • arrow functions
    • new keywords
    • spreading
    • ...
  • promises

CSS

  • CSS block model
  • flex box

Browser APIs

  • canvas
  • fetch
  • local storage
  • websockets
  • sse server side events
  • built in types

TypeScript

- - + + \ No newline at end of file diff --git a/docs/frontend/setup.html b/docs/frontend/setup.html index 47427a4..1f208cc 100644 --- a/docs/frontend/setup.html +++ b/docs/frontend/setup.html @@ -9,13 +9,13 @@ - - + +

Frontend setup

How to setup your machine for frontend development

general topics

setting up your IDE

  • vscode

frontend specific

  • yarn / npm / package.json
  • nvm
  • browser extensions
    • preact
    • redux
    • (google analytics)

debugging with emulators / devices

  • xcode
  • android studio

Install software

Mac

First of all install brew from https://brew.sh

brew install nvm
brew cask install iterm2

Linux

Windows

Install Linux ;)

- - + + \ No newline at end of file diff --git a/docs/frontend/stack.html b/docs/frontend/stack.html index 0b7f772..99f3461 100644 --- a/docs/frontend/stack.html +++ b/docs/frontend/stack.html @@ -9,13 +9,13 @@ - - + +

Frontend Stack

Our frontend stack is permanently changing as the underlying eco system does. We are trying to adopt technologies at the "Slope of Enlightenment" in the hype cycle.

Former iterations of our stack were MVC or MVVM frontend frameworks. Since the arrival of React and state mangement libraries like Redux and Zustand we have been building reactive applications.

Reactive frontends are fun to develop and they scale.

TypeScript

Is our language of choice to build frontends for web applications.

We have adopted TypeScript very early, right after its initial public release in 2012. Here is a list of motives:

  • it is a superset JavaScript
  • it does not try to replace JavaScript like many other languages, which compile to JavaScript, it tries to complement and improve it, while staying compatible
  • good type system
  • great tooling

While it seemed to be crazy to trust Microsoft as a friendly open source company, we have not been disappointed so far.

React / Preact

Declarative components and views, fast to develop and fast at runtime.

gotsrpc

Since we are building services in Go and not with Node.js we have created a light weight RPC framework to integrate TypeScript with Go.

When creating TS definitions on the Go side we follow camel case convention of naming fields e.g. lastDateModified instead of LastDateModified.

- - + + \ No newline at end of file diff --git a/docs/frontend/typescript/objects.html b/docs/frontend/typescript/objects.html index 97827dc..92aa7fe 100644 --- a/docs/frontend/typescript/objects.html +++ b/docs/frontend/typescript/objects.html @@ -9,15 +9,15 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/frontend/typescript/spreading.html b/docs/frontend/typescript/spreading.html index 1861f83..1d991dd 100644 --- a/docs/frontend/typescript/spreading.html +++ b/docs/frontend/typescript/spreading.html @@ -9,14 +9,14 @@ - - + +

Spreading

JavaScript spread syntax (...) is a surprisingly powerful construct. It has two main use cases in our applications:

Shallow copying

Spreading creates new instances of objects or array, but we need to be very careful because it only does a shallow copy. If you have a deeply nested object or array, nested entities will still hold a reference to an original value and hence dangerous bugs can occur.

Copying is needed when doing state changes (either local or state management e.g. Redux). If deep cloning is required, structuredClone can be used.

function () {
// let´s start with a relatively simple object
const foo = {
a: 1,
b: 2,
nested: {
child: "is not a copy",
},
};
// this will create a shallow copy of foo and add a field "c"
const bar = { ...foo, c: 3 };
// please note, that the nested property "nested" is not deeply copied,
// thus changes to this object will also affect foo.nested
bar.nested.child += " <- I told you so";

// .foo has been copied though and changes to it affect the original only
foo.a += 1;
return (
<>
<ul>
<li>
<code>foo.a</code> was incremented by 1 and is now <code>2</code>
</li>
<li>
accessing <code>bar.nested.child</code> also changed{" "}
<code>foo.nested.child</code>
</li>
</ul>
<pre>foo: {JSON.stringify(foo)}</pre>
<ul>
<li>
<code>bar.a</code> has kept its original value <code>1</code>
</li>
<li>
a new property <code>c</code> was added
</li>
</ul>
<pre>bar: {JSON.stringify(bar)}</pre>
</>
);
}

Populating jsx attributes

function App() {
// a simple Component, that will render all fields in props
const Foo = (props) => (
<ul>
{
Object
.keys(props)
.map(key => (
<li>
<code>{key}</code>:{props[key]}
</li>
)
)
}
</ul>
);
const data = {
a: 1,
b: 2,
c: 3
}

const { c, ...omittedObject } = data;
return (
<>
{/*
Each of the `data` fields are spreaded as props
*/}
<p>Spread all props</p>
<Foo {...data}/>

<p>Add additional prop</p>
{/*
After we spread, we also add `d` prop.
*/}
<Foo {...data} d={4}/>

<p>Order of props is important</p>
{/*
We will replace `a` with 10
*/}
<Foo {...data} a={10} />
{/*
But this won't replace original `a` with 10
*/}
<Foo a={10} {...data} />

<p>Omit certain properties</p>
{/*
If you wish to omit certain fields, you need to write fields
that you wish to omit and with `...{newVariableName}` you will create a new
object with that name. In our part it's `omittedObject` name.
*/}
<Foo {...omittedObject} />
</>
);
}
- - + + \ No newline at end of file diff --git a/docs/frontend/vscode.html b/docs/frontend/vscode.html index 65f289f..7d04be3 100644 --- a/docs/frontend/vscode.html +++ b/docs/frontend/vscode.html @@ -9,13 +9,13 @@ - - + +
- - + + \ No newline at end of file diff --git a/docs/general/essentials.html b/docs/general/essentials.html index 38562f6..6de982c 100644 --- a/docs/general/essentials.html +++ b/docs/general/essentials.html @@ -9,13 +9,13 @@ - - + +

(Random) essentials

Markdown

Learn markdown(this is a markdown document 🤓)

Vim

Vim is everywhere so no matter if you ssh into a place or try to edit a file in a container, you are going to need it.

Here is a super minimal cheat sheet:

# edit/create a file
vim path/file

Now the shortcuts:

shortcutfunctionnote
:h<enter>helpa bit overwhelming
:q<enter>quitimmediate instinct after looking at help 🤣
:q!<enter>quit ignoring changes! means, that you actually mean it and can be combined with a lot of commands
:w<enter>write / save file
:x<enter> save and exit
:x!<enter> force save and exit
11Ggo to line 11
$ go to last char in line
dddelete line
x delete char
i enter insert modethat is where you can actually edit stuff - press escape to exit insert mode
<esc>exit insert mode
<ctrl>vvisual blockthen use cursors to edit selection
xcut
ppaste

ssh

allows you to remotely and securely log into computers - man ssh should be available on any shell.

digital ocean has a decent intro into ssh https://www.digitalocean.com/community/tutorials/ssh-essentials-working-with-ssh-servers-clients-and-keys

There is a lot of other tutorials - your typical use cases will be:

  • logging into other computers
  • executing commands on those computers
  • tunneling network connections

Once ssh is set up correctly scp, git and a lot of other programs, that rely on ssh under the covers are fun to use too.

git

https://git-scm.com/doc

github

Yes despite the fact, that m$ bought them, we use them and you have to enable 2fa

Some data formats

yaml

https://yaml.org/ some love it for good a reason, some hate it for another good reason, but atm there is just no way around it.

json

https://www.json.org/json-en.html similar love and hate situation like with yaml - also no way out.

CSV

If you think parsing or serializing CSV is trivial, you have been warned.

https://datatracker.ietf.org/doc/html/rfc4180

protocol buffers

Fast, compact not human readable - not necessary for most use cases

https://developers.google.com/protocol-buffers

xml

https://www.w3.org/TR/REC-xml/ only little love here

- - + + \ No newline at end of file diff --git a/docs/general/intro.html b/docs/general/intro.html index 27657a1..17e139f 100644 --- a/docs/general/intro.html +++ b/docs/general/intro.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/general/package-managers.html b/docs/general/package-managers.html index bd6a517..7f0ed08 100644 --- a/docs/general/package-managers.html +++ b/docs/general/package-managers.html @@ -9,13 +9,13 @@ - - + +

Package managers

Whenever you can, do not manage software on your computer manually ...

  • comfort
  • versions
  • security

macOS

On the mac https://brew.sh/ has become the de facto standard - we use it to distribute binaries of some of our open source projects as well.

Windows

...

- - + + \ No newline at end of file diff --git a/docs/general/security.html b/docs/general/security.html index b7b28f9..5900e33 100644 --- a/docs/general/security.html +++ b/docs/general/security.html @@ -9,13 +9,13 @@ - - + +

Security

TODO: @dreadl0ck knock yourself out

OS / Application setup

  • disable telemetry whereever possible

Keep your data safe

  • encrypt your file system to make sure so that if your computer gets stolen at least your data remain secure
  • make sure to have multiple backups in different places i.e. one backup disk at home and one in the office
- - + + \ No newline at end of file diff --git a/docs/general/setup/computer.html b/docs/general/setup/computer.html index 20e5d5f..0bd4707 100644 --- a/docs/general/setup/computer.html +++ b/docs/general/setup/computer.html @@ -9,13 +9,13 @@ - - + +

Setting up your computer

This is a general setup guide for your computer. Make sure that you setup your worpkplace properly too.

Start

  • read this document before you start
  • only use a computer from a trustworthy source
  • reset the computer
  • encrypt your harddrive in order to be on the safe side, if a computer get stolen
  • upgrade to the latest version of the OS you want to use

Privacy and security

Follow the instructions on security when you setup and maintain your system.

Install software

Mac

First of all install brew from https://brew.sh

brew cask install iterm2

Linux

Windows

Install Linux ;)

- - + + \ No newline at end of file diff --git a/docs/general/setup/workplace.html b/docs/general/setup/workplace.html index bb76741..7ee2ee5 100644 --- a/docs/general/setup/workplace.html +++ b/docs/general/setup/workplace.html @@ -9,13 +9,13 @@ - - + +

Setting up your workplace

No matter where you work, you need a proper workplace:

  • ergonomic office chair / desk combination
  • large high pixel density display
  • ergonomic input devices
  • good lighting
  • correct position in the room

Make sure that you take the time to set up your workplace well, your health depends upon it.

- - + + \ No newline at end of file diff --git a/docs/general/technologies/sse.html b/docs/general/technologies/sse.html index 06d8cf7..c9eafe2 100644 --- a/docs/general/technologies/sse.html +++ b/docs/general/technologies/sse.html @@ -9,13 +9,13 @@ - - + +

SSE Server Sent Events

Server Sent Events SSE are a well established web standard.

They allow a web server to push messages to the client.

I have created a minimal example https://github.com/janhalfar/sse-playground/tree/main/minimal to illustrate the technology.

Minimal Example in action

if you clone https://github.com/janhalfar/sse-playground locally and run go run minimal/main.go you will be able to open http://localhost:8080 :

screenshot of chrome debugging SSE

Minimal server

This server will:

Let´s take a look at the server response with:

curl -v localhost:8080/sse/time

% curl -v localhost:8080/sse/time
* Trying 127.0.0.1:8080...
* Connected to localhost (127.0.0.1) port 8080 (#0)
> GET /sse/time HTTP/1.1
> Host: localhost:8080
> User-Agent: curl/7.79.1
> Accept: */*
>
* Mark bundle as not supporting multiuse
< HTTP/1.1 200 OK
< Content-Type: text/event-stream
< Date: Fri, 16 Sep 2022 14:49:17 GMT
< Transfer-Encoding: chunked
<
event: time
id: 1
data: 2022-09-16T16:49:17.482796+02:00

event: time
id: 2
data: 2022-09-16T16:49:17.498366+02:00

event: time
id: 3
data: 2022-09-16T16:49:17.514079+02:00

event: time
id: 4
data: 2022-09-16T16:49:17.529639+02:00

event: time
id: 5
data: 2022-09-16T16:49:17.545335+02:00

...

serving the frontend

The first thing the server is to serve a frontend, which is described below:

serve the embedded index.html
loading...

serving time events

serve an event stream of 100 server sent events
loading...

Minimal client

a minimal and incomplete client example
loading...
- - + + \ No newline at end of file diff --git a/docs/general/utilities/k9s.html b/docs/general/utilities/k9s.html index 4d0bba3..9df2875 100644 --- a/docs/general/utilities/k9s.html +++ b/docs/general/utilities/k9s.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/general/work/general.html b/docs/general/work/general.html index 504599c..4a2407b 100644 --- a/docs/general/work/general.html +++ b/docs/general/work/general.html @@ -9,13 +9,13 @@ - - + +

Working as a software developer

Working as a software developer in a sustainable way is challenging - here are a few observations of healthy habits:

  • do not work too much
  • physical exercise is essential
  • clearly separate between working and not working
  • reflect and iterate on your working habits
- - + + \ No newline at end of file diff --git a/docs/general/work/remote-work.html b/docs/general/work/remote-work.html index b765a32..e8f8e10 100644 --- a/docs/general/work/remote-work.html +++ b/docs/general/work/remote-work.html @@ -9,13 +9,13 @@ - - + +

Working remotely

Especially since the beginning of the corona pandemy working remotely has become a very important topic. The foomo team has extensive experience working remotely as we are a distributed team.

Disclaimer: we are almost exclusively working in one timezone.

Requirements

  • fast and reliable internet connection
  • a proper workplace

The good parts

  • no commute
  • less distractions from team members

The bad parts

  • harder to separate between work and your private life
  • less informal communication with teammates
  • less human interaction

When working from home

  • make sure, that you have a routine, that still takes you to work
    • fixed working hours
    • "go to work" and "leave work" even if you do not leave your home
    • if possible use a room, that is for work only
    • have clear rules for your family / roomates, when at work
  • when communication with teammates use video calls
  • take time for your teammates and have a virtual coffee break with someone at least once a day
  • invest half the time, that you save by not communiting to exercise
- - + + \ No newline at end of file diff --git a/docs/project-management/intro.html b/docs/project-management/intro.html index b78bd2d..81db383 100644 --- a/docs/project-management/intro.html +++ b/docs/project-management/intro.html @@ -9,13 +9,13 @@ - - + +

Project Managament

We scrumming in our way ...

Values

  • project managers manage projects not people

Epics

Epics are the starting point ....

- - + + \ No newline at end of file diff --git a/docs/project-management/responsibility.html b/docs/project-management/responsibility.html index b068f5b..4e28ee6 100644 --- a/docs/project-management/responsibility.html +++ b/docs/project-management/responsibility.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/projects/architecture.html b/docs/projects/architecture.html index ab9db6d..bf81991 100644 --- a/docs/projects/architecture.html +++ b/docs/projects/architecture.html @@ -9,13 +9,13 @@ - - + +

Architecture overview

backend services

When it comes to writing backend services we have decided for Go https://go.dev - let me list the main reasons why - Go is:

  • simple and fun
  • fast when compiling
  • fast at runtime
  • friendly to your machines
  • friendly to you as a programmer
  • equipped with a top notch runtime
  • extremely well balanced in its design as a language, that is highly consumable for humans and machines
  • not about the features it has
  • about what has been left out
  • easy to read
  • sustainable

Foomo projects supporting development with Go

  • keel - opinionated way to run services
  • gotsrpc - rpc framework / code generator
  • gograpple - human friendly way to debug go programs running in k8s
  • webgrapple - a development proxy

Frontends

Foomo projects supporting development with Next.js

- - + + \ No newline at end of file diff --git a/docs/projects/cms/contentful.html b/docs/projects/cms/contentful.html index 8ca3f07..9c21c83 100644 --- a/docs/projects/cms/contentful.html +++ b/docs/projects/cms/contentful.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/projects/cms/contentserver.html b/docs/projects/cms/contentserver.html index a525896..10e94e2 100644 --- a/docs/projects/cms/contentserver.html +++ b/docs/projects/cms/contentserver.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/projects/cms/gocontentful/api-reference.html b/docs/projects/cms/gocontentful/api-reference.html index f138c88..c7e7a1a 100644 --- a/docs/projects/cms/gocontentful/api-reference.html +++ b/docs/projects/cms/gocontentful/api-reference.html @@ -9,8 +9,8 @@ - - + +
@@ -57,7 +57,7 @@ set it will substitute the IMG tag with the returned HTML snippet. This allows y images, e.g. a PICTURE tag.

type LinkResolverFunc func(url string) (resolvedAttrs map[string]string, resolveError error)

type EntryLinkResolverFunc func(entryID string, locale Locale) (resolvedAttrs map[string]string, resolveError error)

type ImageResolverFunc func(assetID string, locale Locale) (attrs map[string]string, customHTML string, resolveError error)

type EmbeddedEntryResolverFunc func(entryID string, locale Locale) (html string, resolveError error)

All the three functions above can be passed as nil with different levels of graceful degrading.

Constants and global variables

Each generated content type library file exports a constant with the Contentful ID of the content type itself, for example in contentful_vo_lib_person.go:

const ContentTypePerson = "person"

Constants are available for each locale supported by the space at the time of code generation, e.g.:

const SpaceLocaleGerman Locale = "de"
const SpaceLocaleFrench Locale = "fr"
const defaultLocale Locale = SpaceLocaleGerman

Four levels of logging are supported (even if only partially used at this time):

const (
LogDebug = 0
LogInfo = 1
LogWarn = 2
LogError = 3
)

A global variable named SpaceContentTypeInfoMap contains an ID-indexed map of all content types with their names and descriptions

- - + + \ No newline at end of file diff --git a/docs/projects/cms/gocontentful/caching.html b/docs/projects/cms/gocontentful/caching.html index 2550f76..3a24786 100644 --- a/docs/projects/cms/gocontentful/caching.html +++ b/docs/projects/cms/gocontentful/caching.html @@ -9,8 +9,8 @@ - - + +
@@ -50,7 +50,7 @@ of entries (in the order of less than some hundreds) and do that at significant (e.g. every hour). In this case your application code can be simpler and there won't be any performance penalty. The other case is when you need to run a lot of custom queries or use XPath, which is currently not supported by gocontentful directly.

- - + + \ No newline at end of file diff --git a/docs/projects/cms/gocontentful/getting-started.html b/docs/projects/cms/gocontentful/getting-started.html index e82881e..8739075 100644 --- a/docs/projects/cms/gocontentful/getting-started.html +++ b/docs/projects/cms/gocontentful/getting-started.html @@ -9,8 +9,8 @@ - - + +
@@ -20,7 +20,7 @@ A better approach is to log in to Contentful using the official Contentful C To install the CLI refer to the official documentation at Contentful.com.

After installing the CLI log in inside your terminal with:

$ contentful login
$ gocontentful

ERROR: Please specify either a Contentful Space ID and CMA access token or an export file name

SYNOPSIS
gocontentful -spaceid SpaceID -cmakey CMAKey [-contenttypes firsttype,secondtype...lasttype] path/to/target/package

-cmakey string
[Optional] Contentful CMA key
-contenttypes string
[Optional] Content type IDs to parse, comma separated
-environment string
[Optional] Contentful space environment
-exportfile string
Space export file to generate the API from
-help
Print version and exit
-spaceid string
Contentful space ID
-version
Print version and exit

Notes:
- The last segment of the path/to/target/package will be used as package name
- The -cmakey parameter can be omitted if you logged in with the Contentful CLI

Notes:

  • The last segment of the path/to/target/package will be used as package name
  • You need to pass gocontentful either cmakey/spaceid (and optional environment) to generate the API from a live space or exportfile to generate it from a local space export file. The cmakey can be omitted if you are logged in through the Contentful CLI.
- - + + \ No newline at end of file diff --git a/docs/projects/cms/gocontentful/introduction.html b/docs/projects/cms/gocontentful/introduction.html index 13947c3..f551e77 100644 --- a/docs/projects/cms/gocontentful/introduction.html +++ b/docs/projects/cms/gocontentful/introduction.html @@ -9,13 +9,13 @@ - - + +

Introduction

What is Contentful

Contentful is a content platform (often referred to as headless CMS) for micro-content. There's no pages or content trees in Contentful, the CMS focuses on structured data. The data model is built from scratch for the purpose of the consuming application, is completely flexible and can be created and hot-changed through the same Web UI that the content editors use. The model dictates which content types can reference others and the final structure is a graph.

What is gocontentful

A golang API code generator that simplifies interacting with a Contentful space. The generated API:

  • Supports most of the Contentful APIs to perform all read/write operation on entries and assets
  • Hides the complexity of the Contentful REST/JSON APIs behind an idiomatic set of golang functions and methods
  • Allows for in-memory caching of an entire Contentful space

Rationale

Calling a remote CMS repository across the Internet whenever your service needs a piece of content is not an option because of latency and response time. Hence, you need to cache all the content at the running service and make sure your cache is always up-to-date. In addition, you need to deal with generic entries in JSON format and maintain a Go model with value objects to reflect every change at the CMS' content model, writing functions and methods to import and export entries to/from typed objects.

Trust me: both things quickly scale to royal-PITA level.

Gocontentful wipes out both complexities by generating the content model automatically and providing an idiomatic API to handle interaction with remote content while keeping a cache stays constantly in-sync with the Contentful space. If the content model changes, running gocontentful again will update the Go code for the model and API to reflect those changes automatically.

How much code is that? As an example of a real-world production scenario where gocontentful is in use as of 2022, a space content model with 11 content types ranging from 3 to over 40 fields each generated around 50,000 lines of Go code. Do you need all those lines? You might not need all setters if you don't manage content through the API but you'll definitely need most of the getters otherwise those should not be in the model at all.

- - + + \ No newline at end of file diff --git a/docs/projects/cms/gocontentful/test-api.html b/docs/projects/cms/gocontentful/test-api.html index 775822a..3851397 100644 --- a/docs/projects/cms/gocontentful/test-api.html +++ b/docs/projects/cms/gocontentful/test-api.html @@ -9,8 +9,8 @@ - - + +
@@ -35,7 +35,7 @@ The object returned is a generic *testapi.EntryReference that, amon is the actual *testapi.CfBrand. That's why in the second line we had to cast it.

The test now logs the brand company name:

INFO[0000] Brand                                         name="Normann Copenhagen"

What if we want to follow the reference the other way around and find out which entries link to this brand?

parentRefs, errParents := brand.GetParents()
require.NoError(t, errParents)
testLogger.WithField("parent count", len(parentRefs)).Info("Parents")
for _, parentRef := range parentRefs {
switch parentRef.ContentType {
case testapi.ContentTypeProduct:
parentProduct := parentRef.VO.(*testapi.CfProduct)
testLogger.WithField("name", parentProduct.ProductName()).Info("Parent product")
}
}

Again, the GetParents() method returns references and not objects. It's a good idea to use the reference ContentType attribute to switch before casting the VO to the type, because as we just said referenced objects can come in different types and casting to the wrong one would make the runtime panic. Running the test we find out the two products that belong to this brand:

INFO[0000] Parents                                       parent count=2
INFO[0000] Parent product name="Whisk Beater"
INFO[0000] Parent product name="Hudson Wall Cup"
- - + + \ No newline at end of file diff --git a/docs/projects/cms/gocontentful/working-with-gocontentful-api.html b/docs/projects/cms/gocontentful/working-with-gocontentful-api.html index 09397a9..ea3b112 100644 --- a/docs/projects/cms/gocontentful/working-with-gocontentful-api.html +++ b/docs/projects/cms/gocontentful/working-with-gocontentful-api.html @@ -9,8 +9,8 @@ - - + +
@@ -63,7 +63,7 @@ the value object of entry B to the reference field in A. First you need to conve object to a ContentTypeSys object because that's what Contentful expects in reference fields:

(vo *CfPerson) ToReference() (refSys ContentTypeSys)

Finally, you can get the parents (AKA referring) entries of either an entry or an EntryReference with the GetParents() method. This returns a slice of []EntryReference:

(vo *CfPerson) GetParents() (parents []EntryReference, err error)
(ref *EntryReference) GetParents(cc *ContentfulClient) (parents []EntryReference, err error)

Other useful functions

Another thing you might want to know is the content type of an entry with a given ID:

(cc *ContentfulClient) GetContentTypeOfID(ID string) (contentType string)

Caveats and limitations

  • Avoid creating content types that have field IDs equal to reserved Go words (e.g. "type"). Gocontentful won't scan for them and the generated code will break.
- - + + \ No newline at end of file diff --git a/docs/projects/cms/intro.html b/docs/projects/cms/intro.html index d880344..1fc0500 100644 --- a/docs/projects/cms/intro.html +++ b/docs/projects/cms/intro.html @@ -9,13 +9,13 @@ - - + +

Intro

The foomo team has extensive experience with a wide range of CMS systems. Our long journey in the field has brought us very close to the https://jamstack.org . We almost exclusively work with https://app.contentful.com/ but we are watching others like https://www.stripe.com/ very closely.

Expectations and challenges

  • application developers want an environment, that ensures maximum productivity, they do not think of semantic structures, they think of routes
  • customers expect maximum control and flexibility when working with content and do not distinguish between content and applications

The need for dynamic rendering

If you can - render static content and distribute it with a CDN.

There are use cases though, where static site generation does not work like

  • highly personalized content
  • mixing content with entities from other systems, that have a different life cycle, like products

Application routes vs semantic URL structures

Bridging the conceptual gap between sematic content and applications.

Content especially when created with a focus on SEO comes as a semantic graph. This typically conflicts with how application developers see the world.

e-commerce as an example

Let us take a look at a real world example:

application route / ingressapp developers viewSEO URL requirement
/content:id/content/1/
/content/2/mens
/category:id/category/1/mens/shirts
/category/2/mens/shirts/business
/product:id/product/1/mens-shirt-awesome-blue-medium
/store:id/store/1/mens/stores/london-perfect-shirts
/store/2/kids/stores/lego-paradise

our approach

contentserver allows you to resolve URIs to mime types, which can be handled by applications.

Let us return to our ecommerce example.

model content

mime typecms entityapplication eg k8s service
application/x-pagepagehttps://frontend_service_page
application/x-categorycategoryhttps://frontend_service_category
application/x-storestorehttps://frontend_service_store

export contentent to contentserver

contentserver allows you to export your content tree as json

resolving URIs

Use standard application routing whereever possible, since it is the most efficient option.

Once all standard URI resolution fails, query contentserver and it will return the resolved node


Now you have the mime-type and you know which application can handle it.

- - + + \ No newline at end of file diff --git a/docs/projects/gotsrpc.html b/docs/projects/gotsrpc.html index c0984d3..7109710 100644 --- a/docs/projects/gotsrpc.html +++ b/docs/projects/gotsrpc.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/projects/intro.html b/docs/projects/intro.html index 922d12b..2079a73 100644 --- a/docs/projects/intro.html +++ b/docs/projects/intro.html @@ -9,13 +9,13 @@ - - + +

foomo project overview

"Stuff we wrote, because nobody else wanted to"

All foomo projects are available here https://github.com/foomo also see architecture

We are using the best industry standards and only fill in our own bits, when necessary.

The foomo project maintains libraries, utilities and daemons, which power most projects at https://www.bestbytes.com

Projects running on the foomo stack typically are

  • relatively large with hundreds of thousands of lines of code
  • composed of dozens of microservices
  • written / maintained by dozens of contributors
  • using Go on the server side
  • using Next.js / TypeScript on the client side
  • running on k8s
  • deeply integrated with a headless CMS
  • built with a project specific frontend component set

Everything is loosely coupled, since we know, that the only constant is change.

- - + + \ No newline at end of file diff --git a/docs/projects/libraries/keel-circuit-breaker.html b/docs/projects/libraries/keel-circuit-breaker.html index 7fd5ee9..68f6cd6 100644 --- a/docs/projects/libraries/keel-circuit-breaker.html +++ b/docs/projects/libraries/keel-circuit-breaker.html @@ -9,13 +9,13 @@ - - + +

Circuit Breaker

Sometimes things go wrong and a service does not respond anymore. Be it because of maintainance or because the whole data center burned to the ground. In such a scenario, you might not want to wait until your request times out. This is where circuit breakers come in handy.

Simply put a circuit breaker has three different states:

We did not re-invent the wheel (yet), but rather used an existing circuit breaker. However, we extended the functionality a bit. More on that later. For referance, here are links to the underlying circuit breaker and some more information on circuit breakers in general.

How to use

In order to configure the circuit breaker there are two kinds of configuration. The "base" configuration using the CircuitBreakerSettings and optional configuration using CircuitBreakerOptions.

CircuitBreakerSettings

The settings are relatively straight forward and the same as with the underlying repository - with one exception. Our settings are missing the IsSuccessful field.

net/http/roundtripware/circuitbreaker.go
loading...

CircuitBreakerOptions

Currently, there are two options one for metrics and one for somewhat advanced usage.

Metrics

The option for metrics is, again, straigth forward. When the CircuitBreakerWithMetric option is used the roundtripware will create a counter on the provided meter and count the number of requests.

The attributes added to every count are:

  • previous_state (String): the state of the circuit breaker before the current request. Either "closed", "half-open" or "open"
  • current_state (String): the state of the circuit breaker after the current request. Either "closed", "half-open" or "open"
  • state_change (Bool): helper containing current_state != previous_state
  • error (Bool): false if the request was not passed through or was unsuccessful
net/http/roundtripware/circuitbreaker.go
loading...

IsSuccessful

As mentioned previously, the IsSuccessful field was removed from the basic settings. The reason is that the signature of that function was a bit limiting. As you can see below our IsSuccessful-function can use the request and response. Additionally, if copyReqBody and/or copyRespBody are set to true, you can even read from the respective body, without worrying about consuming the io.ReadCloser.

net/http/roundtripware/circuitbreaker.go
loading...

The ignore value that is returned alongside an error indicates whether the result of the call should be registered with the circuit breaker. For most use cases it should be set to false.

When the IsSuccessful-function returns an error (and the ignore value is set to false), the request will be counted as unsuccessful. Accordingly, a nil error paired with ignored set to false indicates a successful request.

Examples

Let's say we want to stop sending requests once we encountered three consecutive failures.

client := keelhttp.NewHTTPClient(
keelhttp.HTTPClientWithRoundTripware(l,
roundtripware.CircuitBreaker(&roundtripware.CircuitBreakerSettings{
Name: "my little circuit breaker™",
// 2 requests can pass in half-open state & it takes 2 consecutive,
// successful requests to change to closed state
MaxRequests: 2,
// counts are not reset in closed state
Interval: 0,
// breaker will go from open to half-open state after 30s
Timeout: 30 * time.Second,
// go to open state after the 3rd consecutive, unsuccessful request
ReadyToTrip: func(counts gobreaker.Counts) bool {
return counts.ConsecutiveFailures >= 3
},
}),
),
)

Now lets say we see we also want to detect network problems such as a BadGateway. For this we can use the IsSuccessful option.

client := keelhttp.NewHTTPClient(
keelhttp.HTTPClientWithRoundTripware(l,
roundtripware.CircuitBreaker(&roundtripware.CircuitBreakerSettings{
// as before ...
},
roundtripware.CircuitBreakerWithIsSuccessful(
func(err error, req *http.Request, resp *http.Response) (error, bool) {
if err != nil {
return err, false
}
if resp.StatusCode >= http.StatusInternalServerError {
return errors.New("invalid status code"), false
}
return nil, false
}, false, false,
),
),
),
)

Lastly, let's assume we use the client for multiple different endpoints. And we only want to base the circuit breakers state on a single endpoint, but stop request on all endpoints once the breaker changes to open. Again we can use the IsSuccessful option and ignore certain endpoints.

client := keelhttp.NewHTTPClient(
keelhttp.HTTPClientWithRoundTripware(l,
roundtripware.CircuitBreaker(&roundtripware.CircuitBreakerSettings{
// as before ...
},
roundtripware.CircuitBreakerWithIsSuccessful(
func(err error, req *http.Request, resp *http.Response) (error, bool) {
if req.URL.Path != "/important/path" {
return err, false
}

// possibly more checks ...

return err, true
}, false, false,
),
),
),
)

General advice & notes of caution

Using ratios in ReadyToTrip

When using ratios in ready to trip, the Interval should be set to a non-zero value in order to reset the counts periodically. Otherwise, after a long period of successful requests it will also take a long time to impact the ratio and trip the breaker.

    ReadyToTrip: func(counts gobreaker.Counts) bool {
failureRatio := float64(counts.TotalFailures) / float64(counts.Requests)
return counts.Requests >= 3 && failureRatio >= 0.6
},
- - + + \ No newline at end of file diff --git a/docs/projects/libraries/keel.html b/docs/projects/libraries/keel.html index 3fdc450..6eec3fc 100644 --- a/docs/projects/libraries/keel.html +++ b/docs/projects/libraries/keel.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/projects/pagespeed-exporter.html b/docs/projects/pagespeed-exporter.html index 422ced4..b0e35ca 100644 --- a/docs/projects/pagespeed-exporter.html +++ b/docs/projects/pagespeed-exporter.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/projects/site-reliability.html b/docs/projects/site-reliability.html index fcb5bab..17e7d7a 100644 --- a/docs/projects/site-reliability.html +++ b/docs/projects/site-reliability.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/projects/utilities/gograpple.html b/docs/projects/utilities/gograpple.html index e9b881e..2c5205e 100644 --- a/docs/projects/utilities/gograpple.html +++ b/docs/projects/utilities/gograpple.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/projects/webgrapple.html b/docs/projects/webgrapple.html index 97cbd94..2ca81ea 100644 --- a/docs/projects/webgrapple.html +++ b/docs/projects/webgrapple.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/tags.html b/docs/tags.html index 51948fe..f5d20c0 100644 --- a/docs/tags.html +++ b/docs/tags.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/tags/frontend.html b/docs/tags/frontend.html index ef6b496..dcf4f0a 100644 --- a/docs/tags/frontend.html +++ b/docs/tags/frontend.html @@ -9,13 +9,13 @@ - - + +

One doc tagged with "frontend"

View All Tags

Stack

Our frontend stack is permanently changing as the underlying eco system does. We are trying to adopt technologies at the "Slope of Enlightenment" in the hype cycle.

- - + + \ No newline at end of file diff --git a/docs/tags/go-basics.html b/docs/tags/go-basics.html index b77a024..953998d 100644 --- a/docs/tags/go-basics.html +++ b/docs/tags/go-basics.html @@ -9,13 +9,13 @@ - - + +

5 docs tagged with "Go Basics"

View All Tags

Defer

In a normal Go application, control flows from the top to the bottom of any function that we call (this is if you don't use branching or looping).

Maps

Maps are unordered key value pairs where each key is unique.

MongoDB

Make sure to use the official golang mongo driver:

- - + + \ No newline at end of file diff --git a/docs/tags/go-intermediate.html b/docs/tags/go-intermediate.html index 3ba887a..cf90ea9 100644 --- a/docs/tags/go-intermediate.html +++ b/docs/tags/go-intermediate.html @@ -9,13 +9,13 @@ - - + +

One doc tagged with "Go Intermediate"

View All Tags
- - + + \ No newline at end of file diff --git a/docs/tags/go.html b/docs/tags/go.html index 0cbc173..d079d1f 100644 --- a/docs/tags/go.html +++ b/docs/tags/go.html @@ -9,13 +9,13 @@ - - + +

One doc tagged with "go"

View All Tags
- - + + \ No newline at end of file diff --git a/docs/tags/javascript.html b/docs/tags/javascript.html index 75759d9..2d97d41 100644 --- a/docs/tags/javascript.html +++ b/docs/tags/javascript.html @@ -9,13 +9,13 @@ - - + +

One doc tagged with "javascript"

View All Tags
- - + + \ No newline at end of file diff --git a/docs/tags/overview.html b/docs/tags/overview.html index 4b63f08..19a01d8 100644 --- a/docs/tags/overview.html +++ b/docs/tags/overview.html @@ -9,13 +9,13 @@ - - + +

One doc tagged with "overview"

View All Tags

Stack

Our frontend stack is permanently changing as the underlying eco system does. We are trying to adopt technologies at the "Slope of Enlightenment" in the hype cycle.

- - + + \ No newline at end of file diff --git a/docs/tags/sse.html b/docs/tags/sse.html index 1826c63..892ffb4 100644 --- a/docs/tags/sse.html +++ b/docs/tags/sse.html @@ -9,13 +9,13 @@ - - + +

One doc tagged with "sse"

View All Tags
- - + + \ No newline at end of file diff --git a/etc/imprint.html b/etc/imprint.html index a5d82ea..7c8315d 100644 --- a/etc/imprint.html +++ b/etc/imprint.html @@ -9,8 +9,8 @@ - - + +
@@ -22,7 +22,7 @@ E-Mail: Haftung für Inhalte

Die Inhalte unserer Seiten wurden mit größter Sorgfalt erstellt. Als Diensteanbieter sind wir gemäß § 7 Abs.1 TMG für eigene Inhalte auf diesen Seiten nach den allgemeinen Gesetzen verantwortlich. Nach §§ 8 bis 10 TMG sind wir als Diensteanbieter jedoch nicht verpflichtet, übermittelte oder gespeicherte fremde Informationen zu überwachen oder nach Umständen zu forschen, die auf eine rechtswidrige Tätigkeit hinweisen. Verpflichtungen zur Entfernung oder Sperrung der Nutzung von Informationen nach den allgemeinen Gesetzen bleiben hiervon unberührt. Eine diesbezügliche Haftung ist jedoch erst ab dem Zeitpunkt der Kenntnis einer konkreten Rechtsverletzung möglich. Bei Bekanntwerden von entsprechenden Rechtsverletzungen werden wir diese Inhalte umgehend entfernen.

Unser Angebot enthält Links zu externen Webseiten Dritter, auf deren Inhalte wir keinen Einfluss haben. Deshalb können wir für diese fremden Inhalte auch keine Gewähr übernehmen. Für die Inhalte der verlinkten Seiten ist stets der jeweilige Anbieter oder Betreiber der Seiten verantwortlich. Die verlinkten Seiten wurden zum Zeitpunkt der Verlinkung auf mögliche Rechtsverstöße überprüft. Rechtswidrige Inhalte waren zum Zeitpunkt der Verlinkung nicht erkennbar. Eine permanente inhaltliche Kontrolle der verlinkten Seiten ist jedoch ohne konkrete Anhaltspunkte einer Rechtsverletzung nicht zumutbar. Bei Bekanntwerden von Rechtsverletzungen werden wir derartige Links umgehend entfernen.

Quellenangaben: Disclaimer eRecht24 - https://www.e-recht24.de/muster-disclaimer.html

Website-Design, Texte, Grafiken und Layout Copyright © 1998 - today bestbytes. ALLE RECHTE VORBEHALTEN. Das Kopieren oder die Reproduktion der gesamten Website bzw. von Teilen dieser Website ist untersagt, es sei denn, bestbytes hat dem vorher schriftlich oder elektronisch zugestimmt. Alle Markenzeichen und Logos, die auf dieser Internetseite verwendet werden, sind urheberrechtlich geschützt.

- - + + \ No newline at end of file diff --git a/index.html b/index.html index aea7821..be7593c 100644 --- a/index.html +++ b/index.html @@ -9,13 +9,13 @@ - - + +

foomo project docs

Stuff we wrote, because nobody else wanted to


foomo is an open source project, that has been maintained by the bestbytes team since 2011. It provides a wide range of utilities, libraries and daemons, that help us to tackle challenging projects.

foomo can help, if you are

  • using Go to write services
  • building frontends with TypeScript and Next.js
  • running your software on k8s
  • looking for a solution to deeply integrate your frontends with a headless cms
If more than two points apply, it will actually help a LOT

General guides

no matter if frontend or backend - there is somtehing for everyone in here

Frontend guides

Build frontends with TypeScript, Next.js, styled components, gotsrpc

Backend guides

Write fast and reliable services in Go and run them in the cloud

Projects

Foomo projects - libraries, utilities, friendly daemons and more

CMS

foomo has extensive support for headless CMS Systems

Architecture

Show me the big picture - how does foomo integrate into the k8s and Next.js eco system

Awesome Software

a curated list of software we use in the cloud, on our desktops and our devices

DevOps

our approach to run cloud native applications where they belong

Blog

random information about programming, our ecosystem and misc fun topics

- - + + \ No newline at end of file diff --git a/search.html b/search.html index ab483b5..ab3d3ab 100644 --- a/search.html +++ b/search.html @@ -9,13 +9,13 @@ - - + + - - + + \ No newline at end of file diff --git a/sitemap.xml b/sitemap.xml index 3f41ec7..72cb050 100644 --- a/sitemap.xml +++ b/sitemap.xml @@ -1 +1 @@ -https://www.foomo.org/awesome-softwareweekly0.5https://www.foomo.org/blogweekly0.5https://www.foomo.org/blog/archiveweekly0.5https://www.foomo.org/blog/debugging-go-map-races-in-k8sweekly0.5https://www.foomo.org/blog/impact-of-3rd-party-scripts-on-performanceweekly0.5https://www.foomo.org/blog/prometheus-cardinality-issuesweekly0.5https://www.foomo.org/blog/searching-for-search-enginesweekly0.5https://www.foomo.org/blog/tagsweekly0.5https://www.foomo.org/blog/tags/backendweekly0.5https://www.foomo.org/blog/tags/bundleweekly0.5https://www.foomo.org/blog/tags/bundle-sizeweekly0.5https://www.foomo.org/blog/tags/cardinalityweekly0.5https://www.foomo.org/blog/tags/debuggingweekly0.5https://www.foomo.org/blog/tags/devopsweekly0.5https://www.foomo.org/blog/tags/foomoweekly0.5https://www.foomo.org/blog/tags/frontendweekly0.5https://www.foomo.org/blog/tags/goweekly0.5https://www.foomo.org/blog/tags/javascriptweekly0.5https://www.foomo.org/blog/tags/k-8-sweekly0.5https://www.foomo.org/blog/tags/memoryweekly0.5https://www.foomo.org/blog/tags/oomweekly0.5https://www.foomo.org/blog/tags/opsweekly0.5https://www.foomo.org/blog/tags/performanceweekly0.5https://www.foomo.org/blog/tags/prometheusweekly0.5https://www.foomo.org/blog/tags/searchweekly0.5https://www.foomo.org/blog/tags/search-engineweekly0.5https://www.foomo.org/blog/welcome-back-2021weekly0.5https://www.foomo.org/blog/why-bundle-size-is-importantweekly0.5https://www.foomo.org/docs/tagsweekly0.5https://www.foomo.org/docs/tags/frontendweekly0.5https://www.foomo.org/docs/tags/goweekly0.5https://www.foomo.org/docs/tags/go-basicsweekly0.5https://www.foomo.org/docs/tags/go-intermediateweekly0.5https://www.foomo.org/docs/tags/javascriptweekly0.5https://www.foomo.org/docs/tags/overviewweekly0.5https://www.foomo.org/docs/tags/sseweekly0.5https://www.foomo.org/etc/imprintweekly0.5https://www.foomo.org/searchweekly0.5https://www.foomo.org/docs/backend/go-by-example/cli-applicationsweekly0.5https://www.foomo.org/docs/backend/go-by-example/contextweekly0.5https://www.foomo.org/docs/backend/go-by-example/deferweekly0.5https://www.foomo.org/docs/backend/go-by-example/embedweekly0.5https://www.foomo.org/docs/backend/go-by-example/filesweekly0.5https://www.foomo.org/docs/backend/go-by-example/goroutines-and-channelsweekly0.5https://www.foomo.org/docs/backend/go-by-example/httpweekly0.5https://www.foomo.org/docs/backend/go-by-example/interfacesweekly0.5https://www.foomo.org/docs/backend/go-by-example/map-racingweekly0.5https://www.foomo.org/docs/backend/go-by-example/mongodbweekly0.5https://www.foomo.org/docs/backend/go-by-example/nil-mapsweekly0.5https://www.foomo.org/docs/backend/go-by-example/panic-and-recoverweekly0.5https://www.foomo.org/docs/backend/go-by-example/rangingweekly0.5https://www.foomo.org/docs/backend/go-by-example/reflectionweekly0.5https://www.foomo.org/docs/backend/go-by-example/type-castingweekly0.5https://www.foomo.org/docs/backend/go-by-example/yaml-magicweekly0.5https://www.foomo.org/docs/backend/gograppleweekly0.5https://www.foomo.org/docs/backend/introweekly0.5https://www.foomo.org/docs/backend/rtfmweekly0.5https://www.foomo.org/docs/backend/setupweekly0.5https://www.foomo.org/docs/devops/bookmarksweekly0.5https://www.foomo.org/docs/devops/infrastructure/conventionsweekly0.5https://www.foomo.org/docs/devops/introweekly0.5https://www.foomo.org/docs/devops/k8sweekly0.5https://www.foomo.org/docs/devops/kubernetes/spot-instance-node-poolsweekly0.5https://www.foomo.org/docs/devops/monitoring/grafanaweekly0.5https://www.foomo.org/docs/devops/monitoring/introweekly0.5https://www.foomo.org/docs/devops/monitoring/jaegerweekly0.5https://www.foomo.org/docs/frontend/cssweekly0.5https://www.foomo.org/docs/frontend/debugging_jsweekly0.5https://www.foomo.org/docs/frontend/introweekly0.5https://www.foomo.org/docs/frontend/performanceweekly0.5https://www.foomo.org/docs/frontend/pitfallsweekly0.5https://www.foomo.org/docs/frontend/rtfmweekly0.5https://www.foomo.org/docs/frontend/setupweekly0.5https://www.foomo.org/docs/frontend/stackweekly0.5https://www.foomo.org/docs/frontend/typescript/objectsweekly0.5https://www.foomo.org/docs/frontend/typescript/spreadingweekly0.5https://www.foomo.org/docs/frontend/vscodeweekly0.5https://www.foomo.org/docs/general/essentialsweekly0.5https://www.foomo.org/docs/general/introweekly0.5https://www.foomo.org/docs/general/package-managersweekly0.5https://www.foomo.org/docs/general/securityweekly0.5https://www.foomo.org/docs/general/setup/computerweekly0.5https://www.foomo.org/docs/general/setup/workplaceweekly0.5https://www.foomo.org/docs/general/technologies/sseweekly0.5https://www.foomo.org/docs/general/utilities/k9sweekly0.5https://www.foomo.org/docs/general/work/generalweekly0.5https://www.foomo.org/docs/general/work/remote-workweekly0.5https://www.foomo.org/docs/project-management/introweekly0.5https://www.foomo.org/docs/project-management/responsibilityweekly0.5https://www.foomo.org/docs/projects/architectureweekly0.5https://www.foomo.org/docs/projects/cms/contentfulweekly0.5https://www.foomo.org/docs/projects/cms/contentserverweekly0.5https://www.foomo.org/docs/projects/cms/gocontentful/api-referenceweekly0.5https://www.foomo.org/docs/projects/cms/gocontentful/cachingweekly0.5https://www.foomo.org/docs/projects/cms/gocontentful/getting-startedweekly0.5https://www.foomo.org/docs/projects/cms/gocontentful/introductionweekly0.5https://www.foomo.org/docs/projects/cms/gocontentful/test-apiweekly0.5https://www.foomo.org/docs/projects/cms/gocontentful/working-with-gocontentful-apiweekly0.5https://www.foomo.org/docs/projects/cms/introweekly0.5https://www.foomo.org/docs/projects/gotsrpcweekly0.5https://www.foomo.org/docs/projects/introweekly0.5https://www.foomo.org/docs/projects/libraries/keelweekly0.5https://www.foomo.org/docs/projects/libraries/keel-circuit-breakerweekly0.5https://www.foomo.org/docs/projects/pagespeed-exporterweekly0.5https://www.foomo.org/docs/projects/site-reliabilityweekly0.5https://www.foomo.org/docs/projects/utilities/gograppleweekly0.5https://www.foomo.org/docs/projects/webgrappleweekly0.5https://www.foomo.org/weekly0.5 \ No newline at end of file +https://www.foomo.org/awesome-softwareweekly0.5https://www.foomo.org/blogweekly0.5https://www.foomo.org/blog/accuracy-of-decimal-computationsweekly0.5https://www.foomo.org/blog/archiveweekly0.5https://www.foomo.org/blog/debugging-go-map-races-in-k8sweekly0.5https://www.foomo.org/blog/impact-of-3rd-party-scripts-on-performanceweekly0.5https://www.foomo.org/blog/prometheus-cardinality-issuesweekly0.5https://www.foomo.org/blog/searching-for-search-enginesweekly0.5https://www.foomo.org/blog/tagsweekly0.5https://www.foomo.org/blog/tags/backendweekly0.5https://www.foomo.org/blog/tags/bundleweekly0.5https://www.foomo.org/blog/tags/bundle-sizeweekly0.5https://www.foomo.org/blog/tags/cardinalityweekly0.5https://www.foomo.org/blog/tags/currencyweekly0.5https://www.foomo.org/blog/tags/debuggingweekly0.5https://www.foomo.org/blog/tags/decimal-accuracyweekly0.5https://www.foomo.org/blog/tags/devopsweekly0.5https://www.foomo.org/blog/tags/foomoweekly0.5https://www.foomo.org/blog/tags/frontendweekly0.5https://www.foomo.org/blog/tags/goweekly0.5https://www.foomo.org/blog/tags/golangweekly0.5https://www.foomo.org/blog/tags/javascriptweekly0.5https://www.foomo.org/blog/tags/k-8-sweekly0.5https://www.foomo.org/blog/tags/memoryweekly0.5https://www.foomo.org/blog/tags/oomweekly0.5https://www.foomo.org/blog/tags/opsweekly0.5https://www.foomo.org/blog/tags/performanceweekly0.5https://www.foomo.org/blog/tags/prometheusweekly0.5https://www.foomo.org/blog/tags/searchweekly0.5https://www.foomo.org/blog/tags/search-engineweekly0.5https://www.foomo.org/blog/welcome-back-2021weekly0.5https://www.foomo.org/blog/why-bundle-size-is-importantweekly0.5https://www.foomo.org/docs/tagsweekly0.5https://www.foomo.org/docs/tags/frontendweekly0.5https://www.foomo.org/docs/tags/goweekly0.5https://www.foomo.org/docs/tags/go-basicsweekly0.5https://www.foomo.org/docs/tags/go-intermediateweekly0.5https://www.foomo.org/docs/tags/javascriptweekly0.5https://www.foomo.org/docs/tags/overviewweekly0.5https://www.foomo.org/docs/tags/sseweekly0.5https://www.foomo.org/etc/imprintweekly0.5https://www.foomo.org/searchweekly0.5https://www.foomo.org/docs/backend/go-by-example/cli-applicationsweekly0.5https://www.foomo.org/docs/backend/go-by-example/contextweekly0.5https://www.foomo.org/docs/backend/go-by-example/deferweekly0.5https://www.foomo.org/docs/backend/go-by-example/embedweekly0.5https://www.foomo.org/docs/backend/go-by-example/filesweekly0.5https://www.foomo.org/docs/backend/go-by-example/goroutines-and-channelsweekly0.5https://www.foomo.org/docs/backend/go-by-example/httpweekly0.5https://www.foomo.org/docs/backend/go-by-example/interfacesweekly0.5https://www.foomo.org/docs/backend/go-by-example/map-racingweekly0.5https://www.foomo.org/docs/backend/go-by-example/mongodbweekly0.5https://www.foomo.org/docs/backend/go-by-example/nil-mapsweekly0.5https://www.foomo.org/docs/backend/go-by-example/panic-and-recoverweekly0.5https://www.foomo.org/docs/backend/go-by-example/rangingweekly0.5https://www.foomo.org/docs/backend/go-by-example/reflectionweekly0.5https://www.foomo.org/docs/backend/go-by-example/type-castingweekly0.5https://www.foomo.org/docs/backend/go-by-example/yaml-magicweekly0.5https://www.foomo.org/docs/backend/gograppleweekly0.5https://www.foomo.org/docs/backend/introweekly0.5https://www.foomo.org/docs/backend/rtfmweekly0.5https://www.foomo.org/docs/backend/setupweekly0.5https://www.foomo.org/docs/devops/bookmarksweekly0.5https://www.foomo.org/docs/devops/infrastructure/conventionsweekly0.5https://www.foomo.org/docs/devops/introweekly0.5https://www.foomo.org/docs/devops/k8sweekly0.5https://www.foomo.org/docs/devops/kubernetes/spot-instance-node-poolsweekly0.5https://www.foomo.org/docs/devops/monitoring/grafanaweekly0.5https://www.foomo.org/docs/devops/monitoring/introweekly0.5https://www.foomo.org/docs/devops/monitoring/jaegerweekly0.5https://www.foomo.org/docs/frontend/cssweekly0.5https://www.foomo.org/docs/frontend/debugging_jsweekly0.5https://www.foomo.org/docs/frontend/introweekly0.5https://www.foomo.org/docs/frontend/performanceweekly0.5https://www.foomo.org/docs/frontend/pitfallsweekly0.5https://www.foomo.org/docs/frontend/rtfmweekly0.5https://www.foomo.org/docs/frontend/setupweekly0.5https://www.foomo.org/docs/frontend/stackweekly0.5https://www.foomo.org/docs/frontend/typescript/objectsweekly0.5https://www.foomo.org/docs/frontend/typescript/spreadingweekly0.5https://www.foomo.org/docs/frontend/vscodeweekly0.5https://www.foomo.org/docs/general/essentialsweekly0.5https://www.foomo.org/docs/general/introweekly0.5https://www.foomo.org/docs/general/package-managersweekly0.5https://www.foomo.org/docs/general/securityweekly0.5https://www.foomo.org/docs/general/setup/computerweekly0.5https://www.foomo.org/docs/general/setup/workplaceweekly0.5https://www.foomo.org/docs/general/technologies/sseweekly0.5https://www.foomo.org/docs/general/utilities/k9sweekly0.5https://www.foomo.org/docs/general/work/generalweekly0.5https://www.foomo.org/docs/general/work/remote-workweekly0.5https://www.foomo.org/docs/project-management/introweekly0.5https://www.foomo.org/docs/project-management/responsibilityweekly0.5https://www.foomo.org/docs/projects/architectureweekly0.5https://www.foomo.org/docs/projects/cms/contentfulweekly0.5https://www.foomo.org/docs/projects/cms/contentserverweekly0.5https://www.foomo.org/docs/projects/cms/gocontentful/api-referenceweekly0.5https://www.foomo.org/docs/projects/cms/gocontentful/cachingweekly0.5https://www.foomo.org/docs/projects/cms/gocontentful/getting-startedweekly0.5https://www.foomo.org/docs/projects/cms/gocontentful/introductionweekly0.5https://www.foomo.org/docs/projects/cms/gocontentful/test-apiweekly0.5https://www.foomo.org/docs/projects/cms/gocontentful/working-with-gocontentful-apiweekly0.5https://www.foomo.org/docs/projects/cms/introweekly0.5https://www.foomo.org/docs/projects/gotsrpcweekly0.5https://www.foomo.org/docs/projects/introweekly0.5https://www.foomo.org/docs/projects/libraries/keelweekly0.5https://www.foomo.org/docs/projects/libraries/keel-circuit-breakerweekly0.5https://www.foomo.org/docs/projects/pagespeed-exporterweekly0.5https://www.foomo.org/docs/projects/site-reliabilityweekly0.5https://www.foomo.org/docs/projects/utilities/gograppleweekly0.5https://www.foomo.org/docs/projects/webgrappleweekly0.5https://www.foomo.org/weekly0.5 \ No newline at end of file