-
Notifications
You must be signed in to change notification settings - Fork 21
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
feat: Add Fastly Edge SDK #723
Changes from all commits
00ecc9a
809e1cd
0914d82
be7aa07
40f05ab
77485da
aff265d
7d47d12
cfddcdc
6079a15
f9b9561
7fd2d96
ce436a2
3717d13
e96311c
9c7c966
91819c9
584eef5
08cb672
53891b1
1fb5431
9c6d3dd
ed6764e
4282427
a422de4
6271e7c
dff0e69
7c1bc03
2eef68d
b41b07b
dda0ce5
389c52e
a73b761
0111854
3a67b2b
36d8136
c567c1a
7cd6bff
c90f23a
fa71eb6
8cb7fd0
67bde4a
58c7cd2
7047f2f
3eea8c2
ccd9d42
01f9784
98bb0e6
bf72203
fd91a2a
ecaad7b
7beddc1
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,24 @@ | ||
name: sdk/fastly | ||
|
||
on: | ||
push: | ||
branches: [main, 'feat/**'] | ||
paths-ignore: | ||
- '**.md' #Do not need to run CI for markdown changes. | ||
pull_request: | ||
branches: [main, 'feat/**'] | ||
paths-ignore: | ||
- '**.md' | ||
|
||
jobs: | ||
build-test-fastly: | ||
runs-on: ubuntu-latest | ||
steps: | ||
- uses: actions/checkout@v4 | ||
- uses: actions/setup-node@v4 | ||
- id: shared | ||
name: Shared CI Steps | ||
uses: ./actions/ci | ||
with: | ||
workspace_name: '@launchdarkly/fastly-server-sdk' | ||
workspace_path: packages/sdk/fastly |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,61 @@ | ||
# LaunchDarkly SDK for Fastly | ||
|
||
The LaunchDarkly SDK for Fastly is designed for use in [Fastly Compute Platform](https://www.fastly.com/documentation/guides/compute/). It follows the server-side LaunchDarkly model for multi-user contexts. It is not intended for use in desktop and embedded systems applications. | ||
|
||
## Install | ||
|
||
```shell | ||
# npm | ||
npm i @launchdarkly/fastly-server-sdk | ||
|
||
# yarn | ||
yarn add @launchdarkly/fastly-server-sdk | ||
``` | ||
|
||
## Usage notes | ||
|
||
- The SDK must be initialized and used when processing requests, not during build-time initialization. | ||
- The SDK caches all KV data during initialization to reduce the number of backend requests needed to fetch KV data. This means changes to feature flags or segments will not be picked up during the lifecycle of a single request instance. | ||
- Events should flushed using the [`waitUntil()` method](https://js-compute-reference-docs.edgecompute.app/docs/globals/FetchEvent/prototype/waitUntil). | ||
|
||
## Quickstart | ||
|
||
See the full [example app](https://github.com/launchdarkly/js-core/tree/main/packages/sdk/fastly/example). | ||
|
||
## Developing this SDK | ||
|
||
```shell | ||
# at js-core repo root | ||
yarn && yarn build && cd packages/sdk/fastly | ||
|
||
# run tests | ||
yarn test | ||
``` | ||
|
||
## Verifying SDK build provenance with the SLSA framework | ||
|
||
LaunchDarkly uses the [SLSA framework](https://slsa.dev/spec/v1.0/about) (Supply-chain Levels for Software Artifacts) to help developers make their supply chain more secure by ensuring the authenticity and build integrity of our published SDK packages. To learn more, see the [provenance guide](PROVENANCE.md). | ||
|
||
## About LaunchDarkly | ||
|
||
- LaunchDarkly is a continuous delivery platform that provides feature flags as a service and allows developers to iterate quickly and safely. We allow you to easily flag your features and manage them from the LaunchDarkly dashboard. With LaunchDarkly, you can: | ||
- Roll out a new feature to a subset of your users (like a group of users who opt-in to a beta tester group), gathering feedback and bug reports from real-world use cases. | ||
- Gradually roll out a feature to an increasing percentage of users, and track the effect that the feature has on key metrics (for instance, how likely is a user to complete a purchase if they have feature A versus feature B?). | ||
- Turn off a feature that you realize is causing performance problems in production, without needing to re-deploy, or even restart the application with a changed configuration file. | ||
- Grant access to certain features based on user attributes, like payment plan (eg: users on the 'gold' plan get access to more features than users in the 'silver' plan). | ||
- Disable parts of your application to facilitate maintenance, without taking everything offline. | ||
- LaunchDarkly provides feature flag SDKs for a wide variety of languages and technologies. Read [our documentation](https://docs.launchdarkly.com/sdk) for a complete list. | ||
- Explore LaunchDarkly | ||
- [launchdarkly.com](https://www.launchdarkly.com/ 'LaunchDarkly Main Website') for more information | ||
- [docs.launchdarkly.com](https://docs.launchdarkly.com/ 'LaunchDarkly Documentation') for our documentation and SDK reference guides | ||
- [apidocs.launchdarkly.com](https://apidocs.launchdarkly.com/ 'LaunchDarkly API Documentation') for our API documentation | ||
- [blog.launchdarkly.com](https://blog.launchdarkly.com/ 'LaunchDarkly Blog Documentation') for the latest product updates | ||
|
||
[sdk-fastly-ci-badge]: https://github.com/launchdarkly/js-core/actions/workflows/fastly.yml/badge.svg | ||
[sdk-fastly-ci]: https://github.com/launchdarkly/js-core/actions/workflows/fastly.yml | ||
[sdk-fastly-npm-badge]: https://img.shields.io/npm/v/@launchdarkly/fastly-server-sdk.svg?style=flat-square | ||
[sdk-fastly-npm-link]: https://www.npmjs.com/package/@launchdarkly/fastly-server-sdk | ||
[sdk-fastly-ghp-badge]: https://img.shields.io/static/v1?label=GitHub+Pages&message=API+reference&color=00add8 | ||
[sdk-fastly-ghp-link]: https://launchdarkly.github.io/js-core/packages/sdk/fastly/docs/ | ||
[sdk-fastly-dm-badge]: https://img.shields.io/npm/dm/@launchdarkly/fastly-server-sdk.svg?style=flat-square | ||
[sdk-fastly-dt-badge]: https://img.shields.io/npm/dt/@launchdarkly/fastly-server-sdk.svg?style=flat-square |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
export const KVStore = jest.fn().mockImplementation(() => ({ | ||
get: jest.fn(), | ||
put: jest.fn(), | ||
delete: jest.fn(), | ||
getMulti: jest.fn(), | ||
putMulti: jest.fn(), | ||
deleteMulti: jest.fn(), | ||
})); |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,130 @@ | ||
import { AsyncStoreFacade, LDFeatureStore } from '@launchdarkly/js-server-sdk-common'; | ||
|
||
import { EdgeFeatureStore } from '../../src/api/EdgeFeatureStore'; | ||
import mockEdgeProvider from '../utils/mockEdgeProvider'; | ||
import * as testData from './testData.json'; | ||
|
||
describe('EdgeFeatureStore', () => { | ||
const clientSideId = 'client-side-id'; | ||
const kvKey = `LD-Env-${clientSideId}`; | ||
const mockLogger = { | ||
error: jest.fn(), | ||
warn: jest.fn(), | ||
info: jest.fn(), | ||
debug: jest.fn(), | ||
}; | ||
const mockGet = mockEdgeProvider.get as jest.Mock; | ||
let featureStore: LDFeatureStore; | ||
let asyncFeatureStore: AsyncStoreFacade; | ||
|
||
beforeEach(() => { | ||
mockGet.mockImplementation(() => Promise.resolve(JSON.stringify(testData))); | ||
featureStore = new EdgeFeatureStore( | ||
mockEdgeProvider, | ||
clientSideId, | ||
'MockEdgeProvider', | ||
mockLogger, | ||
); | ||
asyncFeatureStore = new AsyncStoreFacade(featureStore); | ||
}); | ||
|
||
afterEach(() => { | ||
jest.resetAllMocks(); | ||
}); | ||
|
||
describe('get', () => { | ||
it('can retrieve valid flag', async () => { | ||
const flag = await asyncFeatureStore.get({ namespace: 'features' }, 'testFlag1'); | ||
|
||
expect(mockGet).toHaveBeenCalledWith(kvKey); | ||
expect(flag).toMatchObject(testData.flags.testFlag1); | ||
}); | ||
|
||
it('returns undefined for invalid flag key', async () => { | ||
const flag = await asyncFeatureStore.get({ namespace: 'features' }, 'invalid'); | ||
|
||
expect(flag).toBeUndefined(); | ||
}); | ||
|
||
it('can retrieve valid segment', async () => { | ||
const segment = await asyncFeatureStore.get({ namespace: 'segments' }, 'testSegment1'); | ||
|
||
expect(mockGet).toHaveBeenCalledWith(kvKey); | ||
expect(segment).toMatchObject(testData.segments.testSegment1); | ||
}); | ||
|
||
it('returns undefined for invalid segment key', async () => { | ||
const segment = await asyncFeatureStore.get({ namespace: 'segments' }, 'invalid'); | ||
|
||
expect(segment).toBeUndefined(); | ||
}); | ||
|
||
it('returns null for invalid kv key', async () => { | ||
mockGet.mockImplementation(() => Promise.resolve(null)); | ||
const flag = await asyncFeatureStore.get({ namespace: 'features' }, 'testFlag1'); | ||
|
||
expect(flag).toBeNull(); | ||
}); | ||
}); | ||
|
||
describe('all', () => { | ||
it('can retrieve all flags', async () => { | ||
const flags = await asyncFeatureStore.all({ namespace: 'features' }); | ||
|
||
expect(mockGet).toHaveBeenCalledWith(kvKey); | ||
expect(flags).toMatchObject(testData.flags); | ||
}); | ||
|
||
it('can retrieve all segments', async () => { | ||
const segment = await asyncFeatureStore.all({ namespace: 'segments' }); | ||
|
||
expect(mockGet).toHaveBeenCalledWith(kvKey); | ||
expect(segment).toMatchObject(testData.segments); | ||
}); | ||
|
||
it('returns empty object for invalid DataKind', async () => { | ||
const flag = await asyncFeatureStore.all({ namespace: 'InvalidDataKind' }); | ||
|
||
expect(flag).toEqual({}); | ||
}); | ||
|
||
it('returns empty object for invalid kv key', async () => { | ||
mockGet.mockImplementation(() => Promise.resolve(null)); | ||
const segment = await asyncFeatureStore.all({ namespace: 'segments' }); | ||
|
||
expect(segment).toEqual({}); | ||
}); | ||
}); | ||
|
||
describe('initialized', () => { | ||
it('returns true when initialized', async () => { | ||
const isInitialized = await asyncFeatureStore.initialized(); | ||
|
||
expect(mockGet).toHaveBeenCalledWith(kvKey); | ||
expect(isInitialized).toBeTruthy(); | ||
}); | ||
|
||
it('returns false when not initialized', async () => { | ||
mockGet.mockImplementation(() => Promise.resolve(null)); | ||
const isInitialized = await asyncFeatureStore.initialized(); | ||
|
||
expect(mockGet).toHaveBeenCalledWith(kvKey); | ||
expect(isInitialized).toBeFalsy(); | ||
}); | ||
}); | ||
|
||
describe('init & getDescription', () => { | ||
it('can initialize', (done) => { | ||
const cb = jest.fn(() => { | ||
done(); | ||
}); | ||
featureStore.init(testData, cb); | ||
}); | ||
|
||
it('can retrieve description', async () => { | ||
const description = featureStore.getDescription?.(); | ||
|
||
expect(description).toEqual('MockEdgeProvider'); | ||
}); | ||
}); | ||
}); |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,68 @@ | ||
import { internal } from '@launchdarkly/js-server-sdk-common'; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. All tests in this directory were copied over from |
||
|
||
import LDClient from '../../src/api/LDClient'; | ||
import { createBasicPlatform } from '../createBasicPlatform'; | ||
|
||
jest.mock('@launchdarkly/js-sdk-common', () => { | ||
const actual = jest.requireActual('@launchdarkly/js-sdk-common'); | ||
return { | ||
...actual, | ||
...{ | ||
internal: { | ||
...actual.internal, | ||
DiagnosticsManager: jest.fn(), | ||
EventProcessor: jest.fn(), | ||
}, | ||
}, | ||
}; | ||
}); | ||
|
||
let mockEventProcessor = internal.EventProcessor as jest.Mock; | ||
beforeEach(() => { | ||
mockEventProcessor = internal.EventProcessor as jest.Mock; | ||
mockEventProcessor.mockClear(); | ||
}); | ||
|
||
describe('Edge LDClient', () => { | ||
it('uses clientSideID endpoints', async () => { | ||
const client = new LDClient('client-side-id', createBasicPlatform().info, { | ||
sendEvents: true, | ||
eventsBackendName: 'launchdarkly', | ||
}); | ||
await client.waitForInitialization({ timeout: 10 }); | ||
const passedConfig = mockEventProcessor.mock.calls[0][0]; | ||
|
||
expect(passedConfig).toMatchObject({ | ||
sendEvents: true, | ||
serviceEndpoints: { | ||
includeAuthorizationHeader: false, | ||
analyticsEventPath: '/events/bulk/client-side-id', | ||
diagnosticEventPath: '/events/diagnostic/client-side-id', | ||
events: 'https://events.launchdarkly.com', | ||
polling: 'https://sdk.launchdarkly.com', | ||
streaming: 'https://stream.launchdarkly.com', | ||
}, | ||
}); | ||
}); | ||
it('uses custom eventsUri when specified', async () => { | ||
const client = new LDClient('client-side-id', createBasicPlatform().info, { | ||
sendEvents: true, | ||
eventsBackendName: 'launchdarkly', | ||
eventsUri: 'https://custom-base-uri.launchdarkly.com', | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I added the option to set a custom There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @kinyoklion is there a way to exclude that validation? |
||
}); | ||
await client.waitForInitialization({ timeout: 10 }); | ||
const passedConfig = mockEventProcessor.mock.calls[0][0]; | ||
|
||
expect(passedConfig).toMatchObject({ | ||
sendEvents: true, | ||
serviceEndpoints: { | ||
includeAuthorizationHeader: false, | ||
analyticsEventPath: '/events/bulk/client-side-id', | ||
diagnosticEventPath: '/events/diagnostic/client-side-id', | ||
events: 'https://custom-base-uri.launchdarkly.com', | ||
polling: 'https://custom-base-uri.launchdarkly.com', | ||
streaming: 'https://stream.launchdarkly.com', | ||
}, | ||
}); | ||
}); | ||
}); |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,17 @@ | ||
import { BasicLogger } from '@launchdarkly/js-server-sdk-common'; | ||
|
||
import createOptions, { defaultOptions } from '../../src/api/createOptions'; | ||
|
||
describe('createOptions', () => { | ||
test('default options', () => { | ||
expect(createOptions({})).toEqual(defaultOptions); | ||
}); | ||
|
||
test('override logger', () => { | ||
const logger = new BasicLogger({ name: 'test' }); | ||
expect(createOptions({ logger })).toEqual({ | ||
...defaultOptions, | ||
logger, | ||
}); | ||
}); | ||
}); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We would still expect the mockGet call, like in other tests right? I think we should add that assertion back, but just confirm that it yields "undefined" as the final result.