Pages

Thursday, January 6, 2022

Rescuing a WordPress-based site on Amazon LightSail and moving to another VPS

Don't get me wrong, Amazon LightSail is one of the most affordable VPS services out there.  A big name such as Amazon might give you the feeling of stability, reliability, and so forth. I have no complaint at all so far as my clients are happy with that.

Yet, they still want to move the servers closer to their customers, and given that Amazon is still not available in Vietnam yet, it becomes a new mission. Turn out the limitation of the VPS in terms of CPU and RAM might give me another kind of headache. Normally, I have no problem with moving the site to a new server especially in the case of WordPress as there are tons of already-built plugins on the market.

So "rescuing" in this post simply means it seems impossible to backup a WordPress with a limited source VPS (say 1GB of RAM).

Here are the defined steps I often do to clone and move a WordPress-based website using the Duplicator plugin.

1) Updating WordPress and all active plugins. Remove inactive ones.

2) Clone the current site using Duplicator.

3) Upload the package issued by Duplicator on a brand new VPS and then run installer.php (this file is given by Duplicator). During this process, you will have to create a new database on the new VPS.

4) Make some changes if needed.

However, the above process does not work well as Duplicator cannot clone the whole website when the RAM of the current VPS is limited. And this time it hurt me badly.


Here are what I did to handle it.

1) Cloned the current VPS on LightSail using the snapshot feature and then restored on a new computer with higher resources (4GB of RAM for example).

2) Run Duplicator. This time it worked

3) Following the above steps to restore the package given by Duplicator.

4) Check the site and then change DNS on Cloudflare.

5) Check new DNS via https://check-host.net/ or https://dnschecker.org/

6) Note: for those who are using aapanel, you might need to fix the 404 error as nginx does not handle the .htaccess file. Take a look at the following image:

FYI: In case you use Plesk, then you might take advantage of the Migration feature which allows you to clone and restore your server to another one in a few minutes. I might write another post about that.

Friday, December 24, 2021

Notes when installing WireGuard VPN with Docker (Portainer) and wg-easy

Here are some notices when attempting to install a secure and fast VPN with WireGuard.

- WireGuard and Outline are both great, easy to install on both servers and clients no matter what. If you are familiar with Docker, you can deploy it with a few clicks.

I tried to give WireGuard a shot. The most elegant way to deploy a docker-based container is by implementing Portainer. FYI, Portainer is built by Auckland guys.

PORTAINER WAY: You can simply hit some clicks to customize the ports and configure authentication with Portainer. The docker image on the Docker hub is given below: 

https://hub.docker.com/r/weejewel/wg-easy

COMMAND LINE WAY: You can also install docker using the command `docker-composer up` as usual if you prefer to terminal. The nice thing is that you can also do that within Portainer. So Portainer wins, definitely.

https://raw.githubusercontent.com/WeeJeWel/wg-easy/master/docker-compose.yml


Image: I installed Dante shock5, Outline, and WireGuard on the same, cheap VPS

Noted that you should open two ports 51821 (TCP) and 51820 (UDP) for VPN. You can find this detailed information in the file docker-compose.yml mentioned above.

      - "51820:51820/udp" ----> CRITICAL INFORMATION SHOULD BE NOTED

      - "51821:51821/tcp" ----> CRITICAL INFORMATION SHOULD BE NOTED

You also need to replace your public IP address within the file docker-compose.yml.

It appears that wg-easy cannot be installed on an ARM-based chip. As far as I know, go-lang seems not wholely support this kind of CPU.

My first impression is that it runs really fast and it is easy to configure WireGuard on mobile devices. So after logging to WireGuard via port 51821 (say http://your_public_id_address:51821), you can CRUD a new profile and then give it to your friends if you wish.

Yet, installing WireGuard as a client on Ubuntu seems to take time. Here are some links: 

 https://www.thomas-krenn.com/en/wiki/Ubuntu_Desktop_as_WireGuard_VPN_client_configuration

https://tech.serhatteker.com/post/2021-01/how-to-set-up-wireguard-client-on-ubuntu-desktop/

Todo list: I am thinking about installing nginx proxy manager to manage WireGuard via accessing a real domain. But it does not matter much. FYI: WireGuard is really fast and does not take much CPU or RAM.


PS: It seems that using the traditional method (command-line based) with docker-compose up -d is the workable way to install WireGuard. Then I used Portainer to manage it with a breeze. Normally Portainer serves me pretty well but somehow this time I cannot install WireGuard via this great UI tool frequently used to manage docker. Yet, I can still use this for monitoring WireGuard.

FYI: You might need to add MU value to the server's configuration as discussed here, another one and there.

MTU = 1280 (client, say Ubuntu desktop)

[Add the line MTU = 1280 in the wireguard server interface at: /etc/wireguard/wg0.conf if you are using Ubuntu server]

Updated: I can now install WireGuard with Portainer. You might need to make some changes regarding the right to trigger WireGuard. So NET_ADMIN should be granted.

Updated: Installing Wireguard on Ubuntu client is much much easier than I think. 

1) First, download the configuration file from WireGuard UI above.

