Skip to content

Commit

Permalink
Moving Flask Out of Core Stix-Shifter (#1730)
Browse files Browse the repository at this point in the history
Signed-off-by: DerekRushton <[email protected]>
  • Loading branch information
DerekRushton authored Aug 29, 2024
1 parent de7ed6d commit 8fff6ef
Show file tree
Hide file tree
Showing 4 changed files with 104 additions and 82 deletions.
19 changes: 14 additions & 5 deletions docs/adapter-guide/develop-stix-adapter.md
Original file line number Diff line number Diff line change
Expand Up @@ -94,18 +94,27 @@ If the data source is synchronous, you must include `set_async(False)` in the co

#### Testing a new connector using the proxy host

Work on a new stix-shifter connector occurs after the project has been forked and cloned into a local development environment. Stix-shifter contains a **proxy** connector that facilitates a remote instance of the project calling out to a local instance. While in development, a new connector's working branch can be tested in any project using the stix-shifter library without first merging into the master branch on Github. A host is run on the local instance from the CLI. When a `proxy` data source is passed to the remote instance of stix-shifter, the real connection attributes (data source type, host, and port contained in the options) are passed onto the local instance of stix-shifter running the proxy host. The host will then use the new connector and return results back to the remote stix-shifter instance.
The Proxyhost server routing provides a python file with all of the required mappings to start a flask server that can be used to allow communication through the proxy module. The flask server is run on the local instance from the CLI. When a `proxy` data source is passed to the remote instance of stix-shifter, the real connection attributes (data source type, host, and port contained in the options) are passed onto the local instance of stix-shifter running the proxy host. The host will then use the new connector and return results back to the remote stix-shifter instance.

Open a terminal and navigate to your local stix-shifter directory. Run the host with the following command:
##### Requirements
1. stix-shifter-utils must be installed.
2. flask must be installed.
3. You need to be able to access the proxyhost_server_routing.py from your CLI.

##### Running the tool

To start a flask server with the proxy host routing map you can run the following command.

```
python main.py host "<STIX Identity Object>" "<Host IP address>:<Host Port>"
flask --app 'proxyhost_server_routing:start_proxyhost_flask_server({"type": "identity","id": "identity--f431f809-377b-45e0-aa1c-6a4751cae5ff","name": "Bundle","identity_class": "events"})' run'
```

As an example:
Running this command starts up the server on localhost and on port 5000.

To start the server up the same way as it was done previously (through the stix-shifter CLI)

```
python main.py host '{"type": "identity","id": "identity--f431f809-377b-45e0-aa1c-6a4751cae5ff","name": "Bundle","identity_class": "events"}' "192.168.122.83:5000"
flask --app 'proxyhost_server_routing:start_proxyhost_flask_server({"type": "identity","id": "identity--f431f809-377b-45e0-aa1c-6a4751cae5ff","name": "Bundle","identity_class": "events"})' run -h 127.0.0.1 -p 5001 --cert "/Users/StixShifter/stix-shifter/cert.pem" --key "/Users/StixShifter/stix-shifter/key.pem"
```

##### Calling the proxy host
Expand Down
75 changes: 0 additions & 75 deletions stix_shifter/scripts/stix_shifter.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,20 +3,17 @@
import json
import time
import importlib
from flask import Flask
import logging
import copy
from stix_shifter.stix_translation import stix_translation
from stix_shifter.stix_transmission import stix_transmission
from stix_shifter_utils.utils.proxy_host import ProxyHost
from stix_shifter_utils.utils.module_discovery import process_dialects, modules_list
from stix_shifter_utils.utils import logger as utils_logger
from stix_shifter_utils.utils.logger import exception_to_string

TRANSLATE = 'translate'
TRANSMIT = 'transmit'
EXECUTE = 'execute'
HOST = 'host'
MAPPING = 'mapping'
MODULES = 'modules'
CONFIGS = 'configs'
Expand Down Expand Up @@ -170,28 +167,6 @@ def main():
execute_parser.add_argument('-r', '--results', type=int, default=10,
help='Maximum number of returned results (default 10)')

host_parser = parent_subparsers.add_parser(HOST, help='Host a local query service, for testing and development')
host_parser.add_argument(
'data_source',
type=str,
help='STIX Identity object for the data source'
)
host_parser.add_argument(
'host_address',
type=str,
help='Proxy Host:Port'
)
host_parser.add_argument(
'ssl_cert',
type=str,
help='SSL certificate filename'
)
host_parser.add_argument(
'ssl_key',
type=str,
help='SSL key filename'
)

args = parent_parser.parse_args()

help_and_exit = args.command is None
Expand Down Expand Up @@ -243,56 +218,6 @@ def main():
if help_and_exit:
parent_parser.print_help(sys.stderr)
sys.exit(1)
elif args.command == HOST:
# Host means to start a local web service for STIX shifter, to use in combination with the proxy data source
# module. This combination allows one to run and debug their stix-shifter code locally, while interacting with
# it inside a service provider such as IBM Security Connect
app = Flask("stix-shifter")

@app.route('/transform_query', methods=['POST'])
def transform_query():
host = ProxyHost()
return host.transform_query()

@app.route('/translate_results', methods=['POST'])
def translate_results():
data_source_identity_object = args.data_source
host = ProxyHost()
return host.translate_results(data_source_identity_object)

@app.route('/create_query_connection', methods=['POST'])
def create_query_connection():
host = ProxyHost()
return host.create_query_connection()

@app.route('/create_status_connection', methods=['POST'])
def create_status_connection():
host = ProxyHost()
return host.create_status_connection()

@app.route('/create_results_connection', methods=['POST'])
def create_results_connection():
host = ProxyHost()
return host.create_results_connection()

@app.route('/delete_query_connection', methods=['POST'])
def delete_query_connection():
host = ProxyHost()
return host.delete_query_connection()

@app.route('/ping', methods=['POST'])
def ping_connection():
host = ProxyHost()
return host.ping_connection()

@app.route('/is_async', methods=['POST'])
def is_async():
host = ProxyHost()
return host.is_async()

host_address = args.host_address.split(":")
app.run(debug=True, port=int(host_address[1]), host=host_address[0], ssl_context=(args.ssl_cert, args.ssl_key))

elif args.command == EXECUTE:
# Execute means take the STIX SCO pattern as input, execute query, and return STIX as output

Expand Down
37 changes: 37 additions & 0 deletions stix_shifter_tools/External_Tools.MD
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
Stix Shifter Tools
===============
These are development and testing tools that can be used to assist with developing STIX Shifter. These tools are used externally to the core of STIX Shifter. The intended purpose behind this folder is to hold useful scripts and code that should not be shipped with the stix-shifter package. Included below will be a list of these tools, including their purpose and how to use them.

# Proxy Host Server Routing
The Proxyhost server routing provides a python file with all of the required mappings to start a flask server that can be used to allow communication through the proxy module. The flask server is run on the local instance from the CLI. When a `proxy` data source is passed to the remote instance of stix-shifter, the real connection attributes (data source type, host, and port contained in the options) are passed onto the local instance of stix-shifter running the proxy host. The host will then use the new connector and return results back to the remote stix-shifter instance.

# Requirements
1. stix-shifter-utils must be installed.
2. flask must be installed.
3. You need to be able to access the proxyhost_server_routing.py from your CLI.

# Running the tool

To start a flask server with the proxy host routing map you can run the following command.

```
flask --app 'proxyhost_server_routing:start_proxyhost_flask_server({"type": "identity","id": "identity--f431f809-377b-45e0-aa1c-6a4751cae5ff","name": "Bundle","identity_class": "events"})' run'
```

Running this command starts up the server on localhost and on port 5000.

To start the server up the same way as it was done previously (through the stix-shifter CLI)

```
flask --app 'proxyhost_server_routing:start_proxyhost_flask_server({"type": "identity","id": "identity--f431f809-377b-45e0-aa1c-6a4751cae5ff","name": "Bundle","identity_class": "events"})' run -h 127.0.0.1 -p 5001 --cert "/Users/StixShifter/stix-shifter/cert.pem" --key "/Users/StixShifter/stix-shifter/key.pem"
```

# Calling the proxy host

Each of the translate and transmit CLI commands outlined in the stix-shifter overview can be used to call the proxy host.

As an example:

```
python main.py transmit proxy '{"options": {"proxy_host": "127.0.0.1", "proxy_port": 5000, "destination": {"connection": {"options": {"result_limit": 10000, "time_range": 5, "timeout": 30}, "host": "<HOST>", "port": <PORT>, "type": "qradar"}, "configuration": {"auth": { "SEC": "<SEC TOKEN>"} } } }}' '{}' ping
```
Original file line number Diff line number Diff line change
@@ -1,12 +1,63 @@
import json
from flask import Flask, request

from stix_shifter.stix_translation import stix_translation
from stix_shifter.stix_transmission import stix_transmission
from stix_shifter_utils.utils import logger
from flask import request
import json

# Start a local web service for STIX shifter, to use in combination with the proxy data source
# module. This combination allows one to run and debug their stix-shifter code locally, while interacting with
# it inside a service provider such as IBM Security Connect

def start_proxyhost_flask_server(data_source):
app = Flask("stix-shifter")

class ProxyHost():
@app.route('/transform_query', methods=['POST'])
def transform_query():
host = ProxyHost()
return host.transform_query()

@app.route('/translate_results', methods=['POST'])
def translate_results():
data_source_identity_object = data_source
host = ProxyHost()
return host.translate_results(data_source_identity_object)

@app.route('/create_query_connection', methods=['POST'])
def create_query_connection():
host = ProxyHost()
return host.create_query_connection()

@app.route('/create_status_connection', methods=['POST'])
def create_status_connection():
host = ProxyHost()
return host.create_status_connection()

@app.route('/create_results_connection', methods=['POST'])
def create_results_connection():
host = ProxyHost()
return host.create_results_connection()

@app.route('/delete_query_connection', methods=['POST'])
def delete_query_connection():
host = ProxyHost()
return host.delete_query_connection()

@app.route('/ping', methods=['POST'])
def ping_connection():
host = ProxyHost()
return host.ping_connection()

@app.route('/is_async', methods=['POST'])
def is_async():
host = ProxyHost()
return host.is_async()

return app

class ProxyHost():
def __init__(self):
self.logger = logger.set_logger(__name__)
self.request_args = request.get_json(force=True)
Expand Down Expand Up @@ -69,4 +120,4 @@ async def ping_connection(self):
def is_async(self):
transmission_module = self.connection['type'].lower()
transmission = stix_transmission.StixTransmission(transmission_module, self.connection, self.configuration)
return "{}".format(transmission.is_async())
return "{}".format(transmission.is_async())

0 comments on commit 8fff6ef

Please sign in to comment.