2018-06-05 16:58:04 +02:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
# -*-coding:UTF-8 -*
|
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
"""
|
|
|
|
The Submit paste module
|
|
|
|
================
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
##################################
|
|
|
|
# Import External packages
|
|
|
|
##################################
|
2018-06-05 16:58:04 +02:00
|
|
|
import os
|
|
|
|
import sys
|
|
|
|
import gzip
|
|
|
|
import base64
|
|
|
|
import datetime
|
2018-06-08 16:49:20 +02:00
|
|
|
import time
|
|
|
|
|
2021-06-02 16:04:52 +02:00
|
|
|
sys.path.append(os.environ['AIL_BIN'])
|
2021-04-28 15:24:33 +02:00
|
|
|
##################################
|
|
|
|
# Import Project packages
|
|
|
|
##################################
|
2021-06-02 16:04:52 +02:00
|
|
|
from modules.abstract_module import AbstractModule
|
2023-04-13 14:25:02 +02:00
|
|
|
from lib.objects.Items import ITEMS_FOLDER
|
2021-06-02 16:04:52 +02:00
|
|
|
from lib import ConfigLoader
|
2022-09-01 14:04:00 +02:00
|
|
|
from lib import Tag
|
2023-10-11 12:06:01 +02:00
|
|
|
from lib.objects.Items import Item
|
2018-06-08 16:49:20 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
class SubmitPaste(AbstractModule):
|
|
|
|
"""
|
2021-06-02 16:04:52 +02:00
|
|
|
SubmitPaste module for AIL framework
|
2021-04-28 15:24:33 +02:00
|
|
|
"""
|
2018-06-08 16:49:20 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
expire_time = 120
|
|
|
|
# Text max size
|
|
|
|
TEXT_MAX_SIZE = ConfigLoader.ConfigLoader().get_config_int("SubmitPaste", "TEXT_MAX_SIZE")
|
|
|
|
# File max size
|
|
|
|
FILE_MAX_SIZE = ConfigLoader.ConfigLoader().get_config_int("SubmitPaste", "FILE_MAX_SIZE")
|
|
|
|
# Allowed file type
|
|
|
|
ALLOWED_EXTENSIONS = ConfigLoader.ConfigLoader().get_config_str("SubmitPaste", "FILE_ALLOWED_EXTENSIONS").split(',')
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
"""
|
|
|
|
init
|
|
|
|
"""
|
2023-04-13 14:25:02 +02:00
|
|
|
super(SubmitPaste, self).__init__()
|
2021-04-28 15:24:33 +02:00
|
|
|
|
2023-04-04 14:12:23 +02:00
|
|
|
self.r_serv_db = ConfigLoader.ConfigLoader().get_db_conn("Kvrocks_DB")
|
2021-04-28 15:24:33 +02:00
|
|
|
self.r_serv_log_submit = ConfigLoader.ConfigLoader().get_redis_conn("Redis_Log_submit")
|
|
|
|
|
|
|
|
self.pending_seconds = 3
|
|
|
|
|
|
|
|
self.PASTES_FOLDER = os.path.join(os.environ['AIL_HOME'], ConfigLoader.ConfigLoader().get_config_str("Directories", "pastes")) + '/'
|
|
|
|
|
|
|
|
def compute(self, uuid):
|
|
|
|
"""
|
|
|
|
Main method of the Module to implement
|
|
|
|
"""
|
2023-05-12 15:29:53 +02:00
|
|
|
self.logger.debug(f'compute UUID {uuid}')
|
2021-04-28 15:24:33 +02:00
|
|
|
|
|
|
|
# get temp value save on disk
|
|
|
|
ltags = self.r_serv_db.smembers(f'{uuid}:ltags')
|
|
|
|
ltagsgalaxies = self.r_serv_db.smembers(f'{uuid}:ltagsgalaxies')
|
|
|
|
paste_content = self.r_serv_db.get(f'{uuid}:paste_content')
|
|
|
|
isfile = self.r_serv_db.get(f'{uuid}:isfile')
|
|
|
|
password = self.r_serv_db.get(f'{uuid}:password')
|
|
|
|
source = self.r_serv_db.get(f'{uuid}:source')
|
|
|
|
|
2021-05-28 17:37:46 +02:00
|
|
|
if source in ['crawled', 'tests']:
|
|
|
|
source = 'submitted'
|
|
|
|
|
2023-05-12 15:29:53 +02:00
|
|
|
self.logger.debug(f'isfile UUID {isfile}')
|
|
|
|
self.logger.debug(f'source UUID {source}')
|
|
|
|
self.logger.debug(f'paste_content UUID {paste_content}')
|
2021-04-28 15:24:33 +02:00
|
|
|
|
|
|
|
# needed if redis is restarted
|
|
|
|
self.r_serv_log_submit.set(f'{uuid}:end', 0)
|
|
|
|
self.r_serv_log_submit.set(f'{uuid}:processing', 0)
|
|
|
|
self.r_serv_log_submit.set(f'{uuid}:nb_total', -1)
|
|
|
|
self.r_serv_log_submit.set(f'{uuid}:nb_end', 0)
|
|
|
|
self.r_serv_log_submit.set(f'{uuid}:nb_sucess', 0)
|
|
|
|
|
|
|
|
self.r_serv_log_submit.set(f'{uuid}:processing', 1)
|
|
|
|
|
|
|
|
if isfile == 'True':
|
|
|
|
# file input
|
|
|
|
self._manage_file(uuid, paste_content, ltags, ltagsgalaxies, source)
|
2018-06-08 16:49:20 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
else:
|
|
|
|
# textarea input paste
|
|
|
|
self._manage_text(uuid, paste_content, ltags, ltagsgalaxies, source)
|
|
|
|
|
2021-05-28 17:37:46 +02:00
|
|
|
# new paste created from file, remove uuid ref
|
2021-04-28 15:24:33 +02:00
|
|
|
self.remove_submit_uuid(uuid)
|
|
|
|
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
"""
|
|
|
|
Run Module endless process
|
|
|
|
"""
|
2021-05-28 17:37:46 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
# Endless loop processing messages from the input queue
|
|
|
|
while self.proceed:
|
|
|
|
# Get one message (paste) from the QueueIn (copy of Redis_Global publish)
|
|
|
|
nb_submit = self.r_serv_db.scard('submitted:uuid')
|
2021-05-28 17:37:46 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
if nb_submit > 0:
|
|
|
|
try:
|
|
|
|
uuid = self.r_serv_db.srandmember('submitted:uuid')
|
2023-04-04 15:18:15 +02:00
|
|
|
if isinstance(uuid, list):
|
2023-04-04 15:25:59 +02:00
|
|
|
uuid = uuid[0]
|
2021-04-28 15:24:33 +02:00
|
|
|
# Module processing with the message from the queue
|
2023-05-12 15:29:53 +02:00
|
|
|
self.logger.debug(uuid)
|
2021-04-28 15:24:33 +02:00
|
|
|
self.compute(uuid)
|
|
|
|
except Exception as err:
|
2023-05-12 15:29:53 +02:00
|
|
|
self.logger.critical(err)
|
2021-05-28 17:37:46 +02:00
|
|
|
# Remove uuid ref
|
2021-04-28 15:24:33 +02:00
|
|
|
self.remove_submit_uuid(uuid)
|
|
|
|
else:
|
|
|
|
# Wait before next process
|
2023-05-12 15:29:53 +02:00
|
|
|
self.logger.debug(f'{self.module_name}, waiting for new message, Idling {self.pending_seconds}s')
|
2021-04-28 15:24:33 +02:00
|
|
|
time.sleep(self.pending_seconds)
|
|
|
|
|
|
|
|
def _manage_text(self, uuid, paste_content, ltags, ltagsgalaxies, source):
|
|
|
|
"""
|
|
|
|
Create a paste for given text
|
|
|
|
"""
|
|
|
|
if sys.getsizeof(paste_content) < SubmitPaste.TEXT_MAX_SIZE:
|
|
|
|
self.r_serv_log_submit.set(f'{uuid}:nb_total', 1)
|
|
|
|
self.create_paste(uuid, paste_content.encode(), ltags, ltagsgalaxies, uuid, source)
|
|
|
|
time.sleep(0.5)
|
|
|
|
else:
|
|
|
|
self.abord_file_submission(uuid, f'Text size is over {SubmitPaste.TEXT_MAX_SIZE} bytes')
|
|
|
|
|
|
|
|
def _manage_file(self, uuid, file_full_path, ltags, ltagsgalaxies, source):
|
|
|
|
"""
|
|
|
|
Create a paste for given file
|
|
|
|
"""
|
|
|
|
self.redis_logger.debug('manage')
|
|
|
|
|
|
|
|
if os.path.exists(file_full_path):
|
|
|
|
self.redis_logger.debug(f'file exists {file_full_path}')
|
2021-05-28 17:37:46 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
file_size = os.stat(file_full_path).st_size
|
|
|
|
self.redis_logger.debug(f'file size {file_size}')
|
|
|
|
# Verify file length
|
|
|
|
if file_size < SubmitPaste.FILE_MAX_SIZE:
|
|
|
|
# TODO sanitize filename
|
|
|
|
filename = file_full_path.split('/')[-1]
|
|
|
|
self.redis_logger.debug(f'sanitize filename {filename}')
|
|
|
|
self.redis_logger.debug('file size allowed')
|
|
|
|
|
|
|
|
if not '.' in filename:
|
|
|
|
self.redis_logger.debug('no extension for filename')
|
|
|
|
try:
|
|
|
|
# Read file
|
|
|
|
with open(file_full_path,'r') as f:
|
|
|
|
content = f.read()
|
|
|
|
self.r_serv_log_submit.set(uuid + ':nb_total', 1)
|
|
|
|
self.create_paste(uuid, content.encode(), ltags, ltagsgalaxies, uuid, source)
|
|
|
|
except:
|
|
|
|
self.abord_file_submission(uuid, "file error")
|
2018-06-08 16:49:20 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
else:
|
|
|
|
file_type = filename.rsplit('.', 1)[1]
|
|
|
|
file_type = file_type.lower()
|
|
|
|
self.redis_logger.debug(f'file ext {file_type}')
|
|
|
|
|
|
|
|
if file_type in SubmitPaste.ALLOWED_EXTENSIONS:
|
|
|
|
self.redis_logger.debug('Extension allowed')
|
|
|
|
# TODO enum of possible file extension ?
|
|
|
|
# TODO verify file hash with virus total ?
|
|
|
|
if not self._is_compressed_type(file_type):
|
|
|
|
self.redis_logger.debug('Plain text file')
|
|
|
|
# plain txt file
|
|
|
|
with open(file_full_path,'r') as f:
|
|
|
|
content = f.read()
|
|
|
|
self.r_serv_log_submit.set(uuid + ':nb_total', 1)
|
|
|
|
self.create_paste(uuid, content.encode(), ltags, ltagsgalaxies, uuid, source)
|
|
|
|
else:
|
|
|
|
# Compressed file
|
|
|
|
self.abord_file_submission(uuid, "file decompression should be implemented")
|
|
|
|
# TODO add compress file management
|
|
|
|
# #decompress file
|
|
|
|
# try:
|
|
|
|
# if password == None:
|
|
|
|
# files = unpack(file_full_path.encode())
|
|
|
|
# #print(files.children)
|
|
|
|
# else:
|
|
|
|
# try:
|
|
|
|
# files = unpack(file_full_path.encode(), password=password.encode())
|
|
|
|
# #print(files.children)
|
|
|
|
# except sflock.exception.IncorrectUsageException:
|
|
|
|
# self.abord_file_submission(uuid, "Wrong Password")
|
|
|
|
# raise
|
|
|
|
# except:
|
|
|
|
# self.abord_file_submission(uuid, "file decompression error")
|
|
|
|
# raise
|
|
|
|
# self.redis_logger.debug('unpacking {} file'.format(files.unpacker))
|
|
|
|
# if(not files.children):
|
|
|
|
# self.abord_file_submission(uuid, "Empty compressed file")
|
|
|
|
# raise
|
|
|
|
# # set number of files to submit
|
|
|
|
# self.r_serv_log_submit.set(uuid + ':nb_total', len(files.children))
|
|
|
|
# n = 1
|
|
|
|
# for child in files.children:
|
|
|
|
# if self.verify_extention_filename(child.filename.decode()):
|
|
|
|
# self.create_paste(uuid, child.contents, ltags, ltagsgalaxies, uuid+'_'+ str(n) , source)
|
|
|
|
# n = n + 1
|
|
|
|
# else:
|
|
|
|
# self.redis_logger.error("Error in module %s: bad extention"%(self.module_name))
|
|
|
|
# self.addError(uuid, 'Bad file extension: {}'.format(child.filename.decode()) )
|
|
|
|
|
|
|
|
# except FileNotFoundError:
|
|
|
|
# self.redis_logger.error("Error in module %s: file not found"%(self.module_name))
|
|
|
|
# self.addError(uuid, 'File not found: {}'.format(file_full_path), uuid )
|
2018-06-08 16:49:20 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
else:
|
|
|
|
self.abord_file_submission(uuid, f'File :{file_full_path} too large, over {SubmitPaste.FILE_MAX_SIZE} bytes')
|
|
|
|
else:
|
|
|
|
self.abord_file_submission(uuid, "Server Error, the archive can't be found")
|
2019-04-26 10:22:02 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
def _is_compressed_type(self, file_type):
|
|
|
|
"""
|
|
|
|
Check if file type is in the list of compressed file extensions format
|
|
|
|
"""
|
|
|
|
compressed_type = ['zip', 'gz', 'tar.gz']
|
2018-08-02 15:29:13 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
return file_type in compressed_type
|
2019-07-26 14:28:02 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
def remove_submit_uuid(self, uuid):
|
|
|
|
# save temp value on disk
|
|
|
|
self.r_serv_db.delete(f'{uuid}:ltags')
|
|
|
|
self.r_serv_db.delete(f'{uuid}:ltagsgalaxies')
|
|
|
|
self.r_serv_db.delete(f'{uuid}:paste_content')
|
|
|
|
self.r_serv_db.delete(f'{uuid}:isfile')
|
|
|
|
self.r_serv_db.delete(f'{uuid}:password')
|
|
|
|
self.r_serv_db.delete(f'{uuid}:source')
|
2018-06-08 16:49:20 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
self.r_serv_log_submit.expire(f'{uuid}:end', SubmitPaste.expire_time)
|
|
|
|
self.r_serv_log_submit.expire(f'{uuid}:processing', SubmitPaste.expire_time)
|
|
|
|
self.r_serv_log_submit.expire(f'{uuid}:nb_total', SubmitPaste.expire_time)
|
|
|
|
self.r_serv_log_submit.expire(f'{uuid}:nb_sucess', SubmitPaste.expire_time)
|
|
|
|
self.r_serv_log_submit.expire(f'{uuid}:nb_end', SubmitPaste.expire_time)
|
|
|
|
self.r_serv_log_submit.expire(f'{uuid}:error', SubmitPaste.expire_time)
|
|
|
|
self.r_serv_log_submit.expire(f'{uuid}:paste_submit_link', SubmitPaste.expire_time)
|
2018-06-08 16:49:20 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
# delete uuid
|
|
|
|
self.r_serv_db.srem('submitted:uuid', uuid)
|
|
|
|
self.redis_logger.debug(f'{uuid} all file submitted')
|
2021-05-28 17:37:46 +02:00
|
|
|
print(f'{uuid} all file submitted')
|
2018-06-08 16:49:20 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
def create_paste(self, uuid, paste_content, ltags, ltagsgalaxies, name, source=None):
|
2021-05-28 17:37:46 +02:00
|
|
|
# # TODO: Use Item create
|
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
result = False
|
2018-06-08 16:49:20 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
now = datetime.datetime.now()
|
|
|
|
source = source if source else 'submitted'
|
2021-05-28 17:37:46 +02:00
|
|
|
save_path = source + '/' + now.strftime("%Y") + '/' + now.strftime("%m") + '/' + now.strftime("%d") + '/submitted_' + name + '.gz'
|
2018-06-08 16:49:20 +02:00
|
|
|
|
2023-04-13 14:25:02 +02:00
|
|
|
full_path = os.path.join(ITEMS_FOLDER, save_path)
|
2018-06-08 16:49:20 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
self.redis_logger.debug(f'file path of the paste {full_path}')
|
2018-06-08 16:49:20 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
if not os.path.isfile(full_path):
|
|
|
|
# file not exists in AIL paste directory
|
|
|
|
self.redis_logger.debug(f"new paste {paste_content}")
|
2018-06-08 16:49:20 +02:00
|
|
|
|
2022-10-25 16:25:19 +02:00
|
|
|
gzip64encoded = self._compress_encode_content(paste_content, uuid)
|
2018-06-08 16:49:20 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
if gzip64encoded:
|
2018-06-05 16:58:04 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
# use relative path
|
|
|
|
rel_item_path = save_path.replace(self.PASTES_FOLDER, '', 1)
|
|
|
|
self.redis_logger.debug(f"relative path {rel_item_path}")
|
2019-04-26 10:04:07 +02:00
|
|
|
|
2023-10-11 12:06:01 +02:00
|
|
|
item = Item(rel_item_path)
|
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
# send paste to Global module
|
2023-10-11 12:06:01 +02:00
|
|
|
relay_message = f"submitted {gzip64encoded}"
|
|
|
|
self.add_message_to_queue(obj=item, message=relay_message)
|
2018-06-08 16:49:20 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
# add tags
|
|
|
|
for tag in ltags:
|
2022-09-01 14:04:00 +02:00
|
|
|
Tag.add_object_tag(tag, 'item', rel_item_path)
|
2018-06-08 16:49:20 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
for tag in ltagsgalaxies:
|
2022-09-01 14:04:00 +02:00
|
|
|
Tag.add_object_tag(tag, 'item', rel_item_path)
|
2018-06-05 16:58:04 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
self.r_serv_log_submit.incr(f'{uuid}:nb_end')
|
|
|
|
self.r_serv_log_submit.incr(f'{uuid}:nb_sucess')
|
2019-11-05 15:18:03 +01:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
if self.r_serv_log_submit.get(f'{uuid}:nb_end') == self.r_serv_log_submit.get(f'{uuid}:nb_total'):
|
|
|
|
self.r_serv_log_submit.set(f'{uuid}:end', 1)
|
2018-06-29 10:02:29 +02:00
|
|
|
|
2023-04-04 15:18:15 +02:00
|
|
|
self.redis_logger.debug(f' {rel_item_path} send to Mixer')
|
|
|
|
print(f' {rel_item_path} send to Mixer')
|
2021-04-28 15:24:33 +02:00
|
|
|
self.r_serv_log_submit.sadd(f'{uuid}:paste_submit_link', rel_item_path)
|
2018-06-05 16:58:04 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
curr_date = datetime.date.today()
|
|
|
|
self.redis_logger.debug("paste submitted")
|
|
|
|
else:
|
|
|
|
self.addError(uuid, f'File: {save_path} already exist in submitted pastes')
|
2018-06-05 16:58:04 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
return result
|
2019-11-05 15:18:03 +01:00
|
|
|
|
2022-10-25 16:25:19 +02:00
|
|
|
def _compress_encode_content(self, content, uuid):
|
2021-04-28 15:24:33 +02:00
|
|
|
gzip64encoded = None
|
|
|
|
try:
|
|
|
|
gzipencoded = gzip.compress(content)
|
|
|
|
gzip64encoded = base64.standard_b64encode(gzipencoded).decode()
|
|
|
|
except:
|
|
|
|
self.abord_file_submission(uuid, "file error")
|
|
|
|
return gzip64encoded
|
2018-06-05 16:58:04 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
def addError(self, uuid, errorMessage):
|
|
|
|
self.redis_logger.debug(errorMessage)
|
2021-05-28 17:37:46 +02:00
|
|
|
print(errorMessage)
|
2021-04-28 15:24:33 +02:00
|
|
|
error = self.r_serv_log_submit.get(f'{uuid}:error')
|
2022-10-25 16:25:19 +02:00
|
|
|
if error is not None:
|
2021-04-28 15:24:33 +02:00
|
|
|
self.r_serv_log_submit.set(f'{uuid}:error', error + '<br></br>' + errorMessage)
|
|
|
|
self.r_serv_log_submit.incr(f'{uuid}:nb_end')
|
2018-06-05 16:58:04 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
def abord_file_submission(self, uuid, errorMessage):
|
|
|
|
self.redis_logger.debug(f'abord {uuid}, {errorMessage}')
|
2018-06-08 16:49:20 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
self.addError(uuid, errorMessage)
|
|
|
|
self.r_serv_log_submit.set(f'{uuid}:end', 1)
|
|
|
|
curr_date = datetime.date.today()
|
|
|
|
self.remove_submit_uuid(uuid)
|
2018-06-08 16:49:20 +02:00
|
|
|
|
2021-05-28 17:37:46 +02:00
|
|
|
# # TODO: use Item function
|
2021-04-28 15:24:33 +02:00
|
|
|
def get_item_date(self, item_filename):
|
|
|
|
l_directory = item_filename.split('/')
|
|
|
|
return f'{l_directory[-4]}{l_directory[-3]}{l_directory[-2]}'
|
2018-06-05 16:58:04 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
def verify_extention_filename(self, filename):
|
|
|
|
if not '.' in filename:
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
file_type = filename.rsplit('.', 1)[1]
|
|
|
|
|
2022-10-25 16:25:19 +02:00
|
|
|
# txt file
|
2021-04-28 15:24:33 +02:00
|
|
|
if file_type in SubmitPaste.ALLOWED_EXTENSIONS:
|
|
|
|
return True
|
2018-06-08 16:49:20 +02:00
|
|
|
else:
|
2021-04-28 15:24:33 +02:00
|
|
|
return False
|
2018-06-05 16:58:04 +02:00
|
|
|
|
2021-04-28 15:24:33 +02:00
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
module = SubmitPaste()
|
|
|
|
module.run()
|