2) Install WireGuard on the Ubuntu client.

3) Rename the file above to wg0.conf and put to /etc/wireguard/wg0.conf'

4) Run sudo wg-quick up wg0

5) Check your newest IP with ip2location.com or through sudo wg

Don't forget sudo wg-quick down wg0 if you want to turn the VPN off.

Updated: Watchtower is not recommended to use just in case you want to update docker as it breaks the configuration. 


Friday, December 3, 2021

Khế ước ngẫu nhiên

 Để tạo một số ngẫu nhiên từ 0 tới x, ta phải nhân số cực trên, chính là x, với Math.random().

Giả sử x là hai từ/tiếng đầu tiên trong Kiều, như người ta nói, đời người, số random kia ắt sẽ mỗi người một phận, đã được destined đặt trong khế ước nào đấy, nếu có, với thượng đế.

Nếu có ai đó, như mệ, mà cuộc đời ưu ái thương tặng con số random ấy lớn hơn số cực đại kia, thì cũng là ơn riêng.

Xin vĩnh biệt mệ, người vẫn hay cho chúng tôi những con cá khô mỗi lần giáp hạt.

Trên đời lại vắng một người thỉnh thoảng hỏi tôi, hế lô, chúng mày kiếm ăn ra sao. À, ý là cần cá khô thì mần răng tao không biết, cứ mò về với ngôi nhà có rặng phi lao nhé.

Ai cũng ngơ ngác. Con mèo chắc cũng thế, khi căn nhà bắt đầu vắng tiếng ai đó…

Biển đang cuộn sóng. Nước quanh làng lềnh phềnh. Góc nhà, một bịch cá khô lủ khủ vẫn lủng lẳng bất cần. À, cho những đứa nào cần nhé. Của mệ đấy.


Thursday, May 20, 2021

Covid vào thành

 Nổi tiếng nhất huyện tôi là Chí Phèo. Nổi tiếng nhất làng tôi là Tạ Biên Cương.

Giờ làng nổi lềnh phềnh trên mặt báo vì Covid 19. Ngày thứ hai bị phong tỏa với gần chục ca. Sáng nay, sau khi một cháu lớp đệ nhất dương tính, cả lớp cháu mấy chục bé rồng rắn tí xíu "lên đường" cách li 21 ngày. Bố mẹ nước mắt trông theo. Biết làm sao.

Tôi cũng không biết phải làm sao. Chỉ mong tất cả mọi người bình tĩnh và bình an. Sinh ra và lớn lên ở làng, bạn sẽ như có họ với tất cả mọi người vậy.

Mẹ tôi chắc lại nói, đấy, ma quỷ con ạ. Có thể covid là một trò đùa của Thanos với mã vạch 69X. Mà kệ nó đi mẹ ạ. Mẹ cứ ở trong nhà với đàn gà là được. Thóc còn rất nhiều đúng không ?

Tuesday, May 11, 2021

Covid vây thành

 Làng tôi cách tâm dịch covid lần này có một cánh đồng. Không thể không hoảng hốt khi nghe em tôi nói, covid nó đang đánh võng quanh làng rồi anh ạ. 

Không biết những vạt hoa trồng vội trên cánh đồng có trổ bông mùa này? Tiếng chuông nhà thờ vẫn sẽ bình an như thế? Tôi thật không dám tưởng tượng khi nhớ lại cảnh người ta "thui" các xác chết một cách vội vàng và xuề xòa ở đẩu đâu Ấn Độ.

