Tổng quan về phương pháp tấn công Deserialization trong PHP
Bài đăng này đã không được cập nhật trong 4 năm
Quá trình Serialization và Deserialization đối tượng
- Serialization là quá trình chuyển đổi trạng thái thông tin của một đối tượng thành một hình thức có thể được lưu trữ hoặc truyền đi. Các thông tin chuyển đổi có thể được lưu trữ trên một đĩa. Trong quá trình truyền qua mạng, nó có thể ở dạng byte, XML, JSON,…
- Deserialization là quá trình ngược lại của quá trình serialization, thực hiện lấy dữ liệu từ các định dạng có cấu trúc, khôi phục thông tin theo byte, XML, JSON,... thành các đối tượng .
Quá trình Serialization đối tượng:
- Trong ngôn ngữ PHP, hàm hỗ trợ Serialization đối tượng là
serialize( )
. Input của hàm này là một object và output của hàm sẽ là một chuỗi lưu object đó, cụ thể sẽ lưu class của object và các thuộc tính của object.
- Ở đây, khỏi tạo một class Student với các thuộc tính $name và $age, cùng với những phương thức
setName( )
,getName( )
,setAge( )
, getAge( ), và hai magic method__sleep( )
,__wakeup( )
. Tiếp theo khởi tạo một object $student và khởi tạo giá trị thuộc tính cho object $student thông qua hai phương thứcsetName( )
, vàsetAge( )
. - Dạng của object trước khi được serialize:
- Tiếp theo, gọi hàm serialize( ), tham số truyền vào phương thức là một object, ở truyền vào object $student. Hàm serialize( ) sẽ trả về một chuỗi lưu object được truyền vào. Chuỗi này sẽ được lưu vào $serializeStudent. Dạng của object sau khi được serialize:
Quá trình Deserialization đối tượng
- Trong ngôn ngữ PHP, hàm hỗ trợ Deserialization đối tượng là
unserialize( )
. Input của phương thức này là một chuỗi đại diện cho object. Output là object được xây dựng lại từ chuỗi truyền vào hàmunserialize( )
.
- Ở đây, sử dụng biến $serializeStudent truyền vào hàm
unserialize( )
để xây dựng lại object từ$serializeStudent
và lưu vào biến$deserializeStudent
. Sau đây là kết quả:
- Kết quả nhận được sau khi
deserialize
từ biến$serializeStudent
giống với dạng object trước khi đượcserialize
.
Giới thiệu về lỗ hổng Deserialization
- Trong TOP10 OWASP 2017, OWASP xếp lỗ hổng này ở vị trí thứ 8: Top 10-2017 A8- Insecure Deserialization. Ứng dụng mắc phải lỗ hổng này khi không kiểm tra dữ liệu đầu vào trước khi thực hiện deserialize. Dữ liệu không đúng định dạng hoặc dữ liệu không mong muốn có thể bị lợi dụng để thay đổi luồng xử lý của ứng dụng, gây ra hậu quả có thể là tấn công từ chối dịch vụ hoặc thực thi mã tùy ý.
- Lỗ hổng Deserialization trong PHP hay với một tên gọi khác là PHP Object Injection
có thể giúp kẻ tấn công thực hiện các loại tấn công khác nhau, chẳng hạn như
Code Injection
,SQL Injection
,Path Traversal
,DDos
, tùy thuộc vào ngữ cảnh. Lỗ hổng này xảy ra khi dữ liệu đầu vào không được kiểm tra đúng cách trước khi được chuyển đến hàm PHPunserialize()
. Với các lớp phương thức Magic method__wakeup()
,__destruct()
,__toString()
cùng với cácPOP chain
giúp cho đối tượng tấn công thực thi lỗi này. - Năm 2010, Stefan Ess có bài thuyết trình của tại hội nghị BlackHat đã đề cập đến những
nguy cơ tận dụng lỗi
PHP Object Injection
với nền tảng được xây dựng trên ngôn ngữ PHP là WordPress sử dụngserialize()
vàunserialize()
. Dựa vào đó, năm 2013, Tom Van Goethem, tiến sỹ tại Bỉ đã tìm được lỗ hổngPHP Object Injection
trên nền tảngWordPress
(version 3.6.1).Cũng trong năm 2013, một nền tảng khác làJoomla
cũng bị lỗ hổng tương tự(CVE2013-3242).
Đến năm 2015, Johannes Dahse , tiến sĩ người Đức đã tìm ra lỗ hổngPHP Object Injection
ở một phần mềm thương mại điện tử được xây dựng trên ngôn ngữPHP
làMagento(version 1.9.0.1)
. Theo sau đó, chứng kiến sự gia tăng về lỗ hổngDeserialization
trên nhiều ứng dụng web cho phép cả tấn công thi mã từ xa(RCE)
và quản lý máy tính của nạn nhân từ xa. Trong những năm gần đây có khoảng 60 lỗ hổngDeserialization
có thể thực thi mã từ xa, hơn 80 lỗ hổng đã được báo cáo. Năm 2017,OWASP
xếp lỗ hổngDeserialization
ở vị trí thứ8
trong top10 OWASP
. Những số liệu thống kê trên cho thấy sự nguy hiểm cũng như phổ biến của lỗ hổng trên nền tảngPHP
.
Kỹ thuật Property Oriented Programming
- Các lỗ hổng liên quan đến bộ nhớ như tràn bộ đệm, lỗi sai định dạng, đã được biết đến
từ lâu. Các phương pháp bảo vệ như
ASLR
vàDEP
được triển khai rộng rãi để chống lại việc khai thác các lỗ hổng trên. Tuy nhiên, kẻ tấn công có thể sử dụng các kỹ thuật khác để vượt qua lớp phòng thủ trên, ví dụ như kỹ thuậtCode reuse
. Các kỹ thuậtCode reuse
nhưReturntoLibc
,Return-Oriented Programming
vàJump-Oriented Programming
có thể vượt qua một số biện pháp phòng thủ trên. VớiROP
vàJOP
, kẻ tấn công có thể sử dụng lại các đoạn mã nguồn trong chương trình (được gọi làgadgets
) và kết hợp chúng lại với nhau để xây dựng thànhpayload
(gadget chains
). - Vào năm 2009, Esser đã trình diễn kỹ thuật
Code reuse
tồn tại trên ứng dụngPHP
. Cụ thể, anh đã giới thiệu khả năng tấn công chèn đối tượngWeb
mà kẻ tấn công có thể sửa đổi các thuộc tính của đối tượng. Do đó, dữ liệu và luồng điều khiển của ứng dụng có thể bị thay đổi. Và anh cũng là người đã đặt ra thuật ngữProperty-Oriented Programming
(POP
). Property-Oriented Programming
(POP
) là khi chúng ta có thể điều khiển thuộc tính của đối tượng và có thể làm ảnh hưởng đến luồng thực thi của chương trình. MộtPOP gadget
là đoạn mã mà chúng ta có thể làm ảnh hưởng đến thuộc tính của một số đối tượng. Đây chính là mộtROP
cấp cao (một kỹ thuật được sử dụng trong khai thác tham nhũng bộ nhớ). Thay vìROP gadget
đẩy giá trị lên ngăn xếp, thìPOP gadget
cho phép chúng ta ghi một số dữ liệu vàofile
. Tấn côngCode reuse
cũng có thể xảy ra dựa trên các đoạn mã đã có sẵn, được gọi làgadge
t, được thực thi để thực hiện một hành động không mong muốn như thực thi lệnh hệ thống tùy ý. Quá trìnhDeserialization
có thể đặt các giá trị của biến tùy ý, cho phép kẻ tấn công có thể kiểm soát một số dữ liệu. Điều này cũng cho phép kẻ tấn công sử dụng mộtgadget
để gọi mộtgadget
thứ hai, vì các phương thức thường được gọi trên các đối tượng và được lưu trữ cho các biến của thực thể. Khi một loại cácgadget
liên kết với nhau, nó được làgadget chain
.- Một điểm quan trọng ở đây là khai thác lỗ hổng
Deserialize
không phải là gửi các đoạncode
lên chương trình để thực thi. Chúng ta chỉ đơn giản là gửi thuộc tính của các lớp mà máy chủ đã có biết để thực hiện tác thao tác mã đã có, liên quan đến các thuộc tính đó. Để khai thác thành công lỗ hổngDeserialization
cần có 2 điều kiện:- Điểm đầu vào, là nơi mà kẻ tấn công gửi các dữ liệu đã được
serialize
đến mục tiêu và mục tiêu sẽ thực hiệndeserialize
đoạn dữ liệu này. - Kẻ tấn công có thể thao túng được một hoặc nhiều đoạn mã thông qua
quá trình
deserialize
.
- Điểm đầu vào, là nơi mà kẻ tấn công gửi các dữ liệu đã được
- Ví dụ, có hai điều kiện tiên quyết mà một ứng dụng
PHP
cần phải đáp ứng để có thể sử dụngPOP
để khai thác lỗ hổngPHP Object Injection
. Đầu tiên, có ít nhất mộtmagic method
được gọi lúc ứng dụng đang chạy, được xác định đúng trong lớp thuộc đối tượng mà kẻ tấn công muốninject
. Thứ hai, lớp được chọn cần phải được load trong phạm vi của lệnh gọiunserialize()
, và kẻ tấn công có thể điều khiển được đầu vào của hàmunserialize()
. - Trong kịch bản dưới đây, Có 3
gadgets
được kết hợp để tạo thành lỗ hổng xoá file tuỳ ý.
- Lỗ hổng
POI
tồn tại ở dòng19
, nơi mà người dùng có thể nhập dữ liệu đầu vào và chương trình thực hiệndeserialize
. Ở dòng20
,21
kẻ tấn công đã chèn một đối tượngDatabase
với thuộc tínhhandle
được đặt là một đối tượng của lớpTempFile
. Thuộc tínhfilename
được đặt với nội dung là../../.htaccess
. - Khi ứng dụng kết thúc, đối tượng database đã được chèn sẽ được thực thi nhờ hàm
__destruct()
tự động gọi. Bởi vì thuộc tính handle đã được chèn thành đối tượng lớpTempFile
, gọi đến hàmshutdown()
của lớpFile
. Trong hàmshutdown()
sẽ gọi đến phương thứcclose
được kế thừa ở lớpTempFile
. Ở đây, vớifilename
màattacker
truyền vào sẽ được chương trình xoá file đó. Kết quả là file.htaccess
đã bị xoá.
Tobe continue...
All rights reserved