Skip to content

Commit 2276d32

Browse files
committed
2019-12-01-ctfzone-quals + classifier9000
1 parent 191b261 commit 2276d32

File tree

7 files changed

+101
-0
lines changed

7 files changed

+101
-0
lines changed

2019-12-01-ctfzone-quals/README.md

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
# CTFZone 2019 quals
2+
3+
Team: (...)
4+
5+
### Table of contents
6+
7+
* [Classifier9000](classifier9000)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
# Classifier9000
2+
3+
In this task we have access to a neural network image classifier which returns `(predictions, percentage_probabilities)` pair for each input image. We also have starting image "car.jpg" which is classified as `sports_car`. The task is to forge this image to be classified as `racer`.
4+
5+
![car.jpg](car.jpg)
6+
7+
The classifier has web interface and its source code in python is available. The analysis of source code (server.py) revealed two important observations:
8+
9+
- the input image is required to be the same size as "car.jpg" which is 224 x 224 px,
10+
- the relative difference between the input image and "car.jpg" cannot exceed 2.
11+
12+
The first step was to obtain any image which is classified as `racer`. Eventually we found appropriate image and saved it as "racer.jpg"
13+
14+
![racer.jpg](racer.jpg)
15+
16+
The initial idea was to modify this image to exploit `diff(I, BASE_IMAGE) > 2` condition. It may be achieved when the output value of `diff` function is `nan`, for example with following transformation:
17+
18+
```
19+
I = np.asarray(Image.open('racer.jpg'))
20+
I2 = 1.1 * I - np.log(I)
21+
```
22+
23+
and new image I2 was still classified as `racer` with 14.859884977340698 probability.
24+
25+
However we could not directly pass the image from the memory to web classifier. We first needed to save it as a file. Unfortunately, during this operation pixel values were rescaled to 0-255 range. This is why this approach was abandoned.
26+
27+
28+
Another idea was to use both "car.jpg" and "racer.jpg" to create falsified image. The new image should be very similar to "car.jpg" but with some characteristics from "racer.jpg" to deceive the classifier. This was successful with the following code:
29+
30+
```
31+
I3 = 0.8 * BASE_IMAGE + 0.1 * I**2
32+
mpimg.imsave('car2.jpg', I3.astype('uint8'))
33+
I3 = np.asarray(Image.open('car2.jpg'))
34+
prediction.predictImage(I3, result_count=1, input_type="array")
35+
```
36+
37+
![car2.jpg](car2.jpg)
38+
39+
The result of `diff(I3, BASE_IMAGE)` is 1.4934139792559178 which is less than 2. And the output of `prediction.predictImage(I3, result_count=1, input_type="array")` is `racer` with 40.07074534893036 probability.
40+
41+
Finally we have the flag! `ctfzone{f1r3-f1r3-pwn-7h3-cl4551f13r}`
14.9 KB
Loading
26.8 KB
Loading
Loading
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,52 @@
1+
from flask import Flask, request, redirect
2+
from imageai.Prediction import ImagePrediction
3+
import numpy as np
4+
from PIL import Image
5+
import os
6+
7+
app = Flask(__name__)
8+
9+
10+
# Heard that these kind of models are vulnerable to adversarial attacks
11+
prediction = ImagePrediction()
12+
prediction.setModelTypeAsResNet()
13+
prediction.setModelPath("resnet50_weights_tf_dim_ordering_tf_kernels.h5")
14+
prediction.loadModel()
15+
16+
BASE_IMAGE = np.asarray(Image.open("car.jpg"))
17+
18+
def normalize(arr):
19+
rng = arr.max()-arr.min()
20+
amin = arr.min()
21+
return (arr-amin)/rng
22+
23+
24+
def diff(arr1, arr2):
25+
arr1 = normalize(arr1)
26+
arr2 = normalize(arr2)
27+
distance = arr1 - arr2
28+
return np.max(np.sum(np.square(distance).reshape(arr1.shape[0], -1), axis=1))
29+
30+
31+
32+
@app.route('/task/upload/', methods=['POST',])
33+
def hello_world():
34+
answer = {
35+
"status": "fail to recognize image"
36+
}
37+
if request.method == 'POST':
38+
f = request.files['file']
39+
I = np.asarray(Image.open(f))
40+
if I.shape != (224,224, 3):
41+
return redirect(os.getenv("base_url", "http://0.0.0.0:8000") + "/fail.html")
42+
if diff(I, BASE_IMAGE) > 2:
43+
return redirect(os.getenv("base_url","http://0.0.0.0:8000" ) + "/fail.html")
44+
predictions, percentage_probabilities = prediction.predictImage(I, result_count=1, input_type="array")
45+
answer['classification_result'] = predictions[0]
46+
if answer['classification_result'] == 'racer':
47+
return {"flag": os.getenv("FLAG", "tutbudetflag")}
48+
return redirect(os.getenv("base_url", "http://0.0.0.0:8000" ) + "/sportscar.html")
49+
50+
51+
if __name__ == '__main__':
52+
app.run("0.0.0.0", 1488)

README.md

+1
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22

33
## 2019
44

5+
* [2019.12.01 **CTFZone 2019 quals**(9th place/1040 teams)](2019-12-01-ctfzone-quals)
56
* [2019.11.16 **ASIS CTF 2019 Finals**(5th place/357 teams)](2019-11-16-asis-finals)
67
* [2019.11.14 **Dragon CTF 2019 Finals**(1st place/14 teams) on site](2019-11-14-dragon-finals)
78
* [2019.11.07 **DefCamp CTF 2019 Finals**(2nd place/16 teams) on site](2019-11-07-defcamp-finals)

0 commit comments

Comments
 (0)