Skip to content

Commit 53bcd69

Browse files
committed
Implement TerraformConfig
1 parent 9123004 commit 53bcd69

File tree

11 files changed

+194
-88
lines changed

11 files changed

+194
-88
lines changed

.github/workflows/release.yml

+62
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,62 @@
1+
name: Release
2+
3+
on:
4+
push:
5+
tags:
6+
- '*.*.*'
7+
8+
jobs:
9+
build:
10+
strategy:
11+
matrix:
12+
os: [ ubuntu-latest, windows-latest, macos-latest ]
13+
python-version: [ '3.6', '3.7', '3.8', '3.9', '3.10' ]
14+
runs-on: ${{ matrix.os }}
15+
steps:
16+
- name: Check out repository code
17+
uses: actions/checkout@v2
18+
- name: Set up Python ${{ matrix.python-version }}
19+
uses: actions/setup-python@v2
20+
with:
21+
python-version: ${{ matrix.python-version }}
22+
- name: Install poetry
23+
run: |
24+
python -m pip install poetry
25+
- name: Fetch terraform repository
26+
run: |
27+
git submodule init
28+
git submodule update
29+
- name: Build distributions
30+
run: |
31+
poetry build -f wheel -vvv
32+
- name: Upload distribution artifacts
33+
uses: actions/upload-artifact@v3
34+
with:
35+
name: libterraform-dist
36+
path: dist
37+
publish:
38+
needs: [ build ]
39+
runs-on: macos-latest
40+
steps:
41+
- name: Checkout repository code
42+
uses: actions/checkout@v2
43+
- name: Set up Python 3.10
44+
uses: actions/setup-python@v2
45+
with:
46+
python-version: '3.10'
47+
- name: Download distribution artifacts
48+
uses: actions/download-artifact@v3
49+
with:
50+
name: libterraform-dist
51+
path: dist
52+
- name: Install poetry
53+
run: |
54+
python -m pip install poetry
55+
- name: Create Release
56+
uses: softprops/action-gh-release@v1
57+
- name: Publish to PyPI
58+
env:
59+
POETRY_PYPI_TOKEN_PYPI: ${{ secrets.PYPI_TOKEN }}
60+
run: |
61+
poetry config pypi-token.pypi $POETRY_PYPI_TOKEN_PYPI
62+
poetry publish

LICENSE.txt

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
MIT License
2+
3+
Copyright (c) 2017 Prodesire
4+
5+
Permission is hereby granted, free of charge, to any person obtaining a copy
6+
of this software and associated documentation files (the "Software"), to deal
7+
in the Software without restriction, including without limitation the rights
8+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9+
copies of the Software, and to permit persons to whom the Software is
10+
furnished to do so, subject to the following conditions:
11+
12+
The above copyright notice and this permission notice shall be included in all
13+
copies or substantial portions of the Software.
14+
15+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21+
SOFTWARE.

README.md

