728x90
반응형
728x90
반응형
728x90
반응형

2020 Defenit CTF Write-up 6/5 09:00 ~ 6/7 09:00 (UTC, 48h)

 

 

 

Forensic

Baby Steganography

I heared you can find hide data in Audio Sub Bit. Do you want to look for it?

 

 

.wav 파일이 주어진다. 문제 지문을 보면 Audio Sub Bit에 데이터를 숨긴다고 한다.

 

Audio Sub Bit...

 

Sub Bit...

 

SB...

 

LSB?

 

LSB가 아닐까 생각을 해보고서 바로 구글링을 했다.

 

 

 

https://medium.com/@sumit.arora/audio-steganography-the-art-of-hiding-secrets-within-earshot-part-2-of-2-c76b1be719b3

 

Audio Steganography : The art of hiding secrets within earshot (part 2 of 2)

In this article we take a look at methods and tools along with code walk-through for performing Audio Steganography.

medium.com

Audio Steganography : LSB

 

 

포스트를 내리다보면 Python으로 작성된 Decode 코드가 있다. 바로 긁어다가 실행해 보았다.

 

 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
# Use wave package (native to Python) for reading the received audio file
import wave
song = wave.open("problem", mode='rb')
# Convert audio to byte array
frame_bytes = bytearray(list(song.readframes(song.getnframes())))
 
# Extract the LSB of each byte
extracted = [frame_bytes[i] & 1 for i in range(len(frame_bytes))]
# Convert byte array back to string
string = "".join(chr(int("".join(map(str,extracted[i:i+8])),2)) for i in range(0,len(extracted),8))
# Cut off at the filler characters
decoded = string.split("###")[0]
 
# Print the extracted text
print("Sucessfully decoded: "+decoded)
song.close()
 
cs

 

Sucessfully decoded: Defenit{Y0u_knOw_tH3_@uD10_5t39@No9rAphy?!}

 

결과는 대성공.

 

 

 

 


Misc

QR Generator

Escape from QR devil!

 

nc 연결을 하자.

 

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
Let's START!
 
< QR >
1 1 1 1 1 1 1 0 1 1 0 0 0 1 1 0 1 1 1 1 1 0 1 1 1 1 1 1 1 
1 0 0 0 0 0 1 0 1 0 1 1 1 0 0 0 1 0 0 0 0 0 1 0 0 0 0 0 1 
1 0 1 1 1 0 1 0 1 0 1 0 1 1 1 0 1 0 0 1 1 0 1 0 1 1 1 0 1 
1 0 1 1 1 0 1 0 1 1 0 0 1 1 0 1 1 0 1 0 1 0 1 0 1 1 1 0 1 
1 0 1 1 1 0 1 0 1 0 1 0 1 0 0 1 0 1 0 0 1 0 1 0 1 1 1 0 1 
1 0 0 0 0 0 1 0 0 0 1 0 0 0 0 0 0 1 0 0 0 0 1 0 0 0 0 0 1 
1 1 1 1 1 1 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 0 1 1 1 1 1 1 1 
0 0 0 0 0 0 0 0 1 1 1 0 1 1 1 0 0 0 1 1 1 0 0 0 0 0 0 0 0 
0 1 0 1 0 1 1 0 1 0 0 1 0 1 1 0 0 0 1 1 0 1 1 0 1 1 1 1 1 
0 1 1 1 1 1 0 1 1 0 1 1 1 1 0 0 0 0 1 1 0 1 0 0 1 1 0 1 1 
0 1 0 1 0 1 1 1 1 0 1 1 0 0 1 1 1 1 0 1 1 0 0 0 0 0 0 0 1 
1 0 1 0 1 0 0 1 1 1 0 0 1 0 0 0 0 1 0 0 0 0 0 1 1 0 0 0 1 
0 0 0 0 1 0 1 1 1 0 0 1 0 0 0 0 0 0 1 0 0 0 1 0 1 0 1 0 1 
1 1 0 1 0 1 0 0 1 0 1 0 0 0 0 0 1 1 1 0 1 0 1 0 1 0 1 0 1 
0 1 1 0 1 0 1 1 1 1 1 1 1 0 1 0 1 1 0 0 1 0 0 1 0 1 0 1 1 
0 1 1 1 1 1 0 1 1 1 1 0 1 1 1 0 1 1 1 1 1 0 1 0 1 0 1 0 1 
1 0 0 1 1 0 1 1 1 1 1 0 0 1 1 0 0 1 0 1 1 0 0 1 1 1 0 0 1 
1 0 0 1 0 0 0 0 1 1 0 0 1 0 0 0 1 0 0 0 1 0 1 0 0 1 0 0 0 
1 1 0 0 1 0 1 1 0 0 1 1 0 1 0 0 1 0 1 1 0 1 1 0 0 0 1 1 0 
0 0 1 1 0 0 0 0 0 0 0 0 0 1 0 0 1 1 0 1 0 0 0 1 0 1 0 1 1 
1 0 0 0 1 0 1 0 1 1 1 1 1 0 0 1 0 1 1 0 1 1 1 1 1 0 0 1 0 
0 0 0 0 0 0 0 0 1 0 1 0 0 0 1 1 0 0 1 1 1 0 0 0 1 0 0 0 0 
1 1 1 1 1 1 1 0 0 0 1 0 0 1 1 0 0 0 0 1 1 0 1 0 1 0 0 0 1 
1 0 0 0 0 0 1 0 1 1 1 1 0 1 0 1 1 1 0 0 1 0 0 0 1 1 1 1 0 
1 0 1 1 1 0 1 0 0 1 0 1 0 1 0 1 1 1 1 1 1 1 1 1 1 1 1 0 0 
1 0 1 1 1 0 1 0 1 0 1 1 1 1 1 1 0 0 0 1 1 0 0 1 0 1 1 1 0 
1 0 1 1 1 0 1 0 0 0 1 0 0 0 0 0 1 1 0 0 1 1 0 0 0 0 0 0 0 
1 0 0 0 0 0 1 0 1 0 0 0 0 1 0 1 0 0 0 0 1 1 0 0 0 0 1 1 0 
1 1 1 1 1 1 1 0 0 0 0 1 1 0 0 0 0 1 0 0 0 1 0 0 0 1 0 0 0 
 
STAGE 1
>> Time Over!
 
cs

 

1과 0으로 된 qrcode를 보내준다.

 

하얀배경 이미지를 불러와서 값이 1이면 해당 위치의 픽셀의 값을 검정으로 바꾸도록 코드를 쓱-싹 작성해 준다.

하얀배경 이미지는 포토샵을 이용한다. 넉넉하게 200px*200px

 

 

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
from pwn import *
from PIL import Image
 
= remote("qr-generator.ctf.defenit.kr"9000)
p.recvuntil('name?')
p.sendline('M4ndU')
img = Image.open('./200.png') #load white background 200*200 image
img = img.convert("RGB")
newData = []
 
