|
2 | 2 | This module provides input / output manipulations on streams / files
|
3 | 3 | """
|
4 | 4 |
|
5 |
| -from datetime import datetime |
6 |
| -import os |
7 | 5 | import io
|
8 | 6 | import json
|
| 7 | +import os |
9 | 8 | import shutil
|
10 |
| -from pathlib import Path |
| 9 | +import subprocess |
11 | 10 | import yaml
|
| 11 | + |
| 12 | +from datetime import datetime |
12 | 13 | from munch import Munch, munchify
|
| 14 | +from pathlib import Path |
13 | 15 |
|
14 | 16 |
|
15 | 17 | def get_events_from_index(elastic_client, index_name: str, rule_tag: str, time_after: datetime) -> list[Munch]:
|
@@ -129,17 +131,62 @@ def exec_command(container_name: str, command: str, param_value: str, resource:
|
129 | 131 | if command == 'chmod':
|
130 | 132 | os.chmod(path=resource, mode=int(param_value, base=8))
|
131 | 133 | elif command == 'chown':
|
132 |
| - uid_gid = param_value.split(':') |
133 |
| - if len(uid_gid) != 2: |
134 |
| - raise Exception( |
135 |
| - "User and group parameter shall be separated by ':' ") |
136 |
| - shutil.chown(path=resource, user=uid_gid[0], group=uid_gid[1]) |
| 134 | + try: |
| 135 | + uid, gid = param_value.split(':') |
| 136 | + except ValueError as exc: |
| 137 | + raise Exception("User and group parameter shall be separated by ':' ") from exc |
| 138 | + |
| 139 | + FsClient.add_users_to_node([uid, gid], in_place=True) |
| 140 | + shutil.chown(path=resource, user=uid, group=gid) |
137 | 141 | elif command == 'unlink':
|
138 | 142 | if not Path(param_value).is_dir():
|
139 | 143 | Path(param_value).unlink()
|
140 | 144 | else:
|
141 | 145 | raise Exception(
|
142 | 146 | f"Command '{command}' still not implemented in test framework")
|
| 147 | + |
| 148 | + @staticmethod |
| 149 | + def add_users_to_node(users: list, in_place: bool): |
| 150 | + """ |
| 151 | + This function creates the given users along with groups with the |
| 152 | + same name, on the local container as well the host node. |
| 153 | + @param users: List of users to create. |
| 154 | + @param in_place: Whether host node configuration files should be modified in-place or overwritten. |
| 155 | + @return: None |
| 156 | + """ |
| 157 | + if in_place: |
| 158 | + host_users_file = Path('/hostfs/etc/passwd') |
| 159 | + host_groups_file = Path('/hostfs/etc/group') |
| 160 | + |
| 161 | + temp_etc = Path('/tmp/etc') |
| 162 | + temp_etc.mkdir(parents=True, exist_ok=True) |
| 163 | + |
| 164 | + temp_users_file = temp_etc / 'passwd' |
| 165 | + temp_groups_file = temp_etc / 'group' |
| 166 | + |
| 167 | + shutil.copyfile(host_users_file, temp_users_file) |
| 168 | + shutil.copyfile(host_groups_file, temp_groups_file) |
| 169 | + |
| 170 | + for user in users: |
| 171 | + # These commands fail silently for users/groups that exist. |
| 172 | + subprocess.run(['groupadd', user, '-P', '/tmp'], capture_output=True) |
| 173 | + subprocess.run(['useradd', user, '-g', user, '-P', '/tmp'], capture_output=True) |
| 174 | + subprocess.run(['useradd', user], capture_output=True) # For container to get around chmod check. |
| 175 | + |
| 176 | + FsClient.in_place_copy(temp_users_file, host_users_file) |
| 177 | + FsClient.in_place_copy(temp_groups_file, host_groups_file) |
| 178 | + |
| 179 | + else: |
| 180 | + # TODO(yashtewari): Implement this section which simulates a "normal" user flow |
| 181 | + # where useradd command overwrites passwd and group files, |
| 182 | + # as part of tests for: https://github.com/elastic/cloudbeat/issues/235 |
| 183 | + pass |
| 184 | + |
| 185 | + @staticmethod |
| 186 | + def in_place_copy(source, destination): |
| 187 | + with open(source, 'r') as sf, open(destination, 'w') as df: |
| 188 | + for line in sf: |
| 189 | + df.write(line) |
143 | 190 |
|
144 | 191 | @staticmethod
|
145 | 192 | def edit_process_file(container_name: str, dictionary, resource: str):
|
|
0 commit comments