Skip to content

Commit f4297dc

Browse files
committed
Merge branch 'master' of https://github.com/shuffle/python-apps
2 parents e98c46c + 31f8d79 commit f4297dc

File tree

9 files changed

+791
-26
lines changed

9 files changed

+791
-26
lines changed
Lines changed: 27 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
# Base our app image off of the WALKOFF App SDK image
2+
FROM frikky/shuffle:app_sdk as base
3+
4+
# We're going to stage away all of the bloat from the build tools so lets create a builder stage
5+
FROM base as builder
6+
7+
# Install all alpine build tools needed for our pip installs
8+
RUN apk --no-cache add --update alpine-sdk libffi libffi-dev musl-dev openssl-dev git
9+
10+
# Install all of our pip packages in a single directory that we can copy to our base image later
11+
RUN mkdir /install
12+
WORKDIR /install
13+
COPY requirements.txt /requirements.txt
14+
RUN pip install --no-cache-dir --prefix="/install" -r /requirements.txt
15+
16+
# Switch back to our base image and copy in all of our built packages and source code
17+
FROM base
18+
COPY --from=builder /install /usr/local
19+
COPY src /app
20+
21+
# Install any binary dependencies needed in our final image
22+
# RUN apk --no-cache add --update my_binary_dependency
23+
RUN apk --no-cache add jq git curl
24+
25+
# Finally, lets run our app!
26+
WORKDIR /app
27+
CMD ["python", "app.py", "--log-level", "DEBUG"]

shuffle-tools-fork/1.0.0/api.yaml

Lines changed: 34 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
---
2+
app_version: 1.0.0
3+
name: Shuffle Tools Fork
4+
description: A tool app for Shuffle. Gives access to most missing features along with Liquid.
5+
tags:
6+
- Testing
7+
- Shuffle
8+
categories:
9+
- Other
10+
contact_info:
11+
name: "@frikkylikeme"
12+
url: https://shuffler.io
13+
14+
actions:
15+
- name: execute_python
16+
description: Runs python with the data input. Any prints will be returned.
17+
parameters:
18+
- name: code
19+
description: The code to run. Can be a file ID from within Shuffle.
20+
required: true
21+
multiline: true
22+
example: print("hello world")
23+
schema:
24+
type: string
25+
- name: packages
26+
description: The code to run. Can be a file ID from within Shuffle.
27+
required: true
28+
multiline: true
29+
example: pandas\nnumpy\nmatplotlib
30+
schema:
31+
type: string
32+
33+
large_image: 
34+
# yamllint disable-line rule:line-length
Lines changed: 15 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
version: '3.4'
2+
services:
3+
shuffle-tools-fork:
4+
build:
5+
context: .
6+
dockerfile: Dockerfile
7+
# image: walkoff_registry:5000/walkoff_app_HelloWorld-v1-0
8+
deploy:
9+
mode: replicated
10+
replicas: 10
11+
restart_policy:
12+
condition: none
13+
restart: "no"
14+
secrets:
15+
- secret1
Lines changed: 11 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
ioc_finder==7.2.1
2+
py7zr==0.11.3
3+
rarfile==4.0
4+
pyminizip==0.2.4
5+
requests==2.25.1
6+
xmltodict==0.11.0
7+
json2xml==5.0.5
8+
ipaddress==1.0.23
9+
google.auth==1.23.0
10+
paramiko==3.1.0
11+
shufflepy

shuffle-tools-fork/1.0.0/run.sh

