-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathlazyown_infinitestorage.py
133 lines (113 loc) · 8.67 KB
/
lazyown_infinitestorage.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
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
import cv2
import numpy as np
import argparse
import os
import subprocess
import tempfile
BANNER = """
▄█ ▄████████ ▄███████▄ ▄██ ▄ ▄██████▄ ▄█ █▄ ███▄▄▄▄
███ ███ ███ ██▀ ▄██ ███ ██▄ ███ ███ ███ ███ ███▀▀▀██▄
███ ███ ███ ▄███▀ ███▄▄▄███ ███ ███ ███ ███ ███ ███
███ ███ ███ ▀█▀▄███▀▄▄ ▀▀▀▀▀▀███ ███ ███ ███ ███ ███ ███
███ ▀███████████ ▄███▀ ▀ ▄██ ███ ███ ███ ███ ███ ███ ███
███ ███ ███ ▄███▀ ███ ███ ███ ███ ███ ███ ███ ███
███▌ ▄ ███ ███ ███▄ ▄█ ███ ███ ███ ███ ███ ▄█▄ ███ ███ ███
█████▄▄██ ███ █▀ ▀████████▀ ▀█████▀ ▀██████▀ ▀███▀███▀ ▀█ █▀
▀
▄█ ███▄▄▄▄ ▄████████ ▄█ ███▄▄▄▄ ▄█ ███ ▄████████
███ ███▀▀▀██▄ ███ ███ ███ ███▀▀▀██▄ ███ ▀█████████▄ ███ ███
███▌ ███ ███ ███ █▀ ███▌ ███ ███ ███▌ ▀███▀▀██ ███ █▀
███▌ ███ ███ ▄███▄▄▄ ███▌ ███ ███ ███▌ ███ ▀ ▄███▄▄▄
███▌ ███ ███ ▀▀███▀▀▀ ███▌ ███ ███ ███▌ ███ ▀▀███▀▀▀
███ ███ ███ ███ ███ ███ ███ ███ ███ ███ █▄
███ ███ ███ ███ ███ ███ ███ ███ ███ ███ ███
█▀ ▀█ █▀ ███ █▀ ▀█ █▀ █▀ ▄████▀ ██████████
▄████████ ███ ▄██████▄ ▄████████ ▄████████ ▄██████▄ ▄████████
███ ███ ▀█████████▄ ███ ███ ███ ███ ███ ███ ███ ███ ███ ███
███ █▀ ▀███▀▀██ ███ ███ ███ ███ ███ ███ ███ █▀ ███ █▀
███ ███ ▀ ███ ███ ▄███▄▄▄▄██▀ ███ ███ ▄███ ▄███▄▄▄
▀███████████ ███ ███ ███ ▀▀███▀▀▀▀▀ ▀███████████ ▀▀███ ████▄ ▀▀███▀▀▀
███ ███ ███ ███ ▀███████████ ███ ███ ███ ███ ███ █▄
▄█ ███ ███ ███ ███ ███ ███ ███ ███ ███ ███ ███ ███
▄████████▀ ▄████▀ ▀██████▀ ███ ███ ███ █▀ ████████▀ ██████████
███ ███
"""
def encode_file_to_images(input_file, temp_dir, frame_size, block_size=4):
with open(input_file, 'rb') as f:
data = f.read()
bit_data = ''.join(format(byte, '08b') for byte in data)
bit_data += '1' + '0' * ((8 - len(bit_data) % 8) % 8)
width, height = frame_size
bits_per_frame = (width // block_size) * (height // block_size)
frame_count = (len(bit_data) + bits_per_frame - 1) // bits_per_frame
for i in range(frame_count):
frame_bits = bit_data[i * bits_per_frame : (i + 1) * bits_per_frame]
frame_bits = frame_bits.ljust(bits_per_frame, '0')
frame = np.zeros((height, width), dtype=np.uint8)
for idx, bit in enumerate(frame_bits):
x = (idx % (width // block_size)) * block_size
y = (idx // (width // block_size)) * block_size
color = 255 if bit == '1' else 0
frame[y:y+block_size, x:x+block_size] = color
frame_path = os.path.join(temp_dir, f'frame_{i:05d}.png')
cv2.imwrite(frame_path, frame)
def encode_file_to_video(input_file, output_file, frame_size, fps, block_size=4):
with tempfile.TemporaryDirectory() as temp_dir:
encode_file_to_images(input_file, temp_dir, frame_size, block_size)
width, height = frame_size
resolution = f"{width}x{height}"
output_file_with_resolution = f"{os.path.splitext(output_file)[0]}_{resolution}.mp4"
subprocess.run([
'ffmpeg', '-y', '-framerate', str(fps), '-i', os.path.join(temp_dir, 'frame_%05d.png'),
'-c:v', 'libx264', '-pix_fmt', 'yuv420p', '-movflags', 'faststart', output_file_with_resolution
])
def decode_video_to_file(input_file, output_file, block_size=4):
cap = cv2.VideoCapture(input_file)
# Extract original resolution from file name
file_name = os.path.splitext(os.path.basename(input_file))[0]
resolution_str = file_name.split('_')[-1]
width, height = map(int, resolution_str.split('x'))
bit_data = ''
while cap.isOpened():
ret, frame = cap.read()
if not ret:
break
gray_frame = cv2.cvtColor(frame, cv2.COLOR_BGR2GRAY)
resized_frame = cv2.resize(gray_frame, (width, height), interpolation=cv2.INTER_AREA)
for y in range(0, height, block_size):
for x in range(0, width, block_size):
if y + block_size > resized_frame.shape[0] or x + block_size > resized_frame.shape[1]:
bit = '0'
else:
bit = '1' if np.mean(resized_frame[y:y+block_size, x:x+block_size]) > 127 else '0'
bit_data += bit
cap.release()
print(f"Extracted bit data length: {len(bit_data)}")
print(f"Extracted bit data (first 512 bits): {bit_data[:512]}")
byte_data = bytearray()
for i in range(0, len(bit_data), 8):
byte_data.append(int(bit_data[i:i+8], 2))
end_marker = b'\x80'
end = byte_data.rfind(end_marker)
if end != -1:
byte_data = byte_data[:end]
with open(output_file, 'wb') as f:
f.write(byte_data)
print(f"Recovered byte data length: {len(byte_data)}")
print(f"Recovered byte data (first 64 bytes): {byte_data[:64]}")
def main():
print(BANNER)
parser = argparse.ArgumentParser(description='Encode and decode files to and from video')
parser.add_argument('--mode', choices=['encode', 'decode'], required=True, help='Operation mode: encode or decode')
parser.add_argument('--input', required=True, help='Input file')
parser.add_argument('--output', required=True, help='Output file')
parser.add_argument('--frame_size', type=int, nargs=2, required=True, help='Frame size (width height)')
parser.add_argument('--fps', type=int, default=30, help='Frames per second for video encoding (default: 30)')
parser.add_argument('--block_size', type=int, default=4, help='Size of the blocks representing bits (default: 4)')
args = parser.parse_args()
if args.mode == 'encode':
encode_file_to_video(args.input, args.output, tuple(args.frame_size), args.fps, args.block_size)
elif args.mode == 'decode':
decode_video_to_file(args.input, args.output, args.block_size)
if __name__ == '__main__':
main()