qrbin = [0* 200
p.recvuntil('< QR >\n')
for i in range(200):
    data = str(p.recvline())
    data = data.replace("b'""")
    data = data.replace("\\n'""")
    data = data.replace(" """)
    if data == "":
        print(p.recvline())
        break
    qrbin[i] = data
 
for x in range(len(qrbin)):
    if qrbin[x] == 0:
        break
    for y in range(len(str(qrbin[x]))):
        if qrbin[x][y] == "1":
            img.putpixel( (x, y), (000)) #black
 
img.save("./dimg/TransparentImage"+str(n)+".png""PNG")
 
 
p.interactive()
 
cs

 

그러면 이러한 qrcode 이미지를 만들어 낼 수 있다.

(제공되는 qrcode 크기가 제각각 이어서 배경 이미지의 크기에 여유를 주었다. 인식하는데에는 문제가 없었다.)

 

 

이제 이 qrcode를 decoding해서 보내주기만 하면 된다. 그러나 이 과정에서 삽질이 있었다.

 

이 qrcode이미지를 디코딩하는 방법에는 여러가지가 있다.

qrcode를 디코딩해주는 사이트에 리퀘스트 날려서 크롤링하는 방법. (이미지 보내고 크롤링 해야되는 무조건 버리고)

파이썬 모듈을 이용해서 디코딩 하는 방법.

 

qrcode를 디코딩할 수 있는 파이썬 모듈에는 여러가지가 있었다.

 

먼저 qrtools.

 

$sudo pip3 install qrtools

 

      qr = qrtools.QR()

AttributeError: module 'qrtools' has no attribute 'QR'

 

음? (오류 1스택)

파이썬 2로 다시시도..

 

 

$sudo pip install qrtools

 

import qrtools
ImportError: No module named qrtools

 

??????? (오류 2스택)

 

다음 시도한 것이 qrcode

 

 d = qrcode.Decoder()
AttributeError: module 'qrcode' has no attribute 'Decoder'

 

문서에서는 이렇게 쓰라고 나와있었는데 뭐지..(오류 3스택)

 

 

pyzbar를 사용해볼까

>>> decode(img)
[]

 

도대체 되는게 뭡니까(오류 4스택)

 

 

$ sudo pip3 install pyqrcode

$ sudo apt-get install python-qrtools

 

>>> import pyqrcode
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
ImportError: No module named pyqrcode
(오류 5스택)

 

>>> import qrtools

>>> qr = qrtools.QR()

>>> qr.decode("./TransparentImage.png")
False

(오류 6스택)

 

AttributeError: module 'pyqrcode' has no attribute 'Decoder'

(오류 7스택)

 

 

...

 

웹으로 해결할까 생각이 들어서 테스트때 디코딩에 성공했던 ZXng Decode Online에 들어가봤다.

 

https://zxing.org/w/decode.jspx

 

ZXing Decoder Online

 

zxing.org

하단 내용을 보면 오픈소스 프로젝트라고 써있다.

 

혹시 파이썬 모듈도 있나 들어가봤다.

 

pyzxing Python wrapper to ZXing library

 

있다.

 

바로 설치.

 

>>> from pyzxing import BarCodeReader

>>> reader = BarCodeReader()

>>> results = reader.decode( './TransparentImage.png')

>>> results
[{'filename': 'file:///home/mandu/Desktop/p/./TransparentImage.png', 'format': 'QR_CODE', 'type': 'TEXT', 'raw': '0pXenF4L00pza6n9XO45WjUvM4RvBUuoQICQAraBTh1rJIBeYUpxxJFNFsGOQ7DFNZ67QzuP0fNzl8of6j3wfObKWkRuUO1xrn2iMLQl7KCBJ6waeBboITBQRPDYtZUmQmHHa85ll4J', 'parsed': '0pXenF4L00pza6n9XO45WjUvM4RvBUuoQICQAraBTh1rJIBeYUpxxJFNFsGOQ7DFNZ67QzuP0fNzl8of6j3wfObKWkRuUO1xrn2iMLQl7KCBJ6waeBboITBQRPDYtZUmQmHHa85ll4J', 'points': [(3.5, 49.5), (3.5, 3.5), (49.5, 3.5), (46.5, 46.5)]}]
>>> results[0]['raw']
'0pXenF4L00pza6n9XO45WjUvM4RvBUuoQICQAraBTh1rJIBeYUpxxJFNFsGOQ7DFNZ67QzuP0fNzl8of6j3wfObKWkRuUO1xrn2iMLQl7KCBJ6waeBboITBQRPDYtZUmQmHHa85ll4J'

 

성공.

 

 

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
from pwn import *
from PIL import Image
from pyzxing import BarCodeReader
 
= remote("qr-generator.ctf.defenit.kr"9000)
p.recvuntil('name?')
p.sendline('M4ndU')
for n in range(100):
    img = Image.open('./200.png')
    img = img.convert("RGB")
    newData = []
 
    qrbin = [0* 200
    p.recvuntil('< QR >\n')
    for i in range(200):
        data = str(p.recvline())
        data = data.replace("b'""")
        data = data.replace("\\n'""")
        data = data.replace(" """)
        if data == "":
            print(p.recvline())
            break
        qrbin[i] = data
 
    for x in range(len(qrbin)):
        if qrbin[x] == 0:
            break
        for y in range(len(str(qrbin[x]))):
            if qrbin[x][y] == "1":
                img.putpixel( (x, y), (000))
 
    img.save("./dimg/TransparentImage"+str(n)+".png""PNG")
 
 
    reader = BarCodeReader()
    results = reader.decode('./dimg/TransparentImage'+str(n)+'.png')
    print(results[0]['raw'])
 
    p.sendline(results[0]['raw'])
 
p.interactive()
 
cs

 

 

 

 

 

 

 


Minesweeper

Can you solve the minesweeper in one minute?

 

 

1분안에 지뢰찾기를 성공하면 되는 문제.

 

 

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
     a   b   c   d   e   f   g   h   i   j   k   l   m   n   o   p   
   -----------------------------------------------------------------
 1 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
 2 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
 3 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
 4 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
 5 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
 6 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
 7 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
 8 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
 9 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
10 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
11 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
12 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
13 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
14 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
15 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
16 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
 
Type the column followed by the row (eg. a5). To put or remove a flag, add 'f' to the cell (eg. a5f). Type 'help' to show this message again.
 
Enter the cell (40 mines left): 
 
cs

 

이런식으로 칸이 주어지고, a1을 입력하면 

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
     a   b   c   d   e   f   g   h   i   j   k   l   m   n   o   p   
   -----------------------------------------------------------------
 1 | 0 | 0 | 0 | 0 | 0 | 1 |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
 2 | 0 | 0 | 0 | 0 | 0 | 1 |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
 3 | 0 | 1 | 2 | 2 | 1 | 1 |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
 4 | 0 | 1 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
 5 | 0 | 1 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
 6 | 1 | 1 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
 7 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
 8 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
 9 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
10 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
11 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
12 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
13 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
14 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
15 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
16 |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
   -----------------------------------------------------------------
 
cs

주변 3*3범위에 폭탄의 개수를 알려준다.

 

지뢰의 위치에는 깃발을 꽂아야 한다. 저기서는 C4자리에 폭탄이 있으니 c4f를 입력해주어야 한다.

다음으로 지뢰가 없는 c5를 입력하여 나아가도록 한다.

 

이 과정을 자동화 하자.

 

 

http://www.hakank.org/numberjack/minesweeper.py

구글링 해서 찾아낸 지뢰찾기 솔버.

 

사용을 위해서 Numberjack 모듈을 설치해준다.

https://github.com/eomahony/Numberjack

 

eomahony/Numberjack

Python Combinatorial Optimisation Platform. Contribute to eomahony/Numberjack development by creating an account on GitHub.

github.com

X = -1

default_game = [
            [2,3,X,2,2,X,2,1],
            [X,X,4,X,X,4,X,2],
            [X,X,X,X,X,X,4,X],
            [X,5,X,6,X,X,X,2],
            [2,X,X,X,5,5,X,2],
            [1,3,4,X,X,X,4,X],
            [0,1,X,4,X,X,X,3],
            [0,1,2,X,2,3,X,2]
            ]

이렇게 지뢰찾기 배열을 넘겨주면

0 0 1 0 0 1 0 0 
1 1 0 0 1 0 1 0 
0 0 1 1 0 1 0 1 
1 0 1 0 1 1 0 0 
0 1 1 1 0 0 1 0 
0 0 0 0 1 1 0 1 
0 0 1 0 1 0 1 0 
0 0 0 1 0 0 1 0 
Nodes: 3  Time: 0.0

 

0: 지뢰없음 1: 지뢰있음 으로 결과를 출력해준다. 확실하지 않은 곳은 확률을 반환해준다. 이것을 개조했다.

 

문제에 아무 위치 한곳을 입력해주고 그 출력값을 위 코드로 분석하여 확실한 지뢰가 아닌 곳(0)과 지뢰인 곳(1)을 구해낼 수 있다. 지뢰인 곳에는 flag를 세우고 지뢰가 아닌 곳은 입력해서 주변 값들도 알아낸다.

 

이제 다시 맵의 결과값을 바탕으로 위 루틴을 전체를 구해낼 때까지 돌린다.

 

 

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
#!/usr/bin/python
"""
This Numberjack model was created by
Hakan Kjellerstrand (hakank@bonetmail.com)
 
See also my Numberjack page http://www.hakank.org/numberjack/
"""
import sys
from Numberjack import *
 
from pwn import *
 
colNum = "abcdefghijklmnop"
 
= -1
alreadyInput = [[0 for ccccc in range(16)] for ddddd in range(16)]
row = [['0' for cola in range(16)] for rowa in range(16)]
gamerow = [[-1 for colas in range(16)] for rowas in range(16)]
 
= remote("minesweeper.ctf.defenit.kr"3333)
p.recvuntil(':')
p.sendline('h7'#중앙을 파면 한 번에 많은 위치를 분석할 수 있지 않을까
 
default_game = gamerow
default_r = 16
default_c = 16
 
#
# Solve the Minesweeper problem
#
def minesweeper(libs, game="", r="", c=""):
 
    # Set default problem
    if game == "":
        game = default_game
        r = default_r
        c = default_c
    else:
        print "rows:", r, " cols:", c
 
    #
    # Decision variables
    # Note: Matrix is defined with cols,rows,...
    #
    mines = Matrix(c,r,0,1)
 
    S = [-1,0,1]  # for the neighbors of this cell
 
    model = Model()
 
    for i in range(r):
        for j in range(c):
            if game[i][j] >= 0:
                model.add(mines[i,j] == 0)
                # this cell is the sum of all the surrounding cells
                model.add(
                    game[i][j] == Sum([mines[i+a,j+b]
                                       for a in S for b in S
                                       if i+a>=0 and
                                          j+b>=0 and
                                          i+a<r  and
                                          j+b<c
                                       ])
                    )
            if game[i][j] > X:
                # This cell cannot be a mine
                model.add(mines[i,j] == 0)
 
    # print model
 
 
    for library in libs:
        solver = model.load(library)
        # print solver
        print ''
        if solver.solve():
            solver.printStatistics()
            print_mines(mines, r, c)
            print 'Nodes:', solver.getNodes(), ' Time:', solver.getTime()
            #while library == 'Mistral' and solver.getNextSolution():
            #    print_mines(mines, r, c)
            #    print 'Nodes:', solver.getNodes(), ' Time:', solver.getTime()
 
        else:
            print "No solution"
            p.interactive()
        print ''
 
#
# Print the mines
#
def print_mines(mines, rows, cols):
    for i in range(rows):
        print i+1,
        for j in range(cols): #이부분을 수정했다. 불확실한 곳은 ?표시
            if len(str(mines[i,j])) > 4:
                print "?",
            else : #확실
                print str(mines[i,j]),
                if str(mines[i,j]) == '1'#지뢰는 플래그를 세우자
                    if alreadyInput[i][j] == 0#중복 입력 방지
                        print(colNum[j]+str(i+1)+"f"+'\n')
                        try:
                            p.send(colNum[j]+str(i+1)+"f"+'\n')
                            p.recvuntil("):")
                        except Exception as e:
                            p.interactive()
                elif str(mines[i,j]) == '0'#지뢰가 아니면 주변을 더 알아내자
                    if alreadyInput[i][j] == 0#중복 입력 방지
                        print(colNum[j]+str(i+1)+'\n')
                        try:
                            p.send(colNum[j]+str(i+1)+'\n')
                            p.recvuntil("):")
                        except Exception as e:
                            p.interactive()
        print ''
 
def print_game(game, rows, cols):
    for i in range(rows):
        print i+1,
        for j in range(cols):
            if game[i][j] == -1:
                print "?"#모르거나 지뢰인 곳은 ?표시
            else :
                print game[i][j],
        print ''
 
 
# file = "minesweeper1.txt"
 
for aa in range(6): #루틴을 몇번 돌릴 것인지. 끝까지 돌리는게 최고지만 정확성이 떨어져서 적당히 맞춰줘야 했다.
    row = [['0' for cola in range(16)] for rowa in range(16)]
    gamerow = [[-1 for colas in range(16)] for rowas in range(16)]
 
    p.recvline()
    p.recvline()
    p.recvline()
    p.recvline()
    p.recvline()
    for i in range(16):
        data = p.recvline()
        row[i] = str(data).split('|')
        p.recvline()
        for j in range(117):
            try:
                if row[i][j] == '   ':
                    gamerow[i][j-1= -1;
                elif row[i][j] == ' F ':
                    alreadyInput[i][j-1= -1 #중복입력을 방지하기 위함. 깃발을 세웠다는건 여기를 더 이상 이 위치를 입력할 필요가 없다는 것
                else :
                    gamerow[i][j-1= int(row[i][j].strip());
                    alreadyInput[i][j-1= 1 #중복입력ㅇ르 방지하기 위함. 이곳의 값이 있다는건 더 이상 이 위치를 입력할 필요가 없다는
            except Exception as e:
                p.interactive()
 
 
    default_game = gamerow
    print_game(default_game, 1616)
    # minesweeper(['NumberjackSolver', 'Mistral'])
    minesweeper(['Mistral'])
    # minesweeper(['SCIP'])
    print "================================================="
    p.sendline("h7")
 
p.interactive()
 
cs

 

 

근데 solver가 완벽하지는 않다. 완벽한 지뢰 위치를 찾아주지 못한다. 그래서 중간에 game over가 나기도 한다.

그래서 약 6번의 루틴을 시도하게 하고, 코드를 수십번 돌려서 운좋게 165행 코드가 실행이 되면 약 3칸정도가 남는다. 이건 눈으로 보고 풀어서 flag를 받을 수 있었다.

 

 

It took you 0 minutes and 17 seconds.

You Win!!!
Defenit{min35w33p3r_i5_ezpz}

 

 

 

 

728x90
반응형

'CTF Write Up' 카테고리의 다른 글

CCE2020 Quals Write-up  (0) 2020.09.26
FIESTA 2020 Write up  (2) 2020.09.07
2020 Defenit CTF Write Up  (2) 2020.06.06
angstrom CTF 2020 write up  (0) 2020.03.14
UTCTF 2020 Write up  (0) 2020.03.07
RiceTeaCatPanda CTF 2020 Write up  (0) 2020.01.22
  1. npk1007 2020.11.17 17:31

    유용한 내용 정말 잘 배우고 가용~

728x90
반응형

Basic RSA (100pt)

 

 

#### PUBLIC KEY ####
e : 925
n : 119401
#### PRIVATE KEY ####
d : 29569
#### ENCODED MESSAGE ####
m : 26018
====================

 

비밀키랑 암호화된 메세지를 준다.

 

rsa 복호화 방법인 modular exponentiation으로 풀면 된다. 나온 값을 문자로 변환해서 보내면 된다.

 

스테이지 수가 많은데, 개수가 정해지지 않은 것 같다. 한 30스테이지까지 돌도록 해서 여러번 돌려주면 플래그를 얻을 수 있다.

 

중간에 오류나는 경우도 있었는데, 다시 돌려서 해당 경우에 안걸리면 된다.

 

 

 

 

 

풀이자가 적어서.. 많아지면 풀이 코드를 올리겠습니다.

728x90
반응형
728x90
반응형

반복적으로 같은 요청을 하는 (도배)를 막는 기능을 추가하겠습니다.




kakao auto_reply에서 제공하는 user_key를 이용하겠습니다.

user_key는 옐로우 아이디 마다 부여되는 고유 키값입니다.


처음에는 user_key값을 이용하여 반복적으로 동일한 요청을 해올 경우 딜레이를 주는 방법을 사용하려고 했지만

구현할 방법을 모르겠어서...


직전에 요청한 유저와 다음 요청해온 유저가 같을 경우 함수를 실행하지 않고 리턴하는 방법을 사용하겠습니다.

이 경우 다른 유저가 요청해오지 않는 이상 한 유저가 연속으로 요청을 할 수가 없게 되죠.


도배로 고통받는 개발자...




모든 요청(버튼)에 대해서 적용하는 방법과 각각의 요청(버튼)에 적용하는 방법이 있습니다.

모든 요청(버튼)에 적용하게 되면 한 요청을 했을 때 바로 다른 요청을 할 수가 없으므로

각각에 요청에 적용하는 방법을 사용하겠습니다.

그러기 위해서 class 를 사용하겠습니다.





코드 작성하기


적용된 전체 코드 :  https://github.com/M4ndU/inhun_kakao_chat_bot_2/blob/master/views.py




1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
class user_chk():
 
    def __init__(self):
        self.pre_key = "" #이전 user_key값
        self.now_key = "" #현재 user_key값
 
    def check(self, key):
        self.now_key = key # now_key값에 현재 user_key값 대입
 
        if self.pre_key == self.now_key : # 비교 하여 같으면 1을 반환
            passcode = 1
        else :
            self.pre_key = self.now_key # 다를 경우 pre_key값에 now_key값을 덮어쓰고 0 반환
            passcode = 0
        return passcode
cs



바로 다음 코드들을 작성하여 값 초기화 설정을 해 둡니다.

필요한 개수 만큼 작성합니다.


1
2
3
4
 
u0 = user_chk()
u1 = user_chk()
 
cs



이렇게 하면, u0.check(user_key)를 실행하게 될 경우

이전 user_key값(=pre_key)과 비교하여 같으면 1 반환, 다르면 pre_key에 저장하고 0을 반환합니다.


u0, u1각각 독립적으로 사용되기 때문에 u0.check("dd")를 했다고 해서 u1.pre_key에 dd가 드러가지 않는 것이죠.

이를 통해 각각 요청(버튼)에 적용하여 사용할 수 있습니다.



이제 user_key값을 가져와 봅시다.


1
2
3
4
...
    datacontent = received_json_data['content']
    user_key = received_json_data['user_key']
 
cs


기존 작성했던 datacontent = received_json_data['content'] 바로 밑에 작성해 줍니다.




user_key값도 불러왔으니 이제 각 분기마다 적용해줍니다.


https://github.com/M4ndU/inhun_kakao_chat_bot_2/blob/master/views.py

92~104


92
93
94
95
96
97
98
99
100
101
102
103
104
    if datacontent == '오늘 급식':
        if u0.check(user_key):
            return ret_proc(PlzStopIt)
        dt = datetime.datetime.today()
        diet = get_meal(dt)
        return ret_proc(diet)
 
    elif datacontent == '내일 급식':
        if u1.check(user_key):
            return ret_proc(PlzStopIt)
        dt = datetime.datetime.today() + datetime.timedelta(days=1)
        diet = get_meal(dt)
        return ret_proc(diet)
cs



        if u0.check(user_key):

            return ret_proc(PlzStopIt)


를 통해서 u0.pre_key = u0.now_key가 되어 1을 반환하게 되면

조건이 True가 되어 밑에 작업을 진행하지 않고 바로 문자열을 바로 반환합니다.

그럴려면 PlzStopIt에 문자열을 저장해주어야 겠죠?


728x90
반응형
  1. 에러 2018.06.04 17:39

    잘못된 응답입니다. response is not OK(200) : 500 / Internal Server Error
    원인이 뭘까요..

    • M4ndU 2018.06.04 19:22 신고

      해당 에러코드만 가지고 제가 원인을 알 수는 없습니다...

  2. 파싱 오류인가용? 2018.06.05 22:24

    자동 응답 시스템이 응답을 하지 않습니다. 라는 말을 뿜어대네요ㅠㅠ. 그냥 텍스트 출력하는 BOT정보, 중간고사 버튼들은 잘 작동하는데 급식 관련이 작동을 안해융....
    초보인데 좀 도와주실 수 있나여?

    [05/Jun/2018 22:21:23] "POST /mealapp/message HTTP/1.1" 500 71923
    오늘 급식
    Internal Server Error: /mealapp/message
    Traceback (most recent call last):
    File "C:\Users\Administrator\mealbot\myvenv\lib\site-packages\django\core\handlers\exception.py", line 41, in inner
    response = get_response(request)
    File "C:\Users\Administrator\mealbot\myvenv\lib\site-packages\django\core\handlers\base.py", line 187, in _get_response
    response = self.process_exception_by_middleware(e, request)
    File "C:\Users\Administrator\mealbot\myvenv\lib\site-packages\django\core\handlers\base.py", line 185, in _get_response
    response = wrapped_callback(request, *callback_args, **callback_kwargs)
    File "C:\Users\Administrator\mealbot\myvenv\lib\site-packages\django\views\decorators\csrf.py", line 58, in wrapped_view
    return view_func(*args, **kwargs)
    File "C:\Users\Administrator\mealbot\mealapp\views.py", line 96, in answer
    diet = get_meal(dt)
    File "C:\Users\Administrator\mealbot\mealapp\views.py", line 48, in get_meal
    l_l = get_diet(2, local_date, local_weekday)
    NameError: name 'get_diet' is not defined

    • 파싱 오류인가용? 2018.06.05 22:53

      (request) 들어간 줄이 다 문제인데 으으 리퀘스트 모듈 문제인가요?

    • M4ndU 2018.06.05 22:55 신고

      오류코드 마지막줄을 보세요.
      File "C:\Users\Administrator\mealbot\mealapp\views.py", line 48, in get_meal
      l_l = get_diet(2, local_date, local_weekday)
      NameError: name 'get_diet' is not defined

      get_deit라는 함수가 정의되지 않은 것입니다.

  3. 파싱 오류인가용? 2018.06.05 23:21

    엇...
    https://github.com/M4ndU/inhun_kakao_chat_bot_2/blob/master/views.py 만두님 소스 사용 하였으며
    def get_meal <- 정의 되어있는 거 아닌가용?

    def get_meal(dt):
    local_date = dt.strftime("%Y.%m.%d")
    local_weekday = dt.weekday()

    l_l = get_diet(2, local_date, local_weekday)

    if len(l_l) == 1:
    lunch = "급식이 없습니다."
    return lunch

    • M4ndU 2018.06.05 23:26 신고

      get_meal과 get_diet는 엄연히 다른 함수입니다. get_diet함수는 parser.py에 정의된 함수이며 제소스를 그대로 사용하셨다면 parser.py파일이 없는게 아닌지 확인해보세요

  4. 파싱 오류인가용? 2018.06.05 23:34

    자꾸 질문해서 죄송합니다 ㅠㅠㅠㅠ
    마지막으로 여쭤보겠습니다. 안되면 어떻게든 해볼게요!

    parser.py 파일을 확인해보았습니다.
    아래와 같이 만두님이 알려주신 parser.py 에 get_diet 함수가 정의되어 있는것을 확인했어요!
    그런데... parser.py 파일은 항상 있었습니다...

    def get_diet(code, ymd, weekday):
    schMmealScCode = code #int 1조식2중식3석식
    schYmd = ymd #str 요청할 날짜 yyyy.mm.dd
    if weekday == 5 or weekday == 6: #토요일,일요일 버림
    element = " " #공백 반환

  5. 파싱 오류인가용? 2018.06.05 23:39

    startapp 폴더. 즉 애플리케이션 폴더에 (views폴더) 에 있었습니다!

    • 아아 2018.06.05 23:40

      views.py 있는 폴더입니다1!

    • M4ndU 2018.06.10 17:48 신고

      python3로 실행하신거 맞으시죠?

      저도 잘 모르겠습니다.. :(

  6. 2018.09.01 13:16

    비밀댓글입니다

  7. onetop 2018.09.13 22:43

    우선 소스 정말 잘봤습니다 공유해주셔서 감사합니다

    다름이 아니라 parser.py 에 문제가 있는것으로 보입니다.
    다른 기능들은 모두 작동하는데
    해당 파일에서 급식을 파싱을 못해오는듯 합니다

    • 1111 2018.09.13 22:45

      해당프로필의 자동응답 시스템이 응답을 하지 않습니다. 라고 뜨네요

      확실히 교육청사이트에서 제대로 파싱을 못하고 있나봅니다

      로그에는
      [13/Sep/2018 22:44:15] "POST /message HTTP/1.1" 500 27

      뜹니다

    • M4ndU 2018.09.13 23:05 신고

      방금 테스트 해본 결과, parser.py는 정상작동 합니다. 급식 파싱 정상적으로 동작합니다.

    • 1111 2018.09.13 23:44

      네 저도 테스트 해본 결과 정상 동작합니다

      결국에는 github 에 올려주신
      것중 sys.path.insert(0, "inhun_discord_chat_bot_2 경로") 때문인듯 한데 설명부탁드립니다

  8. 1111 2018.09.13 23:42

    sys.path.insert(0, "inhun_discord_chat_bot_2 경로")

    git 허브에 보면 이런 부분이 있는데
    저것이 모듈 임포트 아닌가요?
    https://github.com/M4ndU/inhun_discord_chat_bot_2/blob/master/inhun_bot.py 를 저장하고 경로를 넣으라는 설명같은데
    아무리 봐도 이부분은 디스코드 관련 파일이라
    카카오 api 하고 상관없을듯 한데 아닌가요??
    이부분 때문에 제대로 파싱을 못하는것 같아요

    • M4ndU 2018.09.14 00:00 신고

      import sys
      sys.path.insert(0, "inhun_discord_chat_bot_2 경로")

      이 코드가 추가되어있는 이유는 parser.py파일이 (github기준)해당 경로에 있었기 때문입니다. (혼란을 드린 것 같네요.. 죄송합니다.)

      만약 parser.py를 views.py가 있는 경로로 옮겨 두셨다면 위의 코드는 지우셔도 됩니다.

    • 1111 2018.09.14 00:04

      아닙니다 혼란은 무슨 이해못한 제가 멍청한거죠...

      한가지 질문 더 드리겠습니다

      import requests, re
      from bs4 import BeautifulSoup

      def get_diet(code, ymd, weekday):
      schMmealScCode = code #int 1조식2중식3석식
      schYmd = ymd #str 요청할 날짜 yyyy.mm.dd
      if weekday == 5 or weekday == 6: #토요일,일요일 버림
      element = " " #공백 반환
      else:
      num = weekday + 1 #int 요청할 날짜의 요일 0월1화2수3목4금5토6일 파싱한 데이터의 배열이 일요일부터 시작되므로 1을 더해줍니다.
      URL = (
      "https://stu.sen.go.kr/sts_sci_md01_001.do?"
      "schulCode=B100000659"
      "&schulCrseScCode=4"
      "&schulKndScCode=04"
      "&schMmealScCode=%d&schYmd=%s" % (schMmealScCode, schYmd)
      )
      #http://stu.AAA.go.kr/ 관할 교육청 주소 확인해주세요.
      #schulCode= 학교고유코드
      #schulCrseScCode= 1유치원2초등학교3중학교4고등학교
      #schulKndScCode= 01유치원02초등학교03중학교04고등학교

      #기존 get_html 함수부분을 옮겨왔습니다.
      html = ""
      resp = requests.get(URL)
      if resp.status_code == 200 : #사이트가 정상적으로 응답할 경우
      html = resp.text
      soup = BeautifulSoup(html, 'html.parser')
      element_data = soup.find_all("tr")
      element_data = element_data[2].find_all('td')
      try:
      element = str(element_data[num])

      #filter
      element_filter = ['[', ']', '<td class="textC last">', '<td class="textC">', '</td>', '&amp;', '(h)', '.']
      for element_string in element_filter :
      element = element.replace(element_string, '')
      #줄 바꿈 처리
      element = element.replace('<br/>', '\n')
      #모든 공백 삭제
      element = re.sub(r"\d", "", element)

      #급식이 없을 경우
      except:
      element = " " # 공백 반환
      return element

      이게 git 에 있는 코드고

      import requests
      from bs4 import BeautifulSoup
      import re


      def get_html(url):
      _html = ""
      resp = requests.get(url)
      if resp.status_code == 200:
      _html = resp.text
      return _html


      def get_diet(code, ymd, weekday):
      schMmealScCode = code #int
      schYmd = ymd #str

      num = weekday + 1 #int 0월1화2수3목4금5토6일
      URL = (
      "http://stu.sen.go.kr/sts_sci_md01_001.do?"
      "schulCode=B100000659&schulCrseScCode=4&schulKndScCode=04"
      "&schMmealScCode=%d&schYmd=%s" % (schMmealScCode, schYmd)
      )
      html = get_html(URL)
      soup = BeautifulSoup(html, 'html.parser')
      element = soup.find_all("tr")
      element = element[2].find_all('td')
      try:
      element = element[num] #num
      element = str(element)
      element = element.replace('[', '')
      element = element.replace(']', '')
      element = element.replace('<br/>', '\n')
      element = element.replace('<td class="textC last">', '')
      element = element.replace('<td class="textC">', '')
      element = element.replace('</td>', '')
      element = element.replace('(h)', '')
      element = element.replace('.', '')
      element = re.sub(r"\d", "", element)
      except:
      element = " "
      return element

      이것이 블로그에 있는 코드입니다

      해당 코드들로

      from parser import *
      from django.shortcuts import render
      from django.views.decorators.csrf import csrf_exempt
      from django.http import JsonResponse
      from pytz import timezone
      import datetime , json, sys
      sys.path.insert(0, "/home/user/kakao/app/inhun_bot.py")

      dt = datetime.datetime.today()
      local_date = dt.strftime("%Y.%m.%d")
      local_weekday = dt.weekday()

      print (local_date,local_weekday)
      diet = get_diet(2, local_date, local_weekday)
      print(diet)

      이명령어로 실행해보면 블로그 코드는 실행이 되지만 git 코드는 None 라는 값을 출력합니다

    • M4ndU 2018.09.14 00:13 신고

      제가 테스트해본 결과, 둘 다 정상작동하여 같은 결과를 보여줍니다.

      두개의 코드에서 기능적 차이는 없습니다.
      밖으로 나갔던 get_html함수를 안으로 넣은 것과 필터링 부분을 반복문을 사용해 코드 길이를 줄인 것 뿐입니다.

    • M4ndU 2018.09.14 00:15 신고

      아 from parser import * 부분이 코드 최상단에 위치하고 있네요.

      sys.path.insert(0, "/home/user/kakao/app/inhun_bot.py")
      이 코드 하단으로 내려주세요.

    • 1111 2018.09.14 00:27

      흠 저는 아무리 해봐도
      이상하게 git 에 올려주신 코드는 Null 만 출력되네요 무튼 이건 제가 내일 다시 해봐야겠네요....

      그리고 혹시 views 파일에서

      def get_meal(dt):
      local_date = dt.strftime("%Y.%m.%d")
      local_weekday = dt.weekday()

      l_l = get_diet(2, local_date, local_weekday)

      if len(l_l) == 1:
      lunch = "급식이 없습니다."
      return lunch

      else:
      d_d = get_diet(3, local_date, local_weekday)
      if len(d_d) == 1:
      lunch = local_date + " 중식\n" + l_l
      dinner = ""
      return lunch
      else:
      lunch = local_date + " 중식\n" + l_l + "\n"
      dinner = local_date + " 석식\n" + d_d

      return lunch + dinner

      테스트 하려면 어떻게 해야할까요??
      블로그 소스로 만든 parser 정상 받아오는건 확인했고 views 에 정상 동작하는 parser 을 넣었는데 카톡에서는 자동응답 시스템이 응답을 하지 않는다 [14/Sep/2018 00:26:33] "POST /message HTTP/1.1" 500 27
      라고 로그가 뜨네요 ....

    • 1111 2018.09.14 00:27

      Null 이 아니라 None 입니다.

    • M4ndU 2018.09.14 00:31 신고

      코드 중간중간에 print("1")같은 코드를 삽입해주어서 실행했을 때 터미널에 출력되는 개수들을 보고 어디까지 실행되어서 어디에서 막혔는지를 파악하는 방법이 있습니다

  9. 1111 2018.09.14 00:35

    그렇군요 내일 테스트 해보면서 오류 찾아야 겠네요 ..
    늦은시간까지 감사합니다.

    그리고 이건 Nods.js 짠 api 라는데 보시면 req 말고 res 하고 next 를 주던데
    파이썬도 저런옵션 안줘도 되나요? 가끔 req만 주니 메뉴선택창이 사라지는 버그가 있는것 같아서요
    /* 처음 들어왔을 때 */
    router.get('/', function(req, res, next) {
    const menu = {
    type: 'buttons',
    buttons: ["노래 추천해줘.", "네 노래로 추천해줘.","오늘 급식좀."]
    };
    res.set({
    'content-type': 'application/json'
    }).send(JSON.stringify(menu));
    });


    출처: http://wingnim.tistory.com/6 [jinyo의 뇌]

    • M4ndU 2018.09.14 07:34 신고

      파이썬도 return 값에 keyboard값을 포함하여 보내기 때문에 메뉴가 사라지지 않습니다.
      아마 중간에 오류가 나서 return 이 이루어지지 않아서 메뉴가 사라진것 같습니다.

    • 1111 2018.09.14 23:06

      그렇군요 감사합니다
      카카오 측에 물어보니

      안녕하세요.
      확인해보니 Content-Type 이 application/json 으로 내려오고 있습니다.
      https://github.com/plusfriend/auto_reply#specification 를 참고하셔서
      application/json; charset=utf-8 로 변경해주시기 바랍니다.
      수정 후에도 문제가 계속되면 코멘트 달아주세요.
      감사합니다.

      라고 하더군요 .

    • 1111 2018.09.16 12:17

      아무리 구글링을 해봐도
      application/json; charset=utf-8 바꾸는 법을 모르겠네요 .... ㅠㅠ
      그나마 비슷한사례가
      https://github.com/plusfriend/auto_reply/issues/381
      같은데 혹시 어떻게 수정해야 할까요??

    • M4ndU 2018.09.16 13:46 신고

      저도 잘 모르겠네요...

    • 1111 2018.09.16 18:53

      몇가지 연구를 해봤는데

      diet 에 데이터가 담길때

      흑미밥
      차돌박이고추장찌개
      도라지오이생채
      모듬스크램블에그
      갈치카레구이
      배추김치

      형태로 담기는데
      그러면
      diet = "흑미밥 \n 차돌박이고추장찌개 \n"
      담겨야 하는데 이렇게 안담기나 보네요
      print(diet) 해서 터미널에 python views.py 실행해도 출력이 안뜨니 확인도 못하고 답답하네요

      그리고 소스 보니 "텍스트" , '텍스트' 이렇게 2가지 방식을 섞어 쓰던데
      '' 로만 써야 되는거 아닌가요?파이썬은

    • M4ndU 2018.09.16 18:56 신고

      views.py엔 함수정의밖에 없으니 아무것도 출력되지 않는게 당연하죠..

      파이썬은 "string"과 'string' 모두 사용 가능합니다.

728x90
반응형

이번에는 이번에는 맴버 관련 기능을 구현해보도록 하겠습니다.


목차입니다.



1. 맴버가 서버 접속, 퇴장시 서버 텍스트 채널에 메세지를 보내는 기능

2. 맴버가 서버 접속시 1:1 채팅에 메세지를 보내는 기능

3. 주기적으로 공지메세지를 보내는 기능 (백그라운드로 동작)










1. 맴버가 서버 접속, 퇴장시 서버 텍스트 채널에 메세지를 보내는 기능



먼저 코드를 봅시다.


1
2
3
4
5
6
7
8
9
10
11
@client.event
async def on_member_join(member):
    fmt = '{1.name} 에 오신것을 환영합니다., {0.mention} 님'
    channel = member.server.get_channel("channel_id_here")
    await client.send_message(channel, fmt.format(member, member.server))
 
@client.event
async def on_member_remove(member):
    channel = member.server.get_channel("channel_id_here")
    fmt = '{0.mention} 님이 서버에서 나가셨습니다.'
    await client.send_message(channel, fmt.format(member, member.server))
cs



on_member_join 함수는 맴버가 서버에 들어 왔을 때 실행되는 함수이고,

on_member_remove 함수는 맴버가 서버를 나갔을 때 실행되는 함수입니다.


format를 통해서 {0.mention}  에는 맴버가 언급되고,  {1.name} 에는 서버이름이 자동적으로 대입됩니다.


channel_id_here에 들어갈 id를 찾는 방법은 본문 하단에 설명되어 있습니다.









2. 맴버가 서버 접속시 1:1 채팅에 메세지를 보내는 기능



간단합니다. 아래의 코드를 이용하면 끝입니다.


await client.send_message(member, "내용")


여기서 우리는 맴버가 서버에 들어왔을 때 보내도록 할 것이므로, on_member_join 함수 밑에 두겠습니다.



1
2
3
4
5
6
@client.event
async def on_member_join(member):
    fmt = '{1.name} 에 오신걸 환영합니다, {0.mention} 님'
    channel = member.server.get_channel("channel_id_here")
    await client.send_message(channel, fmt.format(member, member.server))
    await client.send_message(member, "내용")
csv



"내용" 에 원하는 내용을 작성하시면 됩니다.









3. 주기적으로 공지사항등의 메세지를 보내는 기능입니다.



백그라운드로 동작합니다.


코드입니다. :


1
2
3
4
5
6
    async def my_background_task():
        await client.wait_until_ready()
        channel = discord.Object(id='channel_id_here')
        while not client.is_closed:
            await client.send_message(channel, "hi")
            await asyncio.sleep(5
cs


@client.event 가 필요 없습니다.



----------------channel_id_here 구하기---------------


channel_id_here 에 메세지를 보낼 텍스트 채널의 id를 입력해주시면 되는데요.

id를 확인하는 방법은 다음과 같습니다:

1. 디스코드 채널을 웹을 통해 접속한다.

2. 텍스트 채널에 들어간다.

3. URL마지막에 있는 18자리 숫자가 id다.




"hi" 대신에 보낼 메세지를 넣어주시면 됩니다.


await asyncio.sleep(5)

에서 5 대신에 자신이 원하는 시간 간격을 초 단위로 작성하여 주시면 됩니다.


EX) await asyncio.sleep(60*60*24) 이라고 하면 1일마다 메세지를 보내게 됩니다.




이 부분을 다 작성하셨으면 전체 코드에서 하단으로 가셔서 client.run() 부분을 찾습니다.

client.run() 코드 바로 위에 아래의 코드를 추가합니다 :


client.loop.create_task(my_background_task())


그러면


client.loop.create_task(my_background_task())

cleint.run('token')


처럼 됩니다.


만약 discord.errors.Forbidden: FORBIDDEN (status code: 403): Missing Access 이런 오류가 발생한다면

권한 문제이니 서버에서 봇의 권한을 설정해 주시면 됩니다.







전체 코드는 다음 링크를 참조해주세요.


https://github.com/M4ndU/inhun_discord_chat_bot_2/blob/master/inhun_bot.py

728x90
반응형
  1. boringpinn 2020.03.05 18:21

    혹시 유저들에게 전체dm을 보내는법도 알고 계신가요?? 응용해보려 하는데 힘드네요...

  2. 국대밥 2020.04.06 07:54

    요즘은 파이썬으로 디스코드 봇만드는 법 강의 안하시나요...?

    • M4ndU 2020.04.06 14:58 신고

      네.. 요즘엔 디스코드 봇을 안만들고 있다보니 그렇네요..

  3. Hoing 2020.08.24 19:34

    요일이나 시간대마다 알림을 주는 명령어는 어떻게 만들수있나요..?ㅠㅠ

    • M4ndU 2020.08.26 15:22 신고

      APScheduler 같은 모듈 이용하시면 될 것 같습니다.

728x90
반응형

이번에는 저번에 만들었던 카카오톡 챗봇에 급식 파서를 활용하여

급식을 알려주는 챗봇을 만들도록 하겠습니다.



이전과정

[

카카오톡 챗봇 만들기 --> http://mandu-mandu.tistory.com/67

급식 파서 만들기 --> http://mandu-mandu.tistory.com/21

]





코드 작성하기


급식을 파싱해서 식단을 출력하는 부분을 구현할 것인데요.

두개의 코드상 차이는 날짜값과 요일값을 가져오는 부분밖에 차이가 없기 때문에
"오늘 급식"에 대한 코드를 작성하겠습니다.

그전에 parser.py를 view.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
from django.shortcuts import render
from django.views.decorators.csrf import csrf_exempt
from django.http import JsonResponse
from pytz import timezone
import datetime , json #datetime 모듈 import
from parser import * #parser.py import
 
 
def keyboard(request):
 
    return JsonResponse({
        'type':'buttons',
        'buttons':['오늘','내일']
    })
 
@csrf_exempt
def answer(request):
 
    json_str = ((request.body).decode('utf-8'))
    received_json_data = json.loads(json_str)
    datacontent = received_json_data['content']
 
    #date
    #오늘
    dt1 = datetime.datetime.today()
 
    local_date1 = dt1.strftime("%Y.%m.%d")
    local_weekday1 = dt1.weekday()
    #오늘
 
    #내일
    dt2 = datetime.datetime.today() + datetime.timedelta(days=1)
 
    local_date2 = dt2.strftime("%Y.%m.%d")
    local_weekday2 = dt2.weekday()
    #내일
    #date
 
    if datacontent == '오늘':
 
        #시간 관련
        meal_date = str(local_date1)
        l_wkday = int(local_weekday1)
        #시간 관련
 
        #파싱
        l_l = get_diet(2, meal_date, l_wkday)
        d_d = get_diet(3, meal_date, l_wkday)
        #파싱
 
        #디스코드 챗봇 만들기(2) 참고바람.
        if len(l_l) == 1:
            lunch = "급식이 없습니다."
            dinner = ""
        elif len(d_d) == 1:
            lunch = meal_date + " 중식\n" + l_l
            dinner = ""
        else:
            lunch = meal_date + " 중식\n" + l_l
            dinner = meal_date + " 석식\n" + d_d
        #디스코드 챗봇 만들기(2) 참고바람.
        
        return JsonResponse({
                'message': {
                    'text': lunch + dinner
                },
                'keyboard': {
                    'type':'buttons',
                    'buttons':['오늘','내일']
                }
 
            })
 
    elif datacontent == '내일':
...
cs


*'내일'에 대한 부분은 '오늘'의 코드와 동일합니다. '#시간관련' 부분의 변수명만 바꾸어서 사용하시면 됩니다.

*#파싱 부분은 중식과 석식만 파싱해옵니다. 조식을 파싱할땐 함수의 첫번째 인자로 1을 주시면 됩니다.

*#디스코드 챗봇 만들기(2) 참고바람. 부분은 http://mandu-mandu.tistory.com/65 여기에 설명이 나와 있습니다.

*자신의 상황에 맞게 수정 또는 삭제 하시면 됩니다.


*'오늘'과 '내일'의 코드는 거의 차이가 없기 때문에 함수 하나로 잘 만들어서 사용하시는게 더 좋을 것 같습니다.

--18.03.03 추가

그래서 함수 하나로 만들어서 최적화를 진행하였습니다.

수정된 코드는 제 깃허브에 있습니다.

https://github.com/M4ndU/inhun_kakao_chat_bot_2

----------


저장하시면 만약 django가 돌아가는 중이었다면 자동으로 서버를 재시작해서 바로 수정된 것을 적용합니다.

모바일 카카오톡으로 접속해서 봇에서 '오늘' 또는 '내일' 버튼을 누르면 해당 버튼에 대한 식단이 출력됩니다.





이상입니다.

728x90
반응형
  1. chpi 2018.06.05 08:59

    json decode에러가 나옵니다.
    json_str = ((request.body).decode('utf-8'))
    received_json_data = json.loads(json_str)
    datacontent = received_json_data['content']
    에서 빈 문자열이 들어오는것같은데 뭐가 문제일까요 아래에는 제 깃허브입니다 확인가능하시면 확인해주실 수 있나요?

    https://github.com/ChoiHyeongu/Meal-bug

  2. choi 2018.06.05 23:24

    오늘 버튼은 급식을 출력하게 했고, 내일 버튼은 내일급식입니다!를 출력하게 했습니다. 내일버튼은 정상작동하는데 오늘버튼은 500에러가 나오네요 뭐가문제일까요?

    https://github.com/ChoiHyeongu/Meal-bug/blob/master/inform/views.py

    • M4ndU 2018.06.05 23:51 신고

      python에서 어떤 오류가 출력되는지를 확인해보면 될거 같슴니다.

  3. 2018.08.07 12:09

    비밀댓글입니다

    • M4ndU 2018.08.07 12:55 신고

      http://mandu-mandu.tistory.com/21
      이글 따라하셔서 만든 parser.py을 view.py와 같은 경로에 두고, view.py에서 parser.py을 import하셨나요?

    • chaeeee 2018.08.07 13:21

      네!! Chat 이라는 폴더 안에 app, manage.py, myproject 가 있고 app 안에 views.py 가 있어서 parser.py 도 app 폴더 안에 넣었어요.

      from django.shortcuts import render
      from django.views.decorators.csrf import csrf_exempt
      from django.http import JsonResponse
      import datetime , json
      from parser import *


      def keyboard(request):

      return JsonResponse({
      'type':'buttons',
      'buttons' : ['오늘 급식은 뭐야?','내일 급식은 뭐야?']
      })

      @csrf_exempt
      def answer(request):

      json_str = ((request.body).decode('utf-8'))
      received_json_data = json.loads(json_str)
      datacontent = received_json_data['content']

      #date
      #오늘
      dt1 = datetime.datetime.today()

      local_date1 = dt1.strftime("%Y.%m.%d")
      lacal_weekday1 = dt1.weekday()
      #오늘

      #내일
      dt2 = datetime.datetime.today() + datetime.timedelta(days=1)

      local_date2 = dt2.strftime("%Y.%m.%d")
      local_weekday2 = dt2.weekday()
      #내일
      #date

      if datacontent == '오늘 급식은 뭐야?':

      #시간 관련
      meal_date = str(local_date1)
      l_wkday = int(local_weekday1)
      #시간 관련

      #파싱
      l_l = get_diet(2, meal_date, l_wkday)
      d_d = get_diet(3, meal_date, l_wkday)
      #파싱

      if len(l_l) == 1:
      lunch = "급식이 없네..."
      dinner = ""
      elif len(d_d) == 1:
      lunch = meal_date + " -중식-\n" + l_l
      dinner = "석식은 없어~"
      else:
      lunch = meal_date + " -중식\n" + l_l
      dinner = meal_date + " -석식\n" + d_d

      today = "오늘 급식은! 두구두구X99..."

      return JsonResponse({
      'message': {
      'text': today + lunch + dinner
      },
      'keyboard':{
      'type':'buttons',
      'buttons':['오늘 급식은 뭐야?','내일 급식은 뭐야?']
      }
      })

      elif datacontent == '내일 급식은 뭐야?':

      #시간 관련
      meal_date2 = str(local_date2)
      l_wkday2 = int(local_weekday2)
      #시간 관련

      #파싱
      l_l2 = get_diet(2, meal_date2, l_wkday2)
      d_d2 = get_diet(3, meal_date2, l_wkday2)
      #파싱

      if len(l_l2) == 1:
      lunch = "급식이 없네..."
      dinner = ""
      elif len(d_d2) == 1:
      lunch = meal_date2 + " -중식-\n" + l_l2
      dinner = "석식은 없어 ~"
      else:
      lunch = meal_date2 + " -중식-\n" + l_l2
      dinner = meal_date2 + "-석식-\n" + d_d2


      tomorrow = "뭐?! 벌써 내일메뉴가 궁금하다고? 돼지같으니라구"

      return JsonResponse({
      'message':{
      'text': tomorrow + lunch + dinner
      },
      'keyboard': {
      'type':'buttons',
      'buttons':['오늘 급식은 뭐야?','내일 급식은 뭐야?']
      }
      })


      이렇게 자잘한 문구 외에 변경없이 따라했는데도 자꾸 서버오류가 떠요. 코드 자체는 실행해보면 오류가 없이 잘 실행되어요

      parser.py 도

      import requests
      from bs4 import BeautifulSoup
      import re


      def get_html(url):
      _html = ""
      resp = requests.get(url)
      if resp.status_code == 200:
      _html = resp.text
      return _html


      def get_diet(code, ymd, weekday):
      schMmealScCode = code
      schYmd = ymd

      num = weekday + 1

      URL = (
      "http://stu.gne.go.kr/sts_sci_md01_001.do?"
      "schulCode=S100000665&schulCrseScCode=4&schulKndScCode=04"
      "&schMmealScCode=%d&schYmd=%s" % (schMmealScCode, schYmd)
      )

      html = get_html(URL)
      soup = BeautifulSoup(html, 'html.parser')
      element = soup.find_all("tr")
      element = element[2].find_all("td")
      try:
      element = element[num]
      element = str(element)
      element = element.replace('[', ' ')
      element = element.replace(']', ' ')
      element = element.replace('<br/>', '\n')
      element = element.replace('<td class="textC">', ' ')
      element = element.replace('<td class="textC last">', ' ')
      element = element.replace('</td>', ' ')
      element = element.replace('(h)', ' ')
      element = element.replace('.',' ')
      element = re.sub(r"\d","",element)
      except:
      element = " "
      return element


      이렇게 블로그 작성하신거랑 url 빼고는 같게했는데..


  4. chaeeee 2018.08.07 13:23

    그리고 저 view.py 에서 #파싱 이부분만 제거하면 그냥 오늘급식은뭐야 ? -> 오늘급식은! 두구두구 이렇게 잘 출력돼요

    • M4ndU 2018.08.07 13:49 신고

      from parser import *

      date = "2018.08.20"
      day = 3

      diet = get_diet(2, date, day)
      print(diet)


      이렇게 파일하나 만드셔서 잘 출력되는지 실행해 보세요

  5. chaeeee 2018.08.07 14:04

    네 거기까지는 잘 돼요!! 흠...

    • chaeee 2018.08.07 14:07

      그리고 서버는 공유기 포트포워딩해서 쓰고있어요 mac os X 로 하고있고욥

    • M4ndU 2018.08.07 14:09 신고

      다른건 되는걸 봐선 서버문제는 아닌 듯 해요

    • chaee 2018.08.07 14:23

      음 .. 그럼 대체 왜 다른데서는 멀정하게 실행이 잘되는게 저안에만 들어가면 서버에러를 일으킬까요..코드 자체에는 문제가 없죠 ..? 정보 불러오는 시간이 너무 길어서 오류가나는건가 ...? ㅠㅠㅜ

  6. 2019.12.12 21:06

    비밀댓글입니다

728x90
반응형

이 글만을 통해서 간단한 텍스트를 출력하는 카카오톡 챗봇을 만들 수 있습니다.


카카오톡 플러스 친구를 이용합니다.




카카오톡 플러스 친구 생성하기



카카오톡 관리자 센터 페이지로 이동합니다.






플러스친구 만들기를 클릭합니다.


카카오톡 계정으로 로그인을 합니다.




모두 작성하고 확인을 클릭합니다.







Django 설치 및 설정하기


https://tutorial.djangogirls.org/ko/installation/


이곳에 설치 방법이 잘 나와있습니다.

저는 가상환경을 사용하지 않았습니다.




<linux>  django 설치하기


~$ pip3 install --upgrade pip


~$ pip3 install django






프로젝트 생성


이제 프로젝트를 생성하겠습니다.


이 내용도

https://tutorial.djangogirls.org/ko/django_start_project/

여기에 잘 나와있습니다.



생성한뒤 폴더나 파일의 이름을 바꾸어선 안됩니다.


$ django-admin startproject [프로젝트 이름] .

마침표 (.) 은 필수입니다.


$ python3 manage.py startapp [어플리케이션 이름]

어플리케이션도 같이 만들겠습니다.


example :

~/project$ mkdir tistory

~/project$ cd tistory/

~/project/tistory$ django-admin startproject mybot .

~/project/tistory$ ls

manage.py  mybot

~/project/tistory$ python3 manage.py startapp testest



$ python3 manage.py migrate

이것도 해줍니다.



설정 변경


다음과 같이 변경합니다.


mybot/settings.py

25: # SECURITY WARNING: don't run with debug turned on in production!

26: DEBUG = False

27: 

28: ALLOWED_HOSTS = ['*']


* DEBUG는 테스트를 위해선 True, 실제 사용시에는 False로 해두세요. 그냥 False로 해두시는 것을 추천

* ALLOWED_HOSTS=['*']는 모든 아이피를 허용한다는 의미입니다.


mybot/settings.py

31: # Application definition

32: 

33: INSTALLED_APPS = [

34:     'django.contrib.admin',

35:     'django.contrib.auth',

36:     'django.contrib.contenttypes',

37:     'django.contrib.sessions',

38:     'django.contrib.messages',

39:     'django.contrib.staticfiles',

40:     'testtest',

41: ]


40:  에 어플리케이션 이름을 추가합니다.



mybot/settings.py

108: TIME_ZONE = 'Asia/Seoul'



mybot/urls.py

16: from django.conf.urls import include, url

17: from django.contrib import admin

18: 

19: urlpatterns = [

20:     url(r'^testtest/', include('testtest.urls')),

21:     url(r'^admin/', admin.site.urls),

22: ]


16: include 를 추가합니다.

20: 를 추가합니다.    testtest/로 온 요청은 testtest/urls.py로 보내겠다는 의미



testtest/urls.py

1
2
3
4
5
6
7
8
9
from django.conf.urls import url
 
from . import views
 
urlpatterns = [
    url(r'^keyboard/', views.keyboard),
    url(r'^message$', views.answer),
]
 
cs


* mybot/ 경로가 아닌 testtest/ 경로입니다.

* 위 소스그대로 복사해서 파일을 만듭니다.





코드 작성하기



개발환경 : UBUNTU 16.04.3

개발언어: PYTHON 3.5.2

텍스트/코드 에디터: ATOM



testtest/views.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
from django.shortcuts import render
from django.views.decorators.csrf import csrf_exempt
from django.http import JsonResponse
import json
 
 
 
def keyboard(request):
 
    return JsonResponse({
        'type':'buttons',
        'buttons':['오늘','내일']
    })
 
@csrf_exempt
def answer(request):
 
    json_str = ((request.body).decode('utf-8'))
    received_json_data = json.loads(json_str)
    datacontent = received_json_data['content']
 
    if datacontent == '오늘':
        today = "오늘 급식"
 
        return JsonResponse({
                'message': {
                    'text': today
                },
                'keyboard': {
                    'type':'buttons',
                    'buttons':['오늘','내일']
                }
 
            })
 
    elif datacontent == '내일':
        tomorrow = "내일 급식"
 
        return JsonResponse({
                'message': {
                    'text': tomorrow
                },
                'keyboard': {
                    'type':'buttons',
                    'buttons':['오늘','내일']
                }
 
            })
 
cs


8에서 13라인은 카카오톡에 버튼 정보를 보내는 부분.

15부터 48라인은 각 버튼에 대한 응답 내용.





카카오톡 플러스 친구 설정 및 실행, 테스트하기





관리자 센터에서 스마트채팅으로 들어갑니다.




api형 설정하기를 클릭합니다.





앱 이름, 앱 URL, 앱 설명을 작성합니다.


알림받을 전화번호는 카카오톡 가입에 사용한 전화번호를 사용하세요.



앱 URL은 자신의 서버URL을 사용합니다.


$ python3 manage.py runserver port


여기에 사용한 포트 그대로 사용


http://자신의아이피:포트/디렉토리


그리고 api테스트를 클릭하여 정상적으로 작동하는지 확인합니다.



그리고 API형 저정하기를 누릅니다.


시작하기를 눌러서 서비스를 시작합니다.




이제 모바일 카카오톡에 들어갑니다. (모바일 기기에서 진행하세요.)


플러스친구 채팅으로 이동해서


버튼이 잘 뜨는지 확인하고 버튼을 눌렀을때 정상적인 응답이 오는지 확인합니다.




다음은 급식 파서를 활용한 카카오톡 챗봇을 만들겠습니다.

728x90
반응형
  1. 나도만들고싶다 2018.04.16 20:05

    안녕하세요 저도 카카오봇을 만드려고 하는데요
    이런식으로는 화면에 띄워지는데
    from django.http import HttpResponse

    def keyboard(request):
    return HttpResponse("Hello, world. You're at the polls index.")


    from django.http import JsonResponse

    def keyboard(request):
    return JsonResponse({
    'type': 'buttons',
    'buttons': ['오늘', '내일']
    })
    은 플러스친구에서
    Required*
    keyboard Fail 415
    wrong keyboard initialization
    이런 에러를 내나요? ㅠㅠ 아직 초보자라서 친절히 설명해주셨으면 좋겠어요 ㅠㅠ

  2. 나도만들고싶다 2018.04.17 20:13

    제가 파이참과 django2.0.4 버젼을 사용합니다.
    <setting.py>
    INSTALLED_APPS = [
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
    'globalHaksik',
    ]
    <urls.py>
    from django.contrib import admin
    from django.urls import path, include
    from globalHaksik import views

    urlpatterns = [
    path('admin/', admin.site.urls),
    path('globalHaksik/', include('globalHaksik.urls')),
    ]

    <globalHaksik\urls.py>
    from django.urls import path
    from . import views

    urlpatterns = [
    path('keyboard/', views.keyboard),

    ]
    이렇게 했습니다.

    • M4ndU 2018.04.17 21:54 신고

      views.py에서 코드 들여쓰기 사용하신 것 맞나요?

      일단 urls.py에서 path값 설정할 때 정규표현식 사용해주세요.

      이 외에 코드상에서 문제가 딱히 안보이네요..
      서버 아이피랑 포트 잘 확인하셨는지..
      본인의 네트워크구조에 따라 포트포워딩이 필요할 수 있습니다.

  3. 나도만들고싶다 2018.04.18 15:33

    그 들여쓰기해서 댓글에 적었는데 올라갈때 그냥 붙여서 올라가요.
    그리고 장고 2.0으로 업데이트되면서 정규식 안쓰고 path로 저렇게 쓰는 방식으로 바뀐거 같아요.
    포스포워딩이 무엇인가요??
    그 홈페이지로들어가서 xxxx.xxxxx/globalHalsik/keyboard/ 로 들어가면 막 json 파일 다운로드하라고 뜨는거와 관련이 있나요??무언가 패키지를 받아야하는건가요???

  4. eubee 2018.05.14 23:01

    파이썬 2 버전으로 만들었는데 똑같이 testtest/views.py를 써두 되나요? 아니면 다르게 써야할 명령어가 따로 있나요??
    elif에서 밑줄로 오류가 난다고 뜨는데 어떻게 해결해야 할까요..?

    • eubee 2018.05.14 23:05

      [pylint] E0001:invalid syntax (<string>, line 32) 이런 오류입니다.

    • M4ndU 2018.05.14 23:10 신고

      코드를 보여주세요

  5. 이기백 2018.05.22 14:32

    M4ndU님
    저도 질문 하나 여쭤보고 싶습니다. 일단 게시글 너무 잘보았구요! 정말 도움 많이 되었습니다.
    궁금한 점은 자신의 서버 ip는 한마디로 공인 ip가 필요한 것이지요? AWS나 pythonanywhere과 같이 무언가 서버를 실질적으로 갖고 있어야만 쓸 수 있는 것이죠? 위 게시글 그대로 따라한다고 되는 건 아니지요?
    제 생각에는 많은 분들이 실패하시는 부분이 서버 ip에서 실패하실 것 같은데, 제 생각이 맞을지 모르겠네요. 아직 초보라.....
    M4ndU님은 어떤 식으로 서버 ip를 사용하신 건지 궁금하네요! 추상적으로라도 좋으니... 조금 방향을 제시해주시면 감사하겠습니다!

    • M4ndU 2018.05.22 15:10 신고

      네 당연히 공인 ip로 설정하셔야 작동합니다.
      저는 저희집 공인 ip를 사용했습니다.

  6. nuttingon@naver.com 2018.07.03 16:06

    특정 단어를 입력 했을 때 구글 스프레드 시트의 특정 문자열을 띄우게 할 수 있을까요???

    • M4ndU 2018.07.03 16:51 신고

      특정 문자열을 받았을때 구글 스프레드 시트를 파싱해서 그 값을 보내주면 될 것 같습니다.

  7. hi django 2018.07.21 17:11

    그 어플리케이션로 지정한 경로에 url.py이 없는데 왜 이러는거죠...
    프로젝트에 지정된 경로에만 있네여..

  8. 캔따개 2018.07.30 10:31

    게시물 너무 잘 봤습니다 상세하게 알려주셔서 너무 감사합니다.

    제가 지금 aws ec2환경에서 저 게시물을 따라햇는데 키보드 오늘 내일은 나오고 버튼을 눌렀을때 응답하는 함수가 작동하지 않은것같습니다 혹시 어떻게 할지 알려주실수 있으신가요?

    • M4ndU 2018.07.30 11:01 신고

      일단 봇이 텍스트를 받아서 답을 보내는 모든 과정에 print("1") 이런식의 코드를 끼워넣어서 하나하나 어디서 막혔는지 찾아보아야 할 것 같습니다.

  9. 캔따개 2018.07.30 15:32

    print("1") 을 넣고 테스트를 하려는데

    print를 넣고 실행하면 서버가 실행이 안되네요 죄송하지만 어디에다가 넣으면서 테스트를 해야될까요?

    • M4ndU 2018.07.30 15:41 신고

      print명령 하나 넣었다고 서버 실행이 안될리가요...

  10. 다둥빠 2018.08.27 13:18

    안녕하세요, 제가 파이썬을 활용하여 만들려고 하는데,

    URL 에서 막혀서 그러는데, URL을 어떻게 만드는지 혹시 알 수 있을까요??

    • M4ndU 2018.08.27 13:20 신고

      네트워크 관련 지식이 조금 있으셔야 합니다.
      일반적으로 사용중인 외부아이피를 입력하시면 되는데, 공유기등을 이용하시는 경우 포트포워딩등의 작업을 하셔야 합니다.

  11. test 2018.09.04 00:26

    좋은 강좌 잘 보았습니다.

    저는 구글에서 제공하는 가상 리눅스 환경에서 작업중인데 프로젝트와 애플리케이션 설정 후

    '설정 변경하기'에서 막힌 상황입니다.

    리눅스 환경에서는 별도의 IDE를 사용하지 않고 콘솔에서 직접 수정해야 하는지,

    코드 수정은 어떻게 하는건지 (Vi 명령어 사용하면 되나요..!? 초보라 죄송합니다.) 알려주시면 감사하겠습니다!

    • M4ndU 2018.09.04 00:29 신고

      gui형태의 텍스트 에디터를 사용하셔도 되고(윈도우의 메모장처럼..), vi를 사용하셔도 됩니다.

  12. onetop 2018.09.10 13:34

    django 2.0 부터 urls 파일의 구문이 바꼈네요

    프로젝트
    from django.contrib import admin
    from django.urls import path, include


    urlpatterns = [
    path('admin/', admin.site.urls),
    path('smcbot/',include('smcbot.urls')),
    ]

    from django.urls import path, include
    from . import views

    app_name = 'smcbot'

    urlpatterns = [
    path('keyboard/', views.keyboard),
    path('message',views.answer),
    ]

    안에 이름은 알아서 수정하세요

  13. no2. 2018.12.29 00:13

    플러스친구 스마트채팅에서 API가 사라졌네요.
    https://center-pf.kakao.com/
    새로 만들어진 챗봇쪽으로 해야 할까요?

  14. 장재웅 2021.09.06 11:08

    정말 좋은 내용 감사드립니다. 그런데 올리신 것을 똑같게 코딩을 마친 후 runserver을 입력하니 404 not found가 뜨는걸 봐서 제 공인 ip를 어딘가 넣어야 될 거 같은데 어디에 넣어야 하는지 알 수 있을까요??

728x90
반응형

이제 디스코드 봇에 부가적인 요소들을 추가해 보겠습니다.




예제 코드 참고하기



discord.py를 설치할때 들어갔던 링크로 들어가보시면,




examples 폴더에 여러 예제들이 있습니다.

거기에 있는 소스들을 가져다가 상황에 맞게 수정하셔서 사용하시면 됩니다.




~~~플레이 중 추가하기



타 봇들을 보면 프로필에 ~~~플레이 중 을 띄워서 도움말을 보는 명령어등을 알려주는 역할을 하죠.



이 것도 코드를 추가해서 할 수 있습니다.


@client.event

async def on_ready():

    print('Logged in as')

    print(client.user.name)

    print(client.user.id)

    print('---------')

    await client.change_presence(game=discord.Game(name="!a for help"))


이렇게요. (기존 코드에 진한 부분만 추가해 주면 됩니다.)








봇 꾸미기 - 메세지 예쁘게 꾸미기 (embeds)


http://discordpy.readthedocs.io/en/latest/api.html 이 문서에 자세히 기술되어 있습니다.



봇이 메세지를 보낼 때

사용자가 보낼 때 처럼 단순한 텍스트형식이 아닌

예쁜 텍스트 상자속에 텍스트가 담겨서 오는 경우가 있는데요.


예시로 ayana의 채팅스샷




이 것을 추가해 보죠.


한 가지만 해볼건데요.

필요한 것은 제목, 내용, 색상이 되겠습니다.


아래 코드처럼 기존 코드를 수정해주시면 되겠습니다.


            embed = discord.Embed(title="No Meal", description="급식이 없습니다.", color=0x00ff00)

            await client.send_message(message.channel, embed=embed)


결과 :



728x90
반응형
  1. 천소율 2018.05.07 19:10 신고

    혹시 따로 디스코드 하시나요? 꼭 여쭙고 싶은게 있습니다.

  2. 2018.08.15 21:32

    비밀댓글입니다

    • M4ndU 2018.08.15 21:36 신고

      메세지를 보낸 뒤에 '봇을 실행한 터미널창' 에 오류메세지같은건 없었나요?

    • 다현 2018.08.17 15:01

      네 아무런 응답이 없었습니다.

    • M4ndU 2018.08.17 16:22 신고

      메세지를 받아서 응답하는 과정에 print("1")같은거 추가해서 뜨는지 확인해보세요

728x90
반응형

이번에는 저번에 만들었던 디스코드 챗봇에 급식 파서를 활용하여

급식을 알려주는 챗봇을 만들도록 하겠습니다.



이전과정

[

디스코드 챗봇 만들기 --> http://mandu-mandu.tistory.com/64

급식 파서 만들기 --> http://mandu-mandu.tistory.com/21

]






코드 작성하기


필요한 모듈로는, datetime 모듈과 만들어둔 parser.py가 필요합니다.


parser.py를 디스코드 봇 소스가 있는 디렉토리로 옮겨두고


아래 코드를, 디스코드 봇 소스 상단에 추가해줍시다.



import datetime

from parser import *





내일의 식단을 출력하기



먼저, 내일의 식단을 출력하는 코드를 작성하겠습니다.


코드를 작성하기 위해 내일의 식단을 출력하는 그 과정을 구상해 봅시다.


1. 디스코드에서 사용자로부터 명령어를 인식합니다.

저는 '!t' 로 설정하겠습니다.


2. 명령어를 인식한 시점으로부터 다음날의 날짜값과 그 날짜의 요일값을 가져옵니다.

파서의 함수가 필요로 하는 인자가 조식.중식.석식 여부 / 날짜 / 요일이기 때문입니다.

모두 datetime 함수를 사용합니다.


3. 급식을 파싱하는 함수를 호출합니다. 그리고 그 값을 저장합니다.


4. 그 값을 출력합니다.

제가 다니는 학교의 급식 상황을 보면, 조식이 전혀 없고 중식이 없을 경우 100% 석식이 없습니다.

이에 따라 조건절을 추가하여 출력되는 메세지를 '최적화'하였습니다.

(식단 내용 외 날짜등의 정보도 같이 출력해주기 때문입니다.)


중식이 없을 경우 급식이 없다고 출력,

중식이 있고 석식은 없을 경우 중식만 출력,

중식 석식 모두 있을 경우 모두 출력.



이를 코드로 구현하면 아래처럼 됩니다.


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
    if message.content.startswith('!to'):    #메세지의 내용의 시작이 !to로 시작할 경우
        to_tomorrow = datetime.datetime.today() + datetime.timedelta(days=1)    #오늘 날짜에 하루를 더함
        local_date2 = to_tomorrow.strftime("%Y.%m.%d")    #위에서 구한 날짜를 년.월.일 형식으로 저장
        local_weekday2 = to_tomorrow.weekday()    #위에서  구한 날짜의 요일값을 저장
 
        l_diet = get_diet(2, local_date2, local_weekday2)    #점심식단을 파싱해옴
        d_diet = get_diet(3, local_date2, local_weekday2)    #석식식단을 파싱해옴
 
        if len(l_diet) == 1:    #점심식단의 길이가 1일경우 = parser.py에서 식단이 없을경우 공백한자리를 반환함.
            await client.send_message(message.channel, "급식이 없습니다.")    #급식이 없다고 메세지 보냄
        elif len(d_diet) == 1:    #점심식단의 길이가 1이 아니고 석식식단의 길이가 1일경우 = 점심식단만 있을경우
            lunch = local_date2 + " 중식\n" + l_diet    #날짜와 "중식"을 앞에 붙여서
            await client.send_message(message.channel, lunch)    #메세지 보냄
        else:    #둘다 길이가 1이 아닐경우 = 점심, 석식 식단 모두 있을 경우
            lunch = local_date2 + " 중식\n" + l_diet    #앞에 부가적인 내용을 붙여서
            dinner = local_date2 + " 석식\n" + d_diet
            await client.send_message(message.channel, lunch)    #메세지를 보냄
            await client.send_message(message.channel, dinner)
cs



코드상 

@client.event

async def on_message(message):

아래에 위치하게 됩니다.



상황에 따라 if가 이미 있을 경우

if message.content.startswith('!to'):

if를 elif로 바꿔서 사용해야 합니다. (파이썬의 기본...)





특정 날짜의 식단을 출력하기



이제 날짜값을 주면 그 날짜의 급식을 출력하는 코드를 작성하겠습니다.


코드의 구성은 위 '내일의 식단을 출력하기'의 코드에서 2번의 내용만 수정해 주면 됩니다.


1. 디스코드에서 사용자로부터 명령어를 인식합니다.

저는 '!t' 로 설정하겠습니다.


2. 날짜를 입력하라는 메세지를 보냅니다. 그리고 받은 날짜의 요일값을 구합니다. (코드로 구현하면 길어집니다.)

datetime 모듈을 이용합니다.


3. 급식을 파싱하는 함수를 호출합니다. 그리고 그 값을 저장합니다.


4. 그 값을 출력합니다.

제가 다니는 학교의 급식 상황을 보면, 조식이 전혀 없고 중식이 없을 경우 100% 석식이 없습니다.

이에 따라 조건절을 추가하여 출력되는 메세지를 '최적화'하였습니다.

(식단 내용 외 날짜등의 정보도 같이 출력해주기 때문입니다.)


중식이 없을 경우 급식이 없다고 출력,

중식이 있고 석식은 없을 경우 중식만 출력,

중식 석식 모두 있을 경우 모두 출력.


이를 코드로 구현하면 아래와 같습니다.


 

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
   elif message.content.startswith('!g'):
        await client.send_message(message.channel, '날짜를 보내주세요...')    #날짜를 보내달라는 메세지를 보냄
        meal_date = await client.wait_for_message(timeout=15.0, author=message.author)    #제한시간은 15초
 
        if meal_date is None:    #값이 존재하지 않거나 시간이 초과되었을 경우
            await client.send_message(message.channel, '15초내로 입력해주세요. 다시시도 : !g')    #다시 시도하라는 메세지를 보냄
            return
 
        else:    #값이 있다면
            meal_date = str(meal_date.content) # str형으로 변환, (사용자로부터 20180219와 같은 형태로 받아야 합니다.)
            meal_date = '20' + meal_date[:2+ '.' + meal_date[2:4+ '.' + meal_date[4:6]    # 2018.02.19 사이에 점을 추가함
            #(사용자로부터 점이 포함된 값으로 받을경우 위 코드를 삭제해도 됩니다.)
 
            s = meal_date.replace('.'', ')     # 2018, 02, 19 점을 반점으로 교체
            ss = "datetime.datetime(" + s + ").weekday()"    #eval함수를 통해 요일값을 구하기 위한 작업
            try:
                whatday = eval(ss)    #요일값을 구해서 whatday에 저장
            except:    #오류가 날 경우 다시 시도하라는 메세지를 보냄
                await client.send_message(message.channel, '올바른 값으로 다시 시도하세요 : !g')
                return
#이하 '내일 식단을 출력하기'와 같음
            l_diet = get_diet(2, meal_date, whatday)
            d_diet = get_diet(3, meal_date, whatday)
 
            if len(l_diet) == 1:
                l_diet = "급식이 없습니다."
                await client.send_message(message.channel, embed=l_diet)
            elif len(d_diet) == 1:
                lunch = meal_date + " 중식\n" + l_diet
                await client.send_message(message.channel, embed=lunch)
            else:
                lunch = meal_date + " 중식\n" + l_diet
                dinner = meal_date + " 석식\n" + d_diet
                await client.send_message(message.channel, lunch)
                await client.send_message(message.channel, dinner)
 
cs

파서의 함수를 호출하는 부분에서 윗부분이 많이 복잡해진 것을 알 수 있습니다.

아래부분은 기존과 변화가 없기 때문에 주석을 달지 않았습니다.



위 둘을 합친 전체 코드는 아래와 같게 됩니다.



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
import discord
import asyncio
import datetime
from parser import *
 
 
client = discord.Client()
 
@client.event
async def on_ready():
    print('Logged in as')
    print(client.user.name)
    print(client.user.id)
    print('------')
 
@client.event
async def on_message(message):
    if message.content.startswith('!to'):    #메세지의 내용의 시작이 !to로 시작할 경우
        to_tomorrow = datetime.datetime.today() + datetime.timedelta(days=1)    #오늘 날짜에 하루를 더함
        local_date2 = to_tomorrow.strftime("%Y.%m.%d")    #위에서 구한 날짜를 년.월.일 형식으로 저장
        local_weekday2 = to_tomorrow.weekday()    #위에서  구한 날짜의 요일값을 저장
 
        l_diet = get_diet(2, local_date2, local_weekday2)    #점심식단을 파싱해옴
        d_diet = get_diet(3, local_date2, local_weekday2)    #석식식단을 파싱해옴
 
        if len(l_diet) == 1:    #점심식단의 길이가 1일경우 = parser.py에서 식단이 없을경우 공백한자리를 반환함.
            await client.send_message(message.channel, "급식이 없습니다.")    #급식이 없다고 메세지 보냄
        elif len(d_diet) == 1:    #점심식단의 길이가 1이 아니고 석식식단의 길이가 1일경우 = 점심식단만 있을경우
            lunch = local_date2 + " 중식\n" + l_diet    #날짜와 "중식"을 앞에 붙여서
            await client.send_message(message.channel, lunch)    #메세지 보냄
        else:    #둘다 길이가 1이 아닐경우 = 점심, 석식 식단 모두 있을 경우
            lunch = local_date2 + " 중식\n" + l_diet    #앞에 부가적인 내용을 붙여서
            dinner = local_date2 + " 석식\n" + d_diet
            await client.send_message(message.channel, lunch)    #메세지를 보냄
            await client.send_message(message.channel, dinner)
 
 
   elif message.content.startswith('!g'):
        await client.send_message(message.channel, '날짜를 보내주세요...')    #날짜를 보내달라는 메세지를 보냄
        meal_date = await client.wait_for_message(timeout=15.0, author=message.author)    #제한시간은 15초
 
        if meal_date is None:    #값이 존재하지 않거나 시간이 초과되었을 경우
            await client.send_message(message.channel, '15초내로 입력해주세요. 다시시도 : !g')    #다시 시도하라는 메세지를 보냄
            return
 
        else:    #값이 있다면
            meal_date = str(meal_date.content) # str형으로 변환, (사용자로부터 20180219와 같은 형태로 받아야 합니다.)
            meal_date = '20' + meal_date[:2+ '.' + meal_date[2:4+ '.' + meal_date[4:6]    # 2018.02.19 사이에 점을 추가함
            #(사용자로부터 점이 포함된 값으로 받을경우 위 코드를 삭제해도 됩니다.)
 
            s = meal_date.replace('.'', ')     # 2018, 02, 19 점을 반점으로 교체
            ss = "datetime.datetime(" + s + ").weekday()"    #eval함수를 통해 요일값을 구하기 위한 작업
            try:
                whatday = eval(ss)    #요일값을 구해서 whatday에 저장
            except:    #오류가 날 경우 다시 시도하라는 메세지를 보냄
                await client.send_message(message.channel, '올바른 값으로 다시 시도하세요 : !g')
                return
            #이하 '내일 식단을 출력하기'와 같음
            l_diet = get_diet(2, meal_date, whatday)
            d_diet = get_diet(3, meal_date, whatday)
 
            if len(l_diet) == 1:
                l_diet = "급식이 없습니다."
                await client.send_message(message.channel, embed=l_diet)
            elif len(d_diet) == 1:
                lunch = meal_date + " 중식\n" + l_diet
                await client.send_message(message.channel, embed=lunch)
            else:
                lunch = meal_date + " 중식\n" + l_diet
                dinner = meal_date + " 석식\n" + d_diet
                await client.send_message(message.channel, lunch)
                await client.send_message(message.channel, dinner)
 
client.run('token')
 
cs



# 글을 작성하면서 보니

#[23,35]과 [59,72]가 하는 일이 같은데

#함수로 하나 정의해서 같이 쓰는게

#효율적일 것 같다는 생각이 듭니다..


#지금은 귀찮으니 PASS


#############

18.03.03 추가

!g 를 통해 특정 날짜의 급식을 부르는 과정에서

날짜값을 받고 처리하는 과정에서 한자리 달의 경우 오류가 발생,

이 부분을 수정하여 한자리 달의 경우도 가능합니다.

수정된 코드는 깃허브를 확인.

https://github.com/M4ndU/inhun_discord_chat_bot_2

############


728x90
반응형
728x90
반응형

이 글만을 통해서 기본적으로 텍스트를 출력하는 디스코드 봇을 만들 수 있습니다.


1. 봇 생성하고 초대하기

2. 코드 작성하기

3. 테스트 하기




봇 생성하고 초대하기


먼저 봇 계정을 생성해 주어야 합니다.



디스코드 홈페이지로 이동합니다. -> https://discordapp.com/



상단에 알아보기 > 개발자





APPLICATIONS > My Apps 에서 New App을 누릅니다.





이름, 설명, 아이콘을 설정하고 Create App을 누릅니다.





내려서 Create a Bot User을 누릅니다.





Yes, do it!





Token: click to reveal 을 눌러서 토큰 값을 확인합니다. 이 값은 외부에 유출되어서는 안됩니다.


Public Bot을 체크하게 되면 누구나 봇을 서버에 추가할 수 있게 됩니다. 체크하지 않을 경우 오직 주인만이 서버에 추가할 수 있습니다.


Require OAuth2 Code Grant 는 모르겠습니다.. 아시는 분은 덧글 남겨주세요 ㅎㅎ





이번에는 봇 추가 링크를 만들겠습니다.

Generate OAuth2 URL을 클릭합니다.





CLIENT ID에 클라이언트 아이디를 입력합니다. 아마 자동으로 입력되어 있을거에요.

SCOPES에서 bot에 체크합니다.


그러면 밑에 탭이 하나 더 생깁니다.





권한 설정을 할 수 있는데요. 저는 Send Messages 와 Read Message History 이 두개만 체크해 두었습니다.


다시 위로 올라가서 생성된 URL을 복사합니다.


복사된 url로 접속하시고 자신의 서버를 선택하신뒤

승인 버튼을 누르고 리캡챠 체크 하시면 자신의 서버에 봇이 들어와 있게 됩니다.


현재 봇은 오프라인 상태가 됩니다.




코드 작성하기



개발환경 : UBUNTU 16.04.3

개발언어: PYTHON 3.5.2

텍스트 에디터: ATOM


디스코드 봇을 파이썬으로 작성하기 편하게 제작된 모듈이 있습니다.

아래 링크에 들어가서 discord.py를 설치합니다.

설치방법은 아래 링크에 모두 있습니다.


 바로가기 <-- (https://github.com/Rapptz/discord.py)




설치를 마쳤다면,

이제 텍스트 에디터로 가서 소스를 작성해 봅시다.



[2019년 공지사항]

discord.py 모듈의 업데이트로 client.send_message를 사용할 수 없습니다.

대신 message.channel.send를 이용하시면 됩니다.

아래 코드에는 수정되지 않은 상태로 올라가있으니 수정하셔서 사용하시면 됩니다.





기본적인 소스는 다음과 같습니다.


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
import discord
import asyncio
 
client = discord.Client()
 
@client.event
async def on_ready():
    print('Logged in as')
    print(client.user.name)
    print(client.user.id)
    print('------')
 
@client.event
async def on_message(message):
    if message.content.startswith('!test'):
        await client.send_message(message.channel, 'test!!!!')
 
    elif message.content.startswith('!say'):
        await client.send_message(message.channel, 'leave message')
        msg = await client.wait_for_message(timeout=15.0, author=message.author)
 
        if msg is None:
            await client.send_message(message.channel, '15초내로 입력해주세요. 다시시도: !say')
            return
        else:
            await client.send_message(message.channel, msg.content)
 
client.run('token')
 
cs



위에서부터 소스를 분석해보죠.


@client.event

async def on_ready():

    print('Logged in as')

    print(client.user.name)

    print(client.user.id)

    print('------')



봇을 실행하여 로그인에 성공하면, 위에 문장을 출력하게 됩니다.

(터미널 or 콘솔창에 출력됩니다.)

디스코드 내에서는 봇이 온라인으로 표시됩니다.




@client.event

async def on_message(message):

    if message.content.startswith('!test'):

        await client.send_message(message.channel, 'test!!!!')

 


디스코드 텍스트 채널에 !test 라고 메세지를 보내면 봇이 test!!!!라고 메세지를 보냅니다.




    elif message.content.startswith('!say'):

        await client.send_message(message.channel, 'leave message')

        msg = await client.wait_for_message(timeout=15.0, author=message.author)

 


디스코드 텍스트 채널에 !say 라고 메세지를 보내면 봇이 leave message 라고 메세지를 보냅니다.

그리고 15초 동안 그 사용자의 메세지를 기다립니다. 그리고 그 사용자의 메세지를 msg에 저장합니다.


제한시간은 timout의 값을 변경하여 설정할 수 있습니다.




        if msg is None:

            await client.send_message(message.channel, '15초내로 입력해주세요. 다시시도: !say')

            return

        else:

            await client.send_message(message.channel, msg.content)

 


msg의 값이 존재하지 않을경우 (=사용자가 메세지를 보내지 않았을 경우)

문장을 하나 출력합니다.


msg의 값이 존재할 경우 (=사용자가 메세지를 보냈을 경우)

그 메세지를 그대로 보냅니다.




client.run('token')



token자리에 자신의 봇의 토큰값으로 코드를 수정해 주고 실행해야 합니다.


ex) client.run('AAAAAAA.111111111_BBBBBB')


작은따옴표 주의







테스트 하기



코드를 작성하고 저장을 했다면, 이제 실행을 해줍시다.



터미널에서 봇이 로그인이 되었는지 확인하고 디스코드로 갑니다.




로그인이 되었다면, 봇이 온라인으로 표시될 겁니다.




텍스트 채널에서 메세지를 보내서 봇이 정상적으로 동작하는지 확인합니다.




만약 디스코드내에서 봇이 응답하지 못하고, 터미널에서 위와같은 오류가 났을경우,

디스코드 텍스트 채널에서 권한이 없어서 메세지를 보내지 못하는 것이므로

봇에 권한설정을 해 주시면 해결됩니다.

728x90
반응형
  1. 이전 댓글 더보기
  2. daniel-1323 2019.04.12 22:47

    'import discord' 가 이름이 없는 모듈이라는데 어떻게 해결을 해야 하나요..?

  3. 코딩하는사람 2019.04.29 22:18

    혹시
    if msg is None:

    await client.send_message(message.channel, '15초내로 입력해주세요. 다시시도: !say')

    return

    else:

    await client.send_message(message.channel, msg.content)
    여기서 만약 제가 "놀자" 라는 메세지를 말하면
    그떄에만 메세지를 보내게 할수있는 방법은없나요?

    • M4ndU 2019.04.30 07:22 신고

      async def on_message(message):

          if message.content.startswith('놀자'):

              await client.send_message(message.channel, '보낼 메세지')


      이 코드를 사용하시면 됩니다.

  4. 코딩하는사람 2019.04.30 22:47

    어...그게아니라
    if msg is None:

    await client.send_message(message.channel, '15초내로 입력해주세요. 다시시도: !say')

    return

    else:

    await client.send_message(message.channel, msg.content)
    이코드에서 15초안에 놀자 라는 메세지를 입력하면 답을하고 만약 메세지를 입력하지 않으면 다른 메세지를 입력하는 그런 방법은 없을까요...

  5. DADO 2019.05.14 22:28

    명령어 쓰고 bot.py라고 저장한 다음에 파일 눌러 실행 해봤지만 봇이 온라인으로 되지 않는데 어떡해야하나요?

    • M4ndU 2019.05.15 07:24 신고

      python3 bot.py를 실행하였을 때 본문에 나온 스크린 샷과 같은 형태의 내용이 나왔나요?

      만약 나왔다면 종료하지 않고 그대로 두셔야 합니다.

  6. 초보자12314235 2019.07.10 23:09

    어떻게 실행하나요?

  7. 초보자123124314 2019.07.10 23:13

    cmd창이잠깐 떳다가 바로꺼지네요

    • M4ndU 2019.07.12 22:34 신고

      cmd창을 켜두신 상태에서 python 파일이름 을 실행하시면 됩니다

  8. NTP 2019.09.07 11:43

    Traceback (most recent call last):
    File "discord.py", line 1, in <module>
    import discord
    File "C:\Users\R6\Desktop\discord.py", line 4, in <module>
    client = discord.Client()
    AttributeError: module 'discord' has no attribute 'Client'
    오류가 뜨는데 어떻게하면 좋죠??

    • M4ndU 2019.09.07 12:34 신고

      파일 명을 discord.py가 아닌 다른 이름으로 변경하세요.

  9. 정보감사합니다그런데... 2019.09.08 04:01

    Ignoring exception in on_message
    Traceback (most recent call last):
    File "C:\Users\파트라슈\AppData\Local\Programs\Python\Python36-32\lib\site-packages\discord\client.py", line 270, in _run_event
    await coro(*args, **kwargs)
    File "C:\Users\파트라슈\Desktop\isabelle_bot.py", line 16, in on_message
    await client.send_message(message.channel, 'test!!!!')
    AttributeError: 'Client' object has no attribute 'send_message'

    강좌덕분에 온라인으로 구동까지는 성공했습니다 감사합니다. 그런데 !test를 입력하면 이러한 빨간 글씨가 뜨네요 혹시 어떻게 해결할 수 있는지 알 수 있을까요?ㅠㅠ

    • M4ndU 2019.09.08 12:42 신고

      await client.send_message 부분이discord.py의 최신버전에서 변경된 것 같습니다.
      await message.channel.send 함수로 변경해주시면 될 것 같습니다.

  10. 로호 2019.09.18 22:06

    Traceback (most recent call last):
    File "/home/wook/.local/lib/python3.6/site-packages/discord/http.py", line 258, in static_login
    data = await self.request(Route('GET', '/users/@me'))
    File "/home/wook/.local/lib/python3.6/site-packages/discord/http.py", line 222, in request
    raise HTTPException(r, data)
    discord.errors.HTTPException: 401 UNAUTHORIZED (error code: 0): 401: Unauthorized

    The above exception was the direct cause of the following exception:

    Traceback (most recent call last):
    File "chat.py", line 15, in <module>
    bot.run('봇의 토오오오오오오오크으으으으으은')
    File "/home/wook/.local/lib/python3.6/site-packages/discord/client.py", line 598, in run
    return future.result()
    File "/home/wook/.local/lib/python3.6/site-packages/discord/client.py", line 579, in runner
    await self.start(*args, **kwargs)
    File "/home/wook/.local/lib/python3.6/site-packages/discord/client.py", line 542, in start
    await self.login(*args, bot=bot)
    File "/home/wook/.local/lib/python3.6/site-packages/discord/client.py", line 400, in login
    await self.http.static_login(token, bot=bot)
    File "/home/wook/.local/lib/python3.6/site-packages/discord/http.py", line 262, in static_login
    raise LoginFailure('Improper token has been passed.') from exc
    discord.errors.LoginFailure: Improper token has been passed.

    뭐가 문제인건가요....?

    • M4ndU 2019.09.18 23:35 신고

      bot.run('봇의 토오오오오오오오크으으으으으은')

      토큰값을 넣어주셔야 합니다.

  11. wkdgus 2019.10.11 23:26

    혹시 비주얼 스튜디오로도 코딩이 가능한가요?
    '

  12. 혹시 2020.02.03 13:38

    텍스트 에디터 뭐 쓰시는지 알 수 있을까요

  13. zero 2020.03.10 18:09

    맥os 에서 실행하는 방법은 없을까요?

  14. 5TreeCrew Bot 2020.04.04 13:46

    E:\Users\user\PycharmProjects\ndc1\venv\Scripts\python.exe E:/Users/user/PycharmProjects/ndc1/ndc2.py
    Traceback (most recent call last):
    File "E:\Users\user\PycharmProjects\ndc1\venv\lib\site-packages\discord\http.py", line 261, in static_login
    data = await self.request(Route('GET', '/users/@me'))
    File "E:\Users\user\PycharmProjects\ndc1\venv\lib\site-packages\discord\http.py", line 225, in request
    raise HTTPException(r, data)
    discord.errors.HTTPException: 401 Unauthorized (error code: 0): 401: Unauthorized

    The above exception was the direct cause of the following exception:

    Traceback (most recent call last):
    File "E:/Users/user/PycharmProjects/ndc1/ndc2.py", line 31, in <module>
    client.run('9E7kaGHp7LmdhA-AtPSRru_Nt647q0jt')
    File "E:\Users\user\PycharmProjects\ndc1\venv\lib\site-packages\discord\client.py", line 640, in run
    return future.result()
    File "E:\Users\user\PycharmProjects\ndc1\venv\lib\site-packages\discord\client.py", line 621, in runner
    await self.start(*args, **kwargs)
    File "E:\Users\user\PycharmProjects\ndc1\venv\lib\site-packages\discord\client.py", line 584, in start
    await self.login(*args, bot=bot)
    File "E:\Users\user\PycharmProjects\ndc1\venv\lib\site-packages\discord\client.py", line 442, in login
    await self.http.static_login(token.strip(), bot=bot)
    File "E:\Users\user\PycharmProjects\ndc1\venv\lib\site-packages\discord\http.py", line 265, in static_login
    raise LoginFailure('Improper token has been passed.') from exc
    discord.errors.LoginFailure: Improper token has been passed.
    Exception ignored in: <function _ProactorBasePipeTransport.__del__ at 0x03B38898>
    Traceback (most recent call last):
    File "C:\Users\user\AppData\Local\Programs\Python\Python38-32\lib\asyncio\proactor_events.py", line 116, in __del__
    self.close()
    File "C:\Users\user\AppData\Local\Programs\Python\Python38-32\lib\asyncio\proactor_events.py", line 108, in close
    self._loop.call_soon(self._call_connection_lost, None)
    File "C:\Users\user\AppData\Local\Programs\Python\Python38-32\lib\asyncio\base_events.py", line 715, in call_soon
    self._check_closed()
    File "C:\Users\user\AppData\Local\Programs\Python\Python38-32\lib\asyncio\base_events.py", line 508, in _check_closed
    raise RuntimeError('Event loop is closed')
    RuntimeError: Event loop is closed

    Process finished with exit code 1

    • M4ndU 2020.04.04 15:58 신고

      토큰값이 보입니다.
      토큰 재발급받으시길..

  15. a 2020.04.13 20:16

    Traceback (most recent call last):
    File "C:\Users\PC\PycharmProjects\untitled2\venv\lib\site-packages\discord\http.py", line 261, in static_login
    data = await self.request(Route('GET', '/users/@me'))
    File "C:\Users\PC\PycharmProjects\untitled2\venv\lib\site-packages\discord\http.py", line 225, in request
    raise HTTPException(r, data)
    discord.errors.HTTPException: 401 Unauthorized (error code: 0): 401: Unauthorized

    The above exception was the direct cause of the following exception:

    Traceback (most recent call last):
    File "C:/Users/PC/PycharmProjects/untitled2/a.py", line 21, in <module>
    client.run('토큰임')
    File "C:\Users\PC\PycharmProjects\untitled2\venv\lib\site-packages\discord\client.py", line 640, in run
    return future.result()
    File "C:\Users\PC\PycharmProjects\untitled2\venv\lib\site-packages\discord\client.py", line 621, in runner
    await self.start(*args, **kwargs)
    File "C:\Users\PC\PycharmProjects\untitled2\venv\lib\site-packages\discord\client.py", line 584, in start
    await self.login(*args, bot=bot)
    File "C:\Users\PC\PycharmProjects\untitled2\venv\lib\site-packages\discord\client.py", line 442, in login
    await self.http.static_login(token.strip(), bot=bot)
    File "C:\Users\PC\PycharmProjects\untitled2\venv\lib\site-packages\discord\http.py", line 265, in static_login
    raise LoginFailure('Improper token has been passed.') from exc
    discord.errors.LoginFailure: Improper token has been passed.
    라고뜨네요 어떻게해야하나요

    • M4ndU 2020.04.14 02:35 신고

      올바른 token값을 사용했는지 확인해 보세요.

  16. a 2020.04.15 00:03

    답글다는법을몰라서 이렇게쓰겠습니다
    토큰을 정확이넣었는데도 이러네요

    • M4ndU 2020.04.15 00:34 신고

      https://github.com/Rapptz/discord.py/blob/master/examples/edits.py

      여기에 있는 4, 19행을 추가해보세요.

      제가 게시물에서 사용했던 discord.py는 구버전이라서 그럴 수도 있습니다.

  17. 평범한 사람 2020.05.08 09:40

    Ignoring exception in on_message
    Traceback (most recent call last):
    File "C:\Users\Admin\AppData\Local\Programs\Python\Python38-32\lib\site-packages\discord\client.py", line 312, in _run_event
    await coro(*args, **kwargs)
    File "C:\Users\Admin\Desktop\discord bot\새 텍스트 문서.py", line 17, in on_message
    await message.channel.send(message.channel, 'test!!!!')
    TypeError: send() takes from 1 to 2 positional arguments but 3 were given

    이렇게 뜨는데 답장 해주시나요?

    • 어려웡 2020.05.20 02:10

      if message.content.startswith(messsage.channel, '!test'):
      await message.channel.send(messsage.channel, 'test!!!!')


      에서 가로안의 messsage.channel, 을 지우니까 해결되었습니다. (아래와 같이)

      if message.content.startswith('!test'):
      await message.channel.send('test!!!!')

    • M4ndU 2020.05.20 02:29 신고

      discord py 버전에 따라서 사용법이 달라집니다.

  18. enhw4295- 2020.11.18 13:42

    재미있는 내용 잘 보고 가용~

  19. ... 2020.12.19 13:05

    분홍색으로 되있는거 클릭했는데 설치하는게 어딧어요?

  20. ... 2020.12.19 16:01

    중간에 분홍색 링크를 클릭했는데 그다음에 어떻게해요?

    • M4ndU 2020.12.19 16:54 신고

      Installing 에 나와있는 대로 하시면 됩니다.

      기본적으로 파이썬을 다루실 줄 아셔야 합니다.

  21. 므시 2021.02.06 00:11

    저 봇 링크 생성하려는데 자꾸 뭐가 인증이 안됐다고 못하게 해요

728x90
반응형

1. Welcome!!



flag : TWCTF{Welcome_To_TWCTF2017!!}





2. Rev Rev Rev


간단한 리버싱 문제다.




32bit ELF 파일이기 때문에 IDA 32bit로 열어보았다.




2개의 분기점을 통과한다. 피해야할 주소값과 가야할 주소값을 찾아서 angr을 이용해 풀었다.




exploit.py

1
2
3
4
5
6
7
8
9
10
11
12
13
import angr
 
 
def main():
= angr.Project("./rev",load_options={'auto_load_libs':False})
ex = p.factory.path_group()
ex.explore(find=0x804867c, avoid=(0x804860d0x804868e))
ex.run()
return ex.found[0].state.posix.dumps(0).split('\0')[0]
 
 
if __name__ == '__main__':
print (main())

cs






Flag : TWCTF{qpzisyDnbmboz76oglxpzYdk}


728x90
반응형

+ Recent posts