Lines changed: 11 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
# Build testing
2+
NAME=frikky/shuffle:shuffle-tools-fork_1.0.0
3+
docker rmi $NAME --force
4+
docker build . -t frikky/shuffle:shuffle-tools-fork_1.0.0
5+
6+
# Run testing
7+
#docker run -e SHUFFLE_SWARM_CONFIG=run -e SHUFFLE_APP_EXPOSED_PORT=33334 frikky/shuffle:shuffle-tools_1.1.0
8+
echo $NAME
9+
#docker service create --env SHUFFLE_SWARM_CONFIG=run --env SHUFFLE_APP_EXPOSED_PORT=33334 $NAME
10+
11+
#cat walkoff_app_sdk/app_base.py #cat walkoff_app_sdk/app_sdk.py
Lines changed: 184 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,184 @@
1+
import hmac
2+
import datetime
3+
import json
4+
import time
5+
import markupsafe
6+
import os
7+
import re
8+
import subprocess
9+
import tempfile
10+
import zipfile
11+
import base64
12+
import importlib
13+
import ipaddress
14+
import hashlib
15+
import shufflepy
16+
from io import StringIO
17+
from contextlib import redirect_stdout
18+
import random
19+
import string
20+
21+
import xmltodict
22+
from json2xml import json2xml
23+
from json2xml.utils import readfromstring
24+
25+
from ioc_finder import find_iocs
26+
from dateutil.parser import parse as dateutil_parser
27+
from google.auth import crypt
28+
from google.auth import jwt
29+
30+
import py7zr
31+
import pyminizip
32+
import rarfile
33+
import requests
34+
import tarfile
35+
import binascii
36+
import struct
37+
38+
import paramiko
39+
import concurrent.futures
40+
import multiprocessing
41+
42+
from pip._internal import main as pip_main
43+
from pip._internal.commands.show import search_packages_info
44+
45+
from walkoff_app_sdk.app_base import AppBase
46+
47+
class Tools(AppBase):
48+
__version__ = "1.2.0"
49+
app_name = (
50+
"Shuffle Tools Fork" # this needs to match "name" in api.yaml for WALKOFF to work
51+
)
52+
53+
def __init__(self, redis, logger, console_logger=None):
54+
"""
55+
Each app should have this __init__ to set up Redis and logging.
56+
:param redis:
57+
:param logger:
58+
:param console_logger:
59+
"""
60+
super().__init__(redis, logger, console_logger)
61+
62+
def dynamic_import(package_name: str):
63+
"""Import a package and return the module"""
64+
return importlib.import_module(package_name.split('==')[0].split('>=')[0].split('<=')[0].split('>')[0].split('<')[0])
65+
66+
67+
def get_missing_packages(required_packages: list) -> list:
68+
"""
69+
Returns a list of packages that aren't currently installed.
70+
71+
Args:
72+
required_packages: List of package names (can include version specs)
73+
74+
Returns:
75+
List of package names that aren't installed
76+
"""
77+
missing = []
78+
for package in required_packages:
79+
# Remove version specifiers if present (e.g., 'pandas>=1.0.0' -> 'pandas')
80+
package_name = package.split('==')[0].split('>=')[0].split('<=')[0].split('>')[0].split('<')[0].strip()
81+
82+
# Check if package exists in environment
83+
if not list(search_packages_info([package_name])):
84+
missing.append(package)
85+
86+
return missing
87+
88+
def install_packages(self, packages=[]) -> None:
89+
"""
90+
Install Python packages using pip's Python interface.
91+
92+
Args:
93+
packages: List of package names to install
94+
"""
95+
96+
packages_not_found = self.get_missing_packages(packages)
97+
98+
for package in packages_not_found:
99+
try:
100+
pip_main(['install', package])
101+
print(f"Successfully installed {package}")
102+
except Exception as e:
103+
print(f"Failed to install {package}: {str(e)}")
104+
105+
def execute_python(self, code, packages) -> dict:
106+
if os.getenv("SHUFFLE_ALLOW_PACKAGE_INSTALL") == "true":
107+
allow_package_install = True
108+
109+
packages = packages.split("\n") if packages else []
110+
111+
if packages:
112+
if allow_package_install:
113+
self.install_packages(packages)
114+
self.dynamic_import(packages)
115+
116+
if len(code) == 36 and "-" in code:
117+
filedata = self.get_file(code)
118+
if filedata["success"] == False:
119+
return {
120+
"success": False,
121+
"message": f"Failed to get file for ID {code}",
122+
}
123+
124+
if ".py" not in filedata["filename"]:
125+
return {
126+
"success": False,
127+
"message": f"Filename needs to contain .py",
128+
}
129+
130+
131+
# Write the code to a file
132+
# 1. Take the data into a file
133+
# 2. Subprocess execute file?
134+
try:
135+
f = StringIO()
136+
def custom_print(*args, **kwargs):
137+
return print(*args, file=f, **kwargs)
138+
139+
#with redirect_stdout(f): # just in case
140+
# Add globals in it too
141+
globals_copy = globals().copy()
142+
globals_copy["print"] = custom_print
143+
144+
# Add self to globals_copy
145+
for key, value in locals().copy().items():
146+
if key not in globals_copy:
147+
globals_copy[key] = value
148+
149+
globals_copy["self"] = self
150+
151+
exec(code, globals_copy)
152+
153+
s = f.getvalue()
154+
f.close() # why: https://www.youtube.com/watch?v=6SA6S9Ca5-U
155+
156+
#try:
157+
# s = s.encode("utf-8")
158+
#except Exception as e:
159+
160+
try:
161+
return {
162+
"success": True,
163+
"message": json.loads(s.strip()),
164+
}
165+
except Exception as e:
166+
try:
167+
return {
168+
"success": True,
169+
"message": s.strip(),
170+
}
171+
except Exception as e:
172+
return {
173+
"success": True,
174+
"message": s,
175+
}
176+
177+
except Exception as e:
178+
return {
179+
"success": False,
180+
"message": f"exception: {e}",
181+
}
182+
183+
if __name__ == "__main__":
184+
Tools.run()

0 commit comments

Comments
 (0)