Trong cơn mưa tầm tã tôi tiễn cha, lần cuối, lên cánh đồng làng gần 10 năm trước, có những khoảng khắc tôi đã cảm thấy mình bất lực đến tuyệt vọng. Từng bước chân một tiễn ông, từng ngôi nhà xe đi qua, những cơn thốc giật của giông gió... Từng giây phút một, tôi có cảm giác mình và mọi người như đang lang thang trong một cái địa ngục trần ai vậy, lửa phần phật thổi, chẳng có ai đuổi phía sau, nhưng phía trước vô định và tuyệt vọng.

Bất chợt cái cảm giác ấy trở lại trong một chiều hay tin covid đã đánh thành tới nơi rồi.

Chỉ mong là, có lẽ, tôi đã nghĩ quá nhiều.

Friday, March 5, 2021

phận thợ dạy

- “Dạ cụ, bạn cụ, cậu thợ dạy lại gọi điện thoại ạ!”

- “Lại hẹn nợ tạm bợ? ”, cụ Nghị bụng phệ, giọng lụ khụ vặn vẹo. 

Thợ dạy lạch bạch chạy lại đoạn cụ Nghị ngụ.

- “Dạ cụ, bệnh dịch đại nạn, định bụng mượn cụ chục bịch gạo, hẹn nợ một chặp ạ...", thợ dạy ngượng nghịu.

- "Thợ điện mượn tạ gạo, thợ dạy mượn chục bị. Bệnh dịch, thợ được dạy, tạm vậy. Chạy vạy đặng vượt đại dịch. Mẹ bọn khựa, loại dựng chuyện bệnh tật. Chuyện gạo thợ dạy nợ... “ cụ Nghị giận. Đoạn “thợ dạy mượn gạo”, giọng cụ dịu lại.

- “Dạ cụ, thật ngại. Phận thợ dạy đạo mạo, gặp dịch bụng sạch gạo, tội thật. Cụ thiệt lợi hại. Thợ dạy nợ cụ”, thợ dạy biện bạch.

- “Được”, cụ Nghị gật gật.

- “Dạ, dạ cụ...”, thợ dạy cậy miệng.

- “Thợ dạy, cậu bạn nhậu cụ, dạy cậu đại cụ, lại ngại ngại vậy?”,

 - “Dạ cụ, thiệt ngại. Thợ dạy định ...định... mượn cụ một triệu ạ”, thợ dạy sượng miệng.

- “Một triệu?”, cụ Nghị định bụng thụi thợ dạy một bịch.

-“Dạ, hẹn cụ nợ một triệu, cộng chục bịch gạo mượn cụ đợt nọ ạ”.

-“Liệu thợ dạy định bụng xạo cụ? Cụ cực giận bọn xạo xạo”. 

Phận thợ dạy, mọi việc trọng đại tạm gạt, lục đục sự nghiệp học, dạy học. Cụ Nghị lặng lặng. Lộn ruột, thợ dạy bợ đại cục gạch, định bụng... đập bậy.

-"Được, thợ dạy ạ, cậu bạn cụ, một triệu vậy”, cụ Nghị sợ chuyện bậy. Thực bụng, đợt bệnh dịch, cụ định bụng... kệ cụ bọn thợ dạy. Trộ mặt thợ dạy giận giận, cụ sợ.

- "Dạ, cụ rộng lượng ạ. Cụ nhận một lậy ạ”, thợ dạy loạng quạng. 

Đoạn, thợ dạy chạy một mạch lại chợ. Một triệu rượu thịt. Một lậy một triệu. 

Chuyện thợ dạy dại dột phận thợ nhậu, một chuyện lạ... một chuyện bịa bậy bạ...

Monday, February 22, 2021

Saturday, February 13, 2021

phụ nữ Huế

 Tôi phải lấy làm ngưỡng mộ những o những mệ người Huế.

Phụ nữ Việt, sinh sống thôn quê, phần đông vai u thịt bắp, gánh gồng gia đình. Nuôi heo đặng nuôi con. Phần lớn khó nghèo và đáng ngưỡng mộ.

Nhưng ở Huế, cái đói khổ có phần khốc liệt hơn. Những người đàn bà suốt đời, phần lớn, bị trói buộc vào rất rất nhiều những quy định không thành luật của tập tục lề thói, khiến cho họ đánh mất đi sự tự do và bản năng sáng tạo. Những cuộc đời lặp lại trong vô thức. Nhẫn nhục, tuân phục và không dám phẫn nộ hoặc phản kháng.