+9-5
Original file line numberDiff line numberDiff line change
@@ -1,19 +1,23 @@
1-
# py-terraform
1+
# Python libterraform
2+
3+
[![Release](https://github.com/Prodesire/py-libterraform/actions/workflows/release.yml/badge.svg)](https://github.com/Prodesire/py-libterraform/actions/workflows/release.yml)
4+
25
Python binding for Terraform.
36

7+
48
## Installation
59
```bash
6-
$ pip install py-terraform
10+
$ pip install libterraform
711
```
812

913
## Usage
10-
For now, only supply `load_config_dir` method which reads the .tf and .tf.json files in the given directory
14+
For now, only supply `TerraformConfig.load_config_dir` method which reads the .tf and .tf.json files in the given directory
1115
as config files and then combines these files into a single Module. This method returns `(mod, diags)`
1216
which are both dict, corresponding to the [*Module](https://github.com/hashicorp/terraform/blob/2a5420cb9acf8d5f058ad077dade80214486f1c4/internal/configs/module.go#L14)
1317
and [hcl.Diagnostic](https://github.com/hashicorp/hcl/blob/v2.11.1/diagnostic.go#L26) structures in Terraform respectively.
1418
```python
15-
>>> import py_terraform
16-
>>> mod, _ =py_terraform.load_config_dir('tests/config/sleep')
19+
>>> from libterraform import TerraformConfig
20+
>>> mod, _ =TerraformConfig.load_config_dir('tests/config/sleep')
1721
>>> mod['ManagedResources'].keys()
1822
dict_keys(['time_sleep.wait'])
1923
```

build.py

+11-10
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,14 @@
11
import os
22
import shutil
33
import subprocess
4+
import platform
45

5-
root = os.path.dirname(os.path.abspath(__file__))
6-
terraform_dirname = os.path.join(root, 'terraform')
7-
lib_filename = 'libterraform.so'
6+
7+
lib_filename = 'libterraform.dll' if platform.system() == 'Windows' else 'libterraform.so'
88
header_filename = 'libterraform.h'
99
tf_filename = 'libterraform.go'
10+
root = os.path.dirname(os.path.abspath(__file__))
11+
terraform_dirname = os.path.join(root, 'terraform')
1012
tf_path = os.path.join(root, tf_filename)
1113

1214

@@ -15,12 +17,12 @@ class BuildError(Exception):
1517

1618

1719
def build(setup_kwargs):
18-
'''
20+
"""
1921
This function is mandatory in order to build the extensions.
20-
'''
21-
if not os.path.exists(terraform_dirname):
22-
raise BuildError(f'The directory {terraform_dirname} not exists. '
23-
f'Please execute `git submodule update` to get it.')
22+
"""
23+
if not os.path.exists(os.path.join(terraform_dirname, '.git')):
24+
raise BuildError(f'The directory {terraform_dirname} not exists or init. '
25+
f'Please execute `git submodule init && git submodule update` to init it.')
2426

2527
target_tf_path = os.path.join(terraform_dirname, tf_filename)
2628
lib_path = os.path.join(terraform_dirname, lib_filename)
@@ -30,10 +32,9 @@ def build(setup_kwargs):
3032
print(' - Building libterraform')
3133
subprocess.check_call(
3234
['go', 'build', '-buildmode=c-shared', f'-o={lib_filename}', tf_filename],
33-
# shell=True,
3435
cwd=terraform_dirname
3536
)
36-
shutil.move(lib_path, os.path.join(root, 'py_terraform', lib_filename))
37+
shutil.move(lib_path, os.path.join(root, 'libterraform', lib_filename))
3738
finally:
3839
for path in (target_tf_path, header_path, lib_path):
3940
if os.path.exists(path):

libterraform.go

+10-4
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,10 @@ import (
1616
"encoding/json"
1717
)
1818

19+
// **********************************************
20+
// Config
21+
// **********************************************
22+
1923
// ShortModule is a container for a set of configuration constructs that are
2024
// evaluated within a common namespace.
2125
// Compared with module, there are fewer non-serializable fields.
@@ -62,15 +66,14 @@ func convertModule(mod *configs.Module) *ShortModule {
6266
return shortMod
6367
}
6468

65-
//export LoadConfigDir
66-
func LoadConfigDir(cPath *C.char) (cMod *C.char, cDiags *C.char, cError *C.char) {
69+
//export ConfigLoadConfigDir
70+
func ConfigLoadConfigDir(cPath *C.char) (cMod *C.char, cDiags *C.char, cError *C.char) {
6771
defer func() {
6872
recover()
6973
}()
7074

7175
parser := configs.NewParser(nil)
7276
path := C.GoString(cPath)
73-
//fmt.Println(path)
7477
mod, diags := parser.LoadConfigDir(path)
7578
modBytes, err := json.Marshal(convertModule(mod))
7679
if err != nil {
@@ -89,10 +92,13 @@ func LoadConfigDir(cPath *C.char) (cMod *C.char, cDiags *C.char, cError *C.char)
8992
cMod = C.CString(string(modBytes))
9093
cDiags = C.CString(string(diagsBytes))
9194
cError = C.CString("")
92-
//fmt.Println(string(modBytes), string(diagsBytes))
9395
return cMod, cDiags, cError
9496
}
9597

98+
// **********************************************
99+
// Utils
100+
// **********************************************
101+
96102
//export Free
97103
func Free(cString *int) {
98104
C.free(unsafe.Pointer(cString))

libterraform/__init__.py

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
import os
2+
import platform
3+
from ctypes import cdll, c_void_p
4+
5+
__version__ = '0.1.0'
6+
7+
root = os.path.dirname(os.path.abspath(__file__))
8+
_lib_filename = 'libterraform.dll' if platform.system() == 'Windows' else 'libterraform.so'
9+
_lib_tf = cdll.LoadLibrary(os.path.join(root, _lib_filename))
10+
11+
_free = _lib_tf.Free
12+
_free.argtypes = [c_void_p]
13+
14+
from .config import TerraformConfig
15+
16+
__all__ = [TerraformConfig]

libterraform/config.py

+46
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
import json
2+
from ctypes import *
3+
4+
from libterraform import _lib_tf, _free
5+
from libterraform.exceptions import LibTerraformError
6+
7+
8+
class LoadConfigDirResult(Structure):
9+
_fields_ = [("r0", c_void_p),
10+
("r1", c_void_p),
11+
("r2", c_void_p)]
12+
13+
14+
_load_config_dir = _lib_tf.ConfigLoadConfigDir
15+
_load_config_dir.argtypes = [c_char_p]
16+
_load_config_dir.restype = LoadConfigDirResult
17+
18+
19+
class TerraformConfig:
20+
@staticmethod
21+
def load_config_dir(path: str) -> (dict, dict):
22+
"""
23+
load_config_dir reads the .tf and .tf.json files in the given directory
24+
as config files and then combines these files into a single Module.
25+
26+
.tf files are parsed using the HCL native syntax while .tf.json files are
27+
parsed using the HCL JSON syntax.
28+
"""
29+
ret = _load_config_dir(path.encode('utf-8'))
30+
r_mod = cast(ret.r0, c_char_p).value
31+
_free(ret.r0)
32+
r_diags = cast(ret.r1, c_char_p).value
33+
_free(ret.r1)
34+
err = cast(ret.r2, c_char_p).value
35+
_free(ret.r2)
36+
37+
if err:
38+
raise LibTerraformError(err)
39+
if r_mod is None:
40+
msg = f'The given directory {path!r} does not exist at all or could not be opened for some reason.'
41+
raise LibTerraformError(msg)
42+
43+
mod = json.loads(r_mod)
44+
diags = json.loads(r_diags)
45+
46+
return mod, diags

libterraform/exceptions.py

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
class LibTerraformError(Exception):
2+
pass

py_terraform/__init__.py

-54
This file was deleted.

pyproject.toml

+8-7
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,13 @@
11
[tool.poetry]
2-
name = "py-terraform"
3-
version = "0.0.1"
4-
description = "Python binding for Terraform"
2+
name = "libterraform"
3+
version = "0.1.0"
4+
description = "Python binding for Terraform."
55
authors = ["Prodesire <[email protected]>"]
66
license = "MIT"
77
readme = "README.md"
8-
homepage = "https://github.com/Prodesire/py-terraform"
9-
repository = "https://github.com/Prodesire/py-terraform"
10-
keywords = ["py-terraform", "terraform"]
8+
homepage = "https://github.com/Prodesire/py-libterraform"
9+
repository = "https://github.com/Prodesire/py-libterraform"
10+
keywords = ["libterraform", "terraform"]
1111
classifiers = [
1212
"Intended Audience :: Developers",
1313
"License :: OSI Approved :: MIT License",
@@ -25,8 +25,9 @@ classifiers = [
2525
"Operating System :: Microsoft :: Windows"
2626
]
2727
packages = [
28-
{ include = "py_terraform/*" },
28+
{ include = "libterraform/*" },
2929
]
30+
include = ["libterraform/libterraform.so", "libterraform/libterraform.dll"]
3031

3132
[tool.poetry.dependencies]
3233
python = "^3.6"

tests/config/test_load_config_dir.py

+9-8
Original file line numberDiff line numberDiff line change
@@ -2,17 +2,18 @@
22

33
import pytest
44

5-
import py_terraform
5+
from libterraform import TerraformConfig
6+
from libterraform.exceptions import LibTerraformError
67

78
cur_dirname = os.path.dirname(os.path.abspath(__file__))
89
sleep_dirname = os.path.join(cur_dirname, 'sleep')
910

1011

11-
def test_load_config_dir():
12-
mod, diags = py_terraform.load_config_dir(sleep_dirname)
13-
assert 'time_sleep.wait' in mod['ManagedResources']
12+
class TestTerraformConfig:
13+
def test_load_config_dir(self):
14+
mod, diags = TerraformConfig.load_config_dir(sleep_dirname)
15+
assert 'time_sleep.wait' in mod['ManagedResources']
1416

15-
16-
def test_load_config_dir_no_exits():
17-
with pytest.raises(py_terraform.LibTerraformError):
18-
py_terraform.load_config_dir('not-exits')
17+
def test_load_config_dir_no_exits(self):
18+
with pytest.raises(LibTerraformError):
19+
TerraformConfig.load_config_dir('not-exits')

0 commit comments

Comments
 (0)