Skip to content

Commit f9835e4

Browse files
committed
Update
1 parent 984e4ff commit f9835e4

8 files changed

+257
-23
lines changed

SEO-for-developers.md

+1
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22

33
## Reference
44

5+
<https://codefrontend.com/introduction-to-seo-for-react-developers/>
56
<https://www.youtube.com/watch?v=JSm4aQl4w_U>
67

78
## What is SEO

css-tips.md

+2-1
Original file line numberDiff line numberDiff line change
@@ -100,8 +100,9 @@ button {
100100

101101
## Background image opacity
102102

103-
Apply opacity to absolute psuedo element, parent element use isolation: isolate
103+
Apply opacity to absolute psuedo element, parent element use `isolation: isolate`, that will create a stacking context and prevent absolute children with negatives `z-index` to break out of parents
104104
<https://www.youtube.com/watch?v=lRPguPbovro>
105+
<https://www.youtube.com/watch?v=o1HzOJfgugE>
105106

106107
## Replace `position: absolute` with `display: grid`
107108

dependancy-injection.md

+4-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,10 @@
11
# Dependancy injection
22

3-
Dependancy injection is a popular pattern that solve the big problem: hardcoded dependancy. So what's hardcoded dependancy?
3+
<https://blog.codeminer42.com/di-with-some-context/>
44

5+
Dependancy injection is a popular pattern used to remove hardcoded dependancy between units in an application. The aim of this technique is to make our code more flexible and scalable
6+
7+
So what's hardcoded dependancy?
58
Hardcode dependancy means class A is tightly coupled to class B, class B is called inside class A. There are some issue with this:
69

710
- It's harder for testing: Let's say we want to test class A. And because class A is using class B, it's really hard to mock class B

interview.md

+21-2
Original file line numberDiff line numberDiff line change
@@ -70,6 +70,10 @@ If you mutate the data directly, the components might not re-render that could l
7070

7171
With `useEffect`, we can control how the side effects run in the component. Normally, the side effects make the state change. So, if you put them outside the `useEffec` hook, it will lead to infinite re-render of the component.
7272

73+
## Why we the callback in `useEffect` mustn't be async function
74+
75+
Because async function return promise. `useEffect` hook isn't expecting us to return a promise. It expects us to return either nothing (like we are above), or a cleanup function.
76+
7377
## When react does not re-render
7478

7579
<https://www.zhenghao.io/posts/react-rerender>
@@ -296,6 +300,7 @@ The difference happens when the `success` callback returns a **rejected promise*
296300
297301
## Rule for `this`
298302
303+
<https://javascript.info/arrow-functions>
299304
<https://blog.tusharcodes.tech/5-rules-to-master-this-in-javascript>
300305
<https://i.imgur.com/tUjhc3r.png>
301306
@@ -375,7 +380,13 @@ Các vấn đế của JWT
375380
376381
- User A authen thành công ➜ server response token ➜ User A bị Man-in-the-middle attack, do User A bất cẩn làm token rơi vào tay hacker ➜ hacker chiếm quyền truy cập account của user A mà server không hề hay biết
377382
- User A đang truy cập các private API, đột nhiên token expired, màn hình đột ngột bị rediect sang trang login vì server response 403
378-
- server chưa cài đặt cơ chế hủy token, phía database thì account user A đã bị ban, nhưng token vẫn đang lưu rải rác tại các client (desktop app, mobile app, web, server khác, ...), client vẫn cứ gửi token, server encode payload token thấy vẫn còn hạn sử dụng, phần mã hash chữ ký vẫn hợp lệ ➜ user vẫn author thành công
383+
- server chưa cài đặt cơ chế hủy token, phía database thì account user A đã bị ban, nhưng token vẫn đang lưu rải rác tại các client (desktop app, mobile app, web, server khác, ...), client vẫn cứ gửi token, server decode payload token thấy vẫn còn hạn sử dụng, phần mã hash chữ ký vẫn hợp lệ ➜ user vẫn author thành công
384+
385+
Cách logout jwt
386+
387+
- Khi gọi API logout thì lưu token đã logout vào blacklist.
388+
- Trong các API được authen, khi có request đến, check token có nằm trong blacklist hay không, nếu có thì coi như unauthorize.
389+
- Bài toán đặt ra là lưu blacklist ở đâu để việc đọc token đảm bảo được về mặt hiệu suất khi mà tần số gọi API dày đặc như vậy? Câu trả lời là lưu trong redis. Mặc khác vì jwt token nó có thời hạn sống ngắn, nên hoàn toàn có thể clean up dữ liệu, xoá các token đã thực sự hết hạn trong redis.
379390
380391
## Debounce và throttle
381392
@@ -583,7 +594,7 @@ const copied = new Map(myMap);
583594
584595
## Else if using operation tenerry
585596
586-
const a = b ? c : d ? e : f
597+
`const a = b ? c : d ? e : f`
587598
588599
## OOP
589600
@@ -593,6 +604,7 @@ const a = b ? c : d ? e : f
593604
594605
Là cơ chế của server cho phép các origin khác (origin = scheme(protocol) + domain + port) ngoài chính nó gửi request.
595606
Cùng origin: cùng scheme, domain và port
607+
596608
- Cùng scheme và cùng domain
597609
598610
http://example.com/app1/index.html
@@ -639,6 +651,7 @@ https://example.com/app2
639651
## Closure
640652
641653
<https://dmitripavlutin.com/simple-explanation-of-javascript-closures/>
654+
642655
- Lexical scope của 1 hàm:
643656
- Là tất cả các outer scope cộng với inner scope của hàm đó.
644657
- Outer scope là môi trường bên ngoài nơi định nghĩa của 1 hàm.
@@ -676,14 +689,18 @@ Primitive type là immutable. Ta chỉ có thể gán lại biến đấy sang 1
676689
Reference type là mutable. Nếu ta copy biến đấy sang 1 biến khác, 2 biến đấy sẽ trỏ vào cùng 1 box chứ 2 biến không trỏ lẫn sang nhau. Và nếu 1 trong 2 biến mutate (ví dụ như là thay đổi giá trị của property) thì biến còn lại cũng thay đổi theo (vì vẫn chung 1 box)
677690
678691
Notes:
692+
679693
- Các giá trị dạng reference là riêng biệt:
694+
680695
Mỗi khi ta khai báo 1 hàm hoặc 1 object, `() => {}` hoặc `{}` thì 1 box mới được tạo ra
681696
```js
682697
const a = () => {}
683698
const b = () => {}
684699
console.log( a === b ) // return false.
685700
```
701+
686702
- Truyền tham số dạng reference có thể xảy ra Mutation không mong muốn
703+
687704
```javascript
688705
function minimum(array) {
689706
array.sort();
@@ -746,6 +763,7 @@ Có 3 loại scope: `global scope`, `local scope` (function) và `block scope` (
746763
## for-in and for-of
747764

748765
`for-in`:
766+
749767
- iterate over enumerable properties of an object. enumerable properties are the "key" of the property the Object or "index" of the Array.
750768
If the propery is defined with the `enumerable` false, it is not interated
751769
```javascript
@@ -760,6 +778,7 @@ Object.defineProperty(obj, 'flameWarTopic', {
760778
- cannot be used with Object like Map() or Set();
761779
762780
`for-of`:
781+
763782
- interate over **value** of iterable objects.
764783
- can be used with built-in `String`, `Array`, array-like objects (e.g., arguments or `NodeList`), `TypedArray`, `Map`, `Set`
765784
- cannot be used with simple Object

nginx-nodejs.md

+5-1
Original file line numberDiff line numberDiff line change
@@ -150,4 +150,8 @@ echo "deb https://dl.yarnpkg.com/debian/ stable main" | sudo tee /etc/apt/source
150150
sudo apt-get update
151151
sudo apt-get install yarn -y
152152
yarn install
153-
```
153+
```
154+
155+
## Setup SSL using certbot
156+
157+
Go to https://certbot.eff.org/ and do as the instructions. When install Certbot, chose `sudo certbot --nginx`, it will automatically set up the SSL

promise.md

+66-1
Original file line numberDiff line numberDiff line change
@@ -80,19 +80,23 @@ Both take an array of promises
8080
## Async
8181

8282
**Syntax**:
83+
8384
```javascript
8485
async function() {
8586
};
8687
```
88+
8789
Put in front of function, then **returns a promise**. This promise will be resolved with the **value** returned by the async function or rejected with an error that is thrown inside async function
8890

8991
If the **value** is not a promise, it will be wrapped in promise.
92+
9093
```javascript
9194
async function foo() {
9295
return 1
9396
}
9497
```
9598
...is similar to:
99+
96100
```javascript
97101
function foo() {
98102
return Promise.resolve(1)
@@ -117,17 +121,19 @@ async function run() {
117121
}
118122
```
119123

120-
121124
## Await
122125

123126
Can only be used inside `async` function.
124127
**Syntax**:
128+
125129
```javascript
126130
const value = await expression;
127131
```
132+
128133
`expression` is a `Promise` or any value. If a value, it convert to `Promise.resolve(expression)`, which is a `Promise`;
129134

130135
`Await` pause the `async` function ultil a `Promise` is fulfilled or rejected.
136+
131137
- If rejected, `await` throw an error.
132138
- If fulfilled, it resumes the `async` function execution and **return the value of the fulfilled Promise**
133139

@@ -136,3 +142,62 @@ You can catch the error right after await, without using `try...catch`
136142
const value = await expression.catch((e) => {})
137143
```
138144

145+
## Handle concurrent promises
146+
147+
[https://www.builder.io/blog/promises](https://www.builder.io/blog/promises)
148+
149+
Use `Promise.allSettled()`
150+
151+
```jsx
152+
// First create a utility funtion
153+
function handleResults(results) {
154+
const errors = results
155+
.filter(result => result.status === 'rejected')
156+
.map(result => result.reason)
157+
158+
if (errors.length) {
159+
// Aggregate all errors into one
160+
throw new AggregateError(errors)
161+
}
162+
163+
return results.map(result => result.value)
164+
}
165+
166+
// Usage
167+
168+
async function getPageData() {
169+
const results = await Promise.allSettled([
170+
fetchUser(), fetchProduct()
171+
])
172+
173+
try {
174+
const [user, product] = handleResults(results)
175+
} catch (err) {
176+
for (const error of err.errors) {
177+
handle(error)
178+
}
179+
}
180+
}
181+
```
182+
183+
Or we await separately after instantiation
184+
185+
```js
186+
async function getPageData() {
187+
// We fire all the promises first, so they can run both in the same time, instead of waiting for each other
188+
const userPromise = fetchUser().catch(onReject)
189+
const productPromise = fetchProduct().catch(onReject)
190+
191+
// Try/catch each
192+
try {
193+
const user = await userPromise
194+
} catch (err) {
195+
handle(err)
196+
}
197+
try {
198+
const product = await productPromise
199+
} catch (err) {
200+
handle(err)
201+
}
202+
}
203+
```

react-query.md

+21
Original file line numberDiff line numberDiff line change
@@ -295,6 +295,27 @@ function Todo({ todoId }) {
295295
}
296296
```
297297
298+
## Handle errors
299+
300+
While most utilities like `axios` automatically throw errors for unsuccessful HTTP calls, some utilities like fetch do not throw errors by default. If that's the case, you'll need to throw them on your own. Here is a simple way to do that with the popular fetch API
301+
302+
```js
303+
// Using axios
304+
useQuery(['todos', todoId], async () => {
305+
const response = await axios.get('/todos/' + todoId)
306+
return response.data
307+
})
308+
309+
// Using fetch
310+
useQuery(['todos', todoId], async () => {
311+
const response = await fetch('/todos/' + todoId)
312+
if (!response.ok) {
313+
throw new Error('Network response was not ok')
314+
}
315+
return response.json()
316+
})
317+
```
318+
298319
299320
## React Query and React Router
300321

0 commit comments

Comments
 (0)