diff --git a/README.md b/README.md index bb5cc083..9be7c9f9 100644 --- a/README.md +++ b/README.md @@ -64,6 +64,7 @@ Currently, we support the following languages/platforms: | [React Web (Browser)](./flipt-client-react) | WASM | N/A | N/A | | [Flutter/Dart](./flipt-client-dart) | FFI | ✅ | ❌ | | [C#](./flipt-client-csharp) | FFI | ✅ | ❌ | +| [Swift](./flipt-client-swift) | FFI | N/A | N/A | Documentation for each client can be found in the README of that client's directory. @@ -78,7 +79,6 @@ Languages we are planning to support: Languages we would like to support but lack expertise in: 1. [Kotlin](https://github.com/flipt-io/flipt-client-sdks/issues/264) -1. [Swift](https://github.com/flipt-io/flipt-client-sdks/issues/263) 1. [React Native](https://github.com/flipt-io/flipt-client-sdks/issues/345) Want to see a client in a language we don't support? [Open an issue](https://github.com/flipt-io/flipt-client-sdks/issues/new?assignees=&labels=new-language&projects=&template=new_language.yml) and let us know! diff --git a/flipt-client-swift/README.md b/flipt-client-swift/README.md new file mode 100644 index 00000000..3995518e --- /dev/null +++ b/flipt-client-swift/README.md @@ -0,0 +1,115 @@ +# Flipt Client Swift + +[![GitHub Release](https://img.shields.io/github/v/release/flipt-io/flipt-client-sdks?filter=flipt-client-swift-*)](https://github.com/flipt-io/flipt-client-sdks/releases) + +The `flipt-client-swift` library contains the Swift source code for the Flipt [client-side evaluation](https://www.flipt.io/docs/integration/client) client. + +## Installation + +### Swift Package Manager + +Add the following to your `Package.swift` file: + +```swift +dependencies: [ + .package(url: "https://github.com/flipt-io/flipt-client-swift.git", from: "0.1.0") +] +``` + +Or add it directly through Xcode: +1. File > Add Package Dependencies +2. Enter package URL: `https://github.com/flipt-io/flipt-client-swift` + +## How Does It Work? + +The `flipt-client-swift` library is a wrapper around the [flipt-engine-ffi](https://github.com/flipt-io/flipt-client-sdks/tree/main/flipt-engine-ffi) library. + +All evaluation happens within the SDK, using the shared library built from the [flipt-engine-ffi](https://github.com/flipt-io/flipt-client-sdks/tree/main/flipt-engine-ffi) library. + +Because the evaluation happens within the SDK, the SDKs can be used in environments where the Flipt server is not available or reachable after the initial data is fetched. + +## Data Fetching + +Upon instantiation, the `flipt-client-swift` library will fetch the flag state from the Flipt server and store it in memory. This means that the first time you use the SDK, it will make a request to the Flipt server. + +### Polling (Default) + +By default, the SDK will poll the Flipt server for new flag state at a regular interval. This interval can be configured using the `updateInterval` option when constructing a client. The default interval is 120 seconds. + +### Streaming (Flipt Cloud Only) + +[Flipt Cloud](https://flipt.io/cloud) users can use the `streaming` fetch method to stream flag state changes from the Flipt server to the SDK. + +When in streaming mode, the SDK will connect to the Flipt server and open a persistent connection that will remain open until the client is closed. The SDK will then receive flag state changes in real-time. + +## Supported Platforms + +This SDK currently supports the following platforms/architectures: + +- iOS arm64 +- iOS Simulator arm64 +- macOS arm64 + +## Usage + +```swift +import FliptClient + +// Initialize the client +let client = try FliptClient( + namespace: "default", + url: "http://localhost:8080", + authentication: .clientToken("your-token"), + updateInterval: 120, + fetchMode: .polling +) + +// Evaluate a boolean flag +let boolResult = try client.evaluateBoolean( + flagKey: "my-flag", + entityId: "user-123", + context: ["key": "value"] +) +print("Flag enabled: \(boolResult.enabled)") + +// Evaluate a variant flag +let variantResult = try client.evaluateVariant( + flagKey: "my-variant-flag", + entityId: "user-123", + context: ["key": "value"] +) +print("Variant key: \(variantResult.variantKey)") + +// Don't forget to close the client when you're done +defer { + client.close() +} +``` + +### Authentication + +The `FliptClient` supports the following authentication strategies: + +- No Authentication (default) +- [Client Token Authentication](https://docs.flipt.io/authentication/using-tokens) +- [JWT Authentication](https://docs.flipt.io/authentication/using-jwts) + +## Memory Management + +The engine that is allocated on the Rust side to compute evaluations for flag state will not be properly deallocated unless you call the `close()` method on a `FliptClient` instance. + +**Please be sure to do this to avoid leaking memory!** + +```swift +defer { + client.close() +} +``` + +## Contributing + +Contributions are welcome! Please feel free to open an issue or submit a Pull Request. + +## License + +This project is licensed under the [MIT License](LICENSE). diff --git a/release/release.py b/release/release.py index 023285e3..e009dda0 100644 --- a/release/release.py +++ b/release/release.py @@ -2,7 +2,7 @@ from semver import VersionInfo from prompt_toolkit.shortcuts import checkboxlist_dialog, radiolist_dialog from colorama import init, Fore, Style -from sdks import GoSDK, JavaSDK, JavaScriptSDK, RubySDK, PythonSDK, DartSDK +from sdks import GoSDK, JavaSDK, JavaScriptSDK, RubySDK, PythonSDK, DartSDK, SwiftSDK, CSharpSDK from sdks.base import SDK, MuslSupportSDK # Initialize colorama @@ -19,6 +19,8 @@ def get_sdk(name: str, path: str) -> SDK: "flipt-client-dart": DartSDK, "flipt-client-python": PythonSDK, "flipt-client-ruby": RubySDK, + "flipt-client-swift": SwiftSDK, + "flipt-client-csharp": CSharpSDK, } return sdk_classes[name](name, path) @@ -43,6 +45,8 @@ def update_sdk_versions(bump_type="patch", sdks_to_update=None): "flipt-client-dart", "flipt-client-python", "flipt-client-ruby", + "flipt-client-swift", + "flipt-client-csharp", ] sdk_dirs = sdks_to_update if sdks_to_update else all_sdk_dirs @@ -97,6 +101,8 @@ def get_sdk_selection(all_sdk_dirs): "flipt-client-dart": "Dart", "flipt-client-python": "Python", "flipt-client-ruby": "Ruby", + "flipt-client-swift": "Swift", + "flipt-client-csharp": "C#", } selected_sdks = checkboxlist_dialog( @@ -149,6 +155,8 @@ def main(): "flipt-client-dart", "flipt-client-python", "flipt-client-ruby", + "flipt-client-swift", + "flipt-client-csharp", ] selected_sdks = get_sdk_selection(all_sdk_dirs) diff --git a/release/sdks/__init__.py b/release/sdks/__init__.py index 5357572e..440efdee 100644 --- a/release/sdks/__init__.py +++ b/release/sdks/__init__.py @@ -4,3 +4,5 @@ from .ruby import RubySDK from .python import PythonSDK from .dart import DartSDK +from .swift import SwiftSDK +from .csharp import CSharpSDK diff --git a/release/sdks/csharp.py b/release/sdks/csharp.py new file mode 100644 index 00000000..37fa33eb --- /dev/null +++ b/release/sdks/csharp.py @@ -0,0 +1,32 @@ +import os +import re +from .base import SDK + + +class CSharpSDK(SDK): + def get_current_version(self): + return self._get_version_from_file("src/FliptClient/FliptClient.csproj") + + def _get_version_from_file(self, filename): + with open(os.path.join(self.path, filename), "r") as f: + content = f.read() + # 0.0.1 + match = re.search(r"([\d.]+)", content) + if match: + return match.group(1) + raise ValueError(f"Version not found in {filename}") + + def update_version(self, new_version): + self._update_version_in_file("src/FliptClient/FliptClient.csproj", new_version) + + def _update_version_in_file(self, filename, new_version): + file_path = os.path.join(self.path, filename) + with open(file_path, "r") as f: + content = f.read() + + updated_content = re.sub( + r"([\d.]+)", f"{new_version}", content + ) + + with open(file_path, "w") as f: + f.write(updated_content) diff --git a/release/sdks/swift.py b/release/sdks/swift.py new file mode 100644 index 00000000..4e56f336 --- /dev/null +++ b/release/sdks/swift.py @@ -0,0 +1,29 @@ +import subprocess +from colorama import Fore, Style +from .base import SDK + + +class SwiftSDK(SDK): + def get_current_version(self) -> str: + return self._get_version_from_tag(f"{self.name}-v*") + + def _get_version_from_tag(self, tag_pattern: str) -> str: + try: + result = subprocess.run( + ["git", "describe", "--tags", "--abbrev=0", "--match", tag_pattern], + capture_output=True, + text=True, + check=True, + ) + latest_tag = result.stdout.strip() + return latest_tag.split("-v")[-1] + except subprocess.CalledProcessError: + print( + f"{Fore.YELLOW}Warning: No tags found for {tag_pattern}. Using 0.0.0 as the base version.{Style.RESET_ALL}" + ) + return "0.0.0" + + def update_version(self, new_version: str): + print( + f"{Fore.YELLOW}Note: Version for {self.name} is managed via Git tags. No files updated.{Style.RESET_ALL}" + ) \ No newline at end of file