Skip to content

Commit e66075b

Browse files
Merge pull request #12 from RohithSurya/features/32100-config_run_compile
[#32100] Customize bash_run and bash_compile. Add support for multi cluster options
2 parents f70b9cb + 5390d81 commit e66075b

19 files changed

+1243
-256
lines changed

.github/workflows/pr.yaml

+1-1
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,7 @@ jobs:
2222
pull_number: context.issue.number
2323
})
2424
const isTitleValid = /^\[#\d+\] /.test(pr.data.title)
25-
const isDescriptionValid = /([Ff]ix(es|ed)?|[Cc]lose(s|d)?|[Rr]esolve(s|d)?|[Pp]art [Oo]f) \(.*\)\[.*\]/.test(pr.data.body)
25+
const isDescriptionValid = /([Ff]ix(es|ed)?|[Cc]lose(s|d)?|[Rr]esolve(s|d)?|[Pp]art [Oo]f) \[.*\]\(.*\)/.test(pr.data.body)
2626
if (isTitleValid && isDescriptionValid) {
2727
return
2828
}

.gitignore

+1-2
Original file line numberDiff line numberDiff line change
@@ -130,8 +130,7 @@ dmypy.json
130130
# Pyre type checker
131131
.pyre/
132132

133-
my_secret.py
134-
test_hpcc.py
133+
tests/my_secret.py
135134

136135
# Miscellaneous folders
137136
Dummy/

.pre-commit-config.yaml

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
repos:
2+
- repo: https://github.com/astral-sh/ruff-pre-commit
3+
rev: "v0.5.0"
4+
hooks:
5+
- id: ruff
6+
args: ["--fix"]
7+
- id: ruff-format

CONTRIBUTING.md

+30-5
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,26 @@ We value your interest in contributing to `PyHPCC.`
44

55
Thank you
66

7+
## Project Structure
8+
```
9+
.
10+
└── pyhpcc/
11+
├── .github # contains build, release, test and other gh actions
12+
├── docs/ # contains files for documentation
13+
├── examples/ # contains starter examples
14+
├── src/
15+
│ ├── pyhpcc/
16+
│ │ ├── handlers/ # contains thor and roxie handler
17+
│ │ └── models/ # contains classes auth, workunit submit
18+
│ └── tests/
19+
│ ├── models/
20+
│ ├── test_files/ # contains resource files needed for testing
21+
│ └── hanlders/
22+
├── pyproject.toml # Project config
23+
├── CONTRIBUTING.md
24+
└── README.md
25+
```
26+
727
## Set up the repository locally.
828
## Prerequisites
929
Before starting to develop, make sure you install the following software:
@@ -25,6 +45,15 @@ To install the dependencies, run the following command, which downloads the depe
2545
poetry install
2646
```
2747

48+
## How to run tests
49+
Since ecl client tools aren't installed in the GitHub runner, some tests are skipped in the github runner.
50+
51+
Some tests will fail if `ecl client tools` aren't installed.
52+
53+
```
54+
pytest run # Run in project root
55+
```
56+
2857
## Linting and Formatting
2958
PyHPCC uses [Ruff](https://docs.astral.sh/ruff/) as its formatter and linter.
3059

@@ -56,8 +85,4 @@ The base branch is the main repo's main branch.
5685
- PR name: copy-and-paste the relevant issue name and include the issue number in front in square brackets, e.g. `[#1020] Make bash_runcommand in WorkUnitSubmit class configurable `
5786
- PR description: mention the issue number in this format: Fixes #1020. Doing so will automatically close the related issue once the PR is merged.
5887
- Please Ensure that "Allow edits from maintainers" is ticked.
59-
- Please describe the changes you have made in your branch and how they resolve the issue.
60-
61-
62-
63-
88+
- Please describe the changes you have made in your branch and how they resolve the issue.

README.md

+3-2
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,8 @@ To use PyHPCC, you need these<br>
1717
1. [Python3](https://www.python.org/downloads/)
1818
2. [ECL Client Tools](https://hpccsystems.com/download/): Select your operating systems to download client tools
1919

20+
21+
2022
Download the latest stable build from releases in GitHub.<br>
2123

2224
``` bash
@@ -36,5 +38,4 @@ Contributions to PyHPCC are welcomed and encouraged.<br>
3638
For specific contribution guidelines, please take a look at [CONTRIBUTING.md](CONTRIBUTING.md).
3739

3840

39-
For more information about the package, please refer to the detailed documentation - https://upgraded-bassoon-daa9d010.pages.github.io/build/html/index.html
40-
41+
For more information about the package, please refer to the detailed documentation - https://upgraded-bassoon-daa9d010.pages.github.io/build/html/index.html

pyproject.toml

+1
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@ elementpath = "^3.0.2" # XPath selectors for XML Data structures
1919
[tool.poetry.group.dev.dependencies]
2020
pytest = "^7.1.3"
2121
pytest-mock = "^3.0.2"
22+
pre-commit="^3.7.1"
2223

2324
[tool.poetry.group.coverage.dependencies]
2425
coverage = "^7.5.3"

src/pyhpcc/command_config.py

+202
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,202 @@
1+
import logging
2+
3+
from pyhpcc.config import (
4+
CLUSTER_OPTION,
5+
COMPILE_OPTIONS,
6+
JOB_NAME_OPTION,
7+
LIMIT_OPTION,
8+
MASKED_PASSWORD,
9+
OUTPUT_FILE_OPTION,
10+
PASSWORD_OPTIONS,
11+
PORT_OPTION,
12+
RUN_AUTH_OPTIONS,
13+
RUN_OPTIONS,
14+
SERVER_OPTIONS,
15+
USER_OPTIONS,
16+
)
17+
from pyhpcc.errors import CompileConfigException, RunConfigException
18+
from pyhpcc.models.auth import Auth
19+
20+
log = logging.getLogger(__name__)
21+
22+
23+
class CompileConfig(object):
24+
"""
25+
Class for eclcc option configuration
26+
27+
Attributes:
28+
----------
29+
options:
30+
Dictionary of keys (option), values (value)
31+
32+
Methods:
33+
-------
34+
validate_options:
35+
Validate if the compiler options are supported or not
36+
37+
set_output_file:
38+
Set name of output file (default a.out if linking to
39+
40+
create_compile_bash_command:
41+
Generate the eclcc command for the given options and input_file
42+
43+
get_option:
44+
Retrieves the compile config option
45+
46+
47+
"""
48+
49+
def __init__(self, options: dict):
50+
self.options = options.copy()
51+
52+
def validate_options(self):
53+
"""Validate if the compiler options are supported or not"""
54+
invalid_options = []
55+
for option in self.options:
56+
if option not in COMPILE_OPTIONS and not (
57+
option.startswith("-R") or option.startswith("-f")
58+
):
59+
invalid_options.append(option)
60+
if len(invalid_options) > 0:
61+
raise CompileConfigException(str(invalid_options))
62+
63+
def set_output_file(self, output_file):
64+
"""Set name of output file (default a.out if linking to"""
65+
self.options[OUTPUT_FILE_OPTION] = output_file
66+
67+
def create_compile_bash_command(self, input_file):
68+
"""Generate the eclcc command for the given options and input_file"""
69+
self.validate_options()
70+
eclcc_command = "eclcc"
71+
for key, value in self.options.items():
72+
if value is bool:
73+
eclcc_command += f" {key}"
74+
else:
75+
eclcc_command += f" {key} {value}"
76+
eclcc_command = f"{eclcc_command} {input_file}"
77+
return eclcc_command
78+
79+
def get_option(self, option):
80+
"""Get the option available for the option"""
81+
return self.options[option]
82+
83+
84+
class RunConfig(object):
85+
"""
86+
Class for ecl run option configuration
87+
88+
Attributes:
89+
----------
90+
options:
91+
Dictionary of keys (option), values (value)
92+
93+
Methods:
94+
-------
95+
validate_options:
96+
Validate if the compiler options are supported or not
97+
98+
create_run_bash_command:
99+
Generate the ecl command for the given options and target_file
100+
101+
set_auth_params:
102+
Set the auth parameters from the auth object passed
103+
104+
set_target:
105+
Specify the job name for the workunit
106+
107+
set_limit:
108+
Sets the result limit for the query
109+
110+
set_server:
111+
Set IP of server running ecl services (eclwatch)
112+
113+
set_port:
114+
Set ECL services port
115+
116+
set_username:
117+
Set username for accessing ecl services
118+
119+
set_password:
120+
Set password for accessing ecl services
121+
122+
get_option:
123+
Retrieves the run config option
124+
125+
"""
126+
127+
def __init__(self, options: dict):
128+
self.options = options.copy()
129+
130+
def validate_options(self):
131+
"""Validate if the runtime options are supported or not"""
132+
invalid_options = set()
133+
for option in self.options:
134+
if option not in RUN_OPTIONS and not (
135+
option.startswith("-X") or option.startswith("-f")
136+
):
137+
invalid_options.add(option)
138+
if len(invalid_options) > 0:
139+
log.error("Entered invalid options %s", str(invalid_options))
140+
raise RunConfigException(
141+
f"Invalid options not supported by pyhpcc {str(invalid_options)}"
142+
)
143+
144+
def create_run_bash_command(self, target_file, password_mask=False):
145+
"""Generate the ecl command for the given options and target_file"""
146+
self.validate_options()
147+
ecl_command = "ecl run"
148+
params = ""
149+
for key, value in self.options.items():
150+
if value is bool:
151+
params += f" {key}"
152+
else:
153+
if key in PASSWORD_OPTIONS and password_mask:
154+
params += f" {key} {MASKED_PASSWORD}"
155+
else:
156+
params += f" {key} {value}"
157+
ecl_command = f"{ecl_command} {target_file}{params}"
158+
log.info(ecl_command)
159+
return ecl_command
160+
161+
def set_auth_params(self, auth: Auth):
162+
"""Set the auth parameters from the auth object passed"""
163+
for option in list(self.options):
164+
if option in RUN_AUTH_OPTIONS:
165+
log.warning(f"Overriding option {option} with Auth object parameters")
166+
del self.options[option]
167+
self.set_server(auth.ip)
168+
self.set_port(auth.port)
169+
self.set_username(auth.oauth[0])
170+
self.set_password(auth.oauth[1])
171+
172+
def set_target(self, target):
173+
"""Set the target"""
174+
self.options[CLUSTER_OPTION] = target
175+
176+
def set_job_name(self, job_name):
177+
"""Specify the job name for the workunit"""
178+
self.options[JOB_NAME_OPTION] = job_name
179+
180+
def set_limit(self, limit):
181+
"""Sets the result limit for the query"""
182+
self.options[LIMIT_OPTION] = limit
183+
184+
def set_server(self, server):
185+
"""Set IP of server running ecl services (eclwatch)"""
186+
self.options[SERVER_OPTIONS[0]] = server
187+
188+
def set_port(self, port):
189+
"""Set ECL services port"""
190+
self.options[PORT_OPTION] = port
191+
192+
def set_username(self, username):
193+
"""Set username for accessing ecl services"""
194+
self.options[USER_OPTIONS[0]] = username
195+
196+
def set_password(self, password):
197+
"""Set password for accessing ecl services"""
198+
self.options[PASSWORD_OPTIONS[0]] = password
199+
200+
def get_option(self, option):
201+
"""Get the option available for the option"""
202+
return self.options[option]

0 commit comments

Comments
 (0)