Trái ngược với sự chặt chẽ đến khô khan của tục lệ mà người phụ nữ tuân phục, đa số nam nhân được ưu ái và do đó thường phóng túng, phóng khoáng hơn hẳn, tuy rằng kha khá người, vì thế, kỉ luật yếu. 

Nhiều nhà thơ mà không phải văn, có thể từ đây xuất hiện. 

Những người phụ nữ tội nghiệp và yếu đuối nhẫn nhịn đến trọn đời. 

Việc cuối cùng họ thường làm là trao lại của cải và cả thái độ sống ấy cho con cháu. Như một di sản.

Chúng ta không có bước nhảy nào cả. Sự phóng túng nam nhân và sự nhẫn nhịn khắc kỉ nữ giới, ở đây, vẫn như hai đoàn tàu ngược chiều.

Friday, March 30, 2018

PHP và UTF-8

Hiện PHP chưa (thật sự) hỗ trợ Unicode ở tầng thấp, do đó, để hiển thị được chuỗi UTF-8, bắt buộc phải có thêm các thao tác xử lý bổ sung, trên web, HTML, SQL.

Bài này mình tóm lược lại về UTF8 và PHP.

UTF-8 ở chính ngôn ngữ PHP

Giả sử ta có character a, á, ư. Thử nghiệm với hai cặp hàm strlen()/mb_strlen() và strpos()/mb_strpos().

Kết quả với hàm tính lượng bytes có kết quả tương ứng như sau:

echo strlen('a'); //1
echo mb_strlen('a'); //1
echo strlen('á'); //2
echo mb_strlen('á'); //1

Ở đây có thể tạm kết luận, với các string multi-bytes (chẳng hạn theo chuẩn Unicode như tiếng Việt),  số lượng bytes sẽ khác nhau giữa hàm strlen() và mb_strlen().
 
Kết quả với hàm tìm character có trong string:

echo strpos("mana", "a"); //1
echo strpos("mán", "á"); //1
echo mb_strpos("mana", "a"); //1

echo strpos("mưán", "á"); //3
echo mb_strpos("mưán", "á"); //2


echo strpos("mán ư", "ư"); //5
echo mb_strpos("mán ư", "ư"); //4

Hàm tìm character cũng vậy, có sự khác biệt rất lớn giữa hàm strpos() và mb_strpos().

Như vậy ta buộc phải dùng các hàm có dạng mb_* khi xử lý string Unicode, đây là các hàm chuyên trị cho Unicode. Tuy nhiên, không phải hàm xử lý chuỗi nào cũng có hàm mb_ tương ứng.

Bạn có thể thiết lập mb_internal_encoding() ở đầu mỗi file PHP và hàm mb_http_output() ngay ở vị trí PHP xuất ra dữ liệu ngoài trình duyệt.

Ngoài ra, nhiều hàm PHP xử lý string có thể có thêm tham số xác định dạng mã hóa. Chẳng hạn như  htmlentities().

UTF-8 ở trên hệ điều hành

Hiện nay PHP có thể chạy ở hầu hết các hệ điều hành, gồm cả Linux và Windows. Nhưng cách PHP xử lý tên file ở mỗi hệ điều hành là có thể khác nhau, trong đó, hỗ trợ Linux tốt nhất. Chẳng hạn trên Windows, nếu dùng PHP tạo một file với mã non-ASCII, lỗi có thể xuất hiện. Trên Linux và OSX, bạn có thể mã hóa tên file dạng UTF-8, nhưng trên Windows, buộc phải dùng chuẩn ISO-8859-1.

UTF-8 khi mần ăn với MySQ 

Lưu ý 1: Để chắc chắn là chuỗi trên PHP sang MYSQL lưu ở định dạng UTF-8, hãy thiết lập character và collation là utf8bm4 (theo kinh nghiệm của mình có khả năng hiển thị các kí tự "lạ" nhiều hơn UTF8 thông thường, chẳng hạn có lần mình làm về phiên âm tiếng Anh thì chỉ khi thiết lập utf8bm4, hệ thống mới hiển thị được).

Lưu ý 2: Phải thiết lập trong kết nối với mysql khi viết bằng PHP. Giờ thì chắc là toàn dân xài PDO nên mình có thể thiết lập như sau:


$db
= new PDO('dblib:host=your_hostname;dbname=your_db;charset=UTF-8', $user, $pass);


Với PHP 5.3.6, tùy chọn charset chưa có nên có thể thực hiện theo tùy chọn sau:


$pdo = new PDO(
    'mysql:host=hostname;dbname=defaultDbName',
    'username',
    'password',
    array(PDO::MYSQL_ATTR_INIT_COMMAND => "SET NAMES utf8")
);

Với mysqli, ta buộc phải thiết lập như sau:

$conn = mysqli_connect('localhost','db_username','password','your_database_name');
 mysqli_set_charset($conn,"utf8"); 

UTF-8 trên trình duyệt

Để chắc chắn là PHP sẽ xuất UTF-8, có thể dùng hàm mb_http_output(). Trên trình duyệt thì ta thường thiết lập  tùy chọn meta charset như sau, đặt trong thẻ <head>
  <meta charset="utf-8">

https://phpbestpractices.org/#utf-8

https://stackoverflow.com/questions/4475548/pdo-mysql-and-broken-utf-8-encoding

Thursday, March 29, 2018

[PHP basic] Chuyển string sang array với split(), str_split(), explode() và pre_split()

Vốn dĩ một chuỗi string trong PHP có thể coi là một array, nhưng chuỗi này chuyển sang array lại dựa vào số byte, và một cách nào đấy có thể tạm nói rằng, khi chuyển qua array thì dựa vào character.

Ví dụ "hello world" thì khi loop với for ta được h-e-l-l-o-[space]-w-o-r-l-d.

Giờ ta muốn tách chuỗi theo các nhu cầu khác thì sao? PHP có sẵn 4 hàm split(), str_split(), explode() và pre_split() để chuyển sang array.

1) split() chú ý là hàm này đã bị loại bỏ trên PHP7, vẫn dùng được trên PHP5 nhưng sẽ có warning lỗi. PHP4 thì okey.

2) str_split() sinh ra để thay thế split() với tính năng tương tự, hoạt động okey trên PHP7.

str_split(string,length) . Như vậy delimiter nằm phía phải, đơn vị là số.

Diễn giải: hàm str_split() sẽ tách chuỗi thành array và việc tách chuỗi này tính theo số lượng character được quy định trong delimiter. Mặc định nếu không truyền vào là 1.

Giả sử:

$str = "This is a text which will be used to do something.";

print_r(str_split($str, 1)); // Array ( [0] => T [1] => h [2] => i [3] => s [4] => [5] => i [6] => s [7] => [8] => a [9] => [10] => t [11] => e [12] => x [13] => t [14] => [15] => w [16] => h [17] => i [18] => c [19] => h [20] => [21] => w [22] => i [23] => l [24] => l [25] => [26] => b [27] => e [28] => [29] => u [30] => s [31] => e [32] => d [33] => [34] => t [35] => o [36] => [37] => d [38] => o [39] => [40] => s [41] => o [42] => m [43] => e [44] => t [45] => h [46] => i [47] => n [48] => g [49] => .

print_r(str_split($str, 5));//Array ( [0] => This [1] => is a [2] => text [3] => which [4] => will [5] => be u [6] => sed t [7] => o do [8] => somet [9] => hing. )

 3) explode() cắt string theo string.

cú pháp: explode ( string $delimiter , string $string [, int $limit ] ).

Như vậy delimiter nằm phía trái. Việc chuyển chuỗi sang array tính theo chính character, không phải theo số lượng character như làm str_split(). Mặc định tham số thứ 2 của hàm str_split() là 1 nên không cần truyền vào tham số thứ 2 nếu muốn chuyển chuỗi dựa vào 1 character, nhưng explode() bắt buộc phải có delimiter.

ví dụ 1:

$str = "This is a text which will be used to do something.";

print_r(explode("t",$str)); //Array ( [0] => This is a [1] => ex [2] => which will be used [3] => o do some [4] => hing. ) 

ví dụ 2:
$str = "dạy nhau học";

print_r(explode("h", $str));//Array ( [0] => dạy n [1] => au [2] => ọc )

Ví dụ 3: 
$str = "Dạy |sớm |dạy| nhau| học";

print_r(explode("|",$str));//Array ( [0] => Dạy [1] => sớm [2] => dạy [3] => nhau [4] => học )

Nhìn ví dụ 3 sẽ thấy một đặc điểm quan trọng của explode() là nó sẽ xóa chính character là delimiter sau khi convert. Mình hay dùng cái này nhất để tách dữ liệu text và chuyển vào database.



Mình tham khảo tại: https://dotoplay.wordpress.com/2015/08/20/differences-between-str_split-and-explode-in-php/