-
Notifications
You must be signed in to change notification settings - Fork 27
Using with Serverless Webpack #26
Comments
@RishitKedia No problem 😄 I think you can use the webpack file-loader to achieve the integration of the Internally, the compile will put the json file (named with a hash) into the build output and will automatically rewrite the require to fetch the file. So you do not need to even think about how to reference it, and you do not need to have any error handling - as the require will be resolved at compile time. If you have to use the dynamic require by serverless-secrets, this approach may not work. |
A different approach might be to use webpack-plugin-copy to copy |
@HyperBrain I tried both the From what I've read, it seems that Webpack does not like dynamic I like your idea of the Thanks for all your help, Frank. 🙏 |
@RishitKedia Can you try with the PR I just opened? The trick is, to add the configuration file with the file-loader now and call If that works, the maintainers here should have a look at it, as it enables the secrets plugin for all serverless-webpack users. |
@HyperBrain Thank you for taking out the time to work on this. 👍 I'm still getting the warning for the dynamic Also, if I remove the dynamic It points to this line, even though my |
@RishitKedia I will have a look at that with a local test project later. The For the |
Hi @HyperBrain and @RishitKedia, Sorry for the long delay. Been super busy with other projects and finally got back to some maintenance. @RishitKedia Have you managed to get @HyperBrain's solution working correctly? If so, is everything necessary documented in his PR or was there more that you needed to do? I'm working on getting this package out of beta ASAP and I'd like to integrate this if possible before that release. If timing doesn't work out, we can always bring it in under 3.1.0. Thanks to both of you for your assistance! |
Hey, @azurelogic! 👋 I tried a few things but wasn't able to get @HyperBrain's PR working correctly since I'm not very experienced with Webpack. Waiting for @HyperBrain to work his magic on his PR once he gets time to test it locally. 😉 Thanks for considering this! 🙏 |
I'm testing right now locally. As soon as I succeed setting up a working configuration I'll write it here so that anyone can test if it really works in a real-world scenario. |
Hi there. I got a setup working. In general, I think this is the right way to go, but it may need some additional tweaks and/or small changes (e.g. add the hints to the README with this PR, etc.). The solution is straight forward without the need of any workarounds or complex configurations. IntroductionI created a small test project that uses a stored SSM parameter to verify that it works. The parameter is named 'testsecret' and has the value 'myvalue' for testing purposes. Upfront, this is the output of the test lambda as reference, and you see that it fetched the variable correctly from SSM and printed the value 'myvalue'.
I documented all steps that have to be made below, including the service, the webpack config and a sample handler. The setup (how to configure the handler and service)The serviceThere are only a few requirements for the service definition. See the sample yaml snippet below to see how things have to be set. You have to add the webpack and the secrets plugin to your Define the Currently It is important to use # serverless.yml
...
plugins:
- serverless-secrets
- serverless-webpack
...
provider:
...
environmentSecrets:
API_KEY: 'testsecret'
...
custom:
webpackIncludeModules:
forceExclude:
- aws-sdk The Webpack configurationBecause serverless-secrets is a development dependency and the actual client, that is used in our production handlers is contained in there, the easiest way to get (only) the client in, is to have it just bundled. module.exports = {
entry: slsw.lib.entries,
target: 'node',
externals: [
nodeExternals({
whitelist: [
"serverless-secrets/client"
]
})
],
...
} The Lambda handlerWith the changes in the PR we now can initialize the secrets plugin client with a preloaded I commented the new behavior in the sample handler below. // Given: a secret named 'testsecret' is stored in SSM with value 'myvalue'
// Given: an environmentSecret named 'API_KEY' exists with a value of 'testsecret'
// This require will be removed by webpack, because it bundles the client now.
const secrets = require('serverless-secrets/client');
// We use the new (PR) init function here, to initialize the secrets plugin client with the
// preloaded pregenerated JSON. With this require, webpack will automatically bundle the JSON
// and removes the needed dynamic require that would fail.
// This has to be a static path. The file is located in the root of the service. So you have to
// specify the proper backwards relative path from your handler directory.
secrets.init(require('./.serverless-secrets.json'));
module.exports.handler = function (event, context, callback) {
const secretsPromise = secrets.load();
secretsPromise.then(() => {
// process.env.API_KEY now contains 'myvalue'
console.log(process.env.API_KEY);
callback(null, { statusCode: 200 });
}).catch(err => {
// handle errors here
callback(null, { statusCode: 500, body: err.message });
});
}; The output from the very top is exactly from this handler. The package / deploymentYou can check with
The critical webpack warning emitted can be ignored, because with the PR the init function will use the argument given to init to initialize the configuration.
The function is compiled in a way that keeps the OR and thus never runs into the previously failing code part because the JSON is bundled and init is called from the handler at the very start. Please let me know, if it works for you, of if there are any additional issues that have to be solved. I will try to use it in one of our bigger projects too, to have some more reliable test results. |
@azurelogic While implementing and testing my PR I found a few things that might be improved. One of them is e.g. that the plugin introduces aws-sdk as dependency (which is by nature and due to the client which of course needs the SDK to access the SSM parameter store). But having it as production dependency of the plugin leads to the fact, that it will be a production dependency of the service (the user code) too. This is imo not very good, because every AWS service will try to exclude the SDK because it is already available on AWS Lambda. In my opinion it would be better if it was a peer dependency in the plugin, so that the Serverless project has to add it (but would just add it to its development dependencies to satisfy the peer dependency constraint). Otherwise the user has to take care of the exclusion by himself (either with Serverless or with the webpack plugin). |
@HyperBrain Hey, Frank, thanks for such a detailed explanation. 💯 But I'm still not able to make it work. 😭
You mentioned that it could be a small configuration issue with the JSON and the {
test: /\.serverless-secrets.json$/,
use: [{
loader: 'file-loader',
options: {
name: '[name].[ext]'
}
}]
}
The bundled file is very huge (4 MB when used with Yarn Workspaces) compared to the same project when not used with Yarn Workspaces (25 kB). It bundles all the dependencies even when using I've also reduced my project to the bare minimal, but I'm still getting the issues. 🤔 |
I did not use the file-loader at all. Only the things I mentioned in the description. So the the json is just a plain require - nothing else. Remove the whole file-loader rule from the config as it is not needed at all. Regarding YARN: serverless-webpack currently completely relies on npm and uses that. I'm sure that's the reason why it does not package the minimum dependencies. Did you set the forceExclude "aws-sdk" and switch to sls-webpack 4.0.0? The 4MB reminds me somehow of the size of the AWS SDK. |
This is the complete webpack config I used for testing:
|
The error goes away, and a similar one crops up pointing to the same line:
Do you want me to create a new issue for this on And, yup, I did |
Ok. That means for some reason your |
... and yes. Feel free to create a feature request that demands the optional use of yarn instead of npm for the packaging. |
Yup, this is the order of the plugins: plugins:
- serverless-secrets
- serverless-webpack
- serverless-offline And, |
I googled a bit and found out that this variable is set online only (in the AWS lambda environment). |
@azurelogic Can you confirm that? |
Ah, damn, I've been testing both online and offline, but this time I only tested offline. Yes, it works when deployed to AWS, finally! 😜 So, it's a bug in Thanks for your time, Frank. I really appreciate it. 🙏 |
😄 Thanks for testing. So I think the PR is finally working, but I have to adapt the README accordingly to have it finished. Maybe I'll just commit my comment from above as section there. |
@HyperBrain I made some assumptions about |
@HyperBrain This issue started because of the dynamic require which really was a hack to not have to do fs.readFileSync followed by a JSON.parse. I've switched to that route now to avoid similar issues in the future. I don't know if that simplifies or complicates integrating with serverless-webpack. Thoughts? As for the _HANDLER problem, I'm looking into that next. I know that AWS sets it. Just need to find a solid solution to that problem. |
The PR with the init() solved the problem for serverless-webpack. The problem with webpack is, that any dedicated read from the client will require that the file is present. |
After this change I've updated @HyperBrain #26 (comment) a little bit. The Webpack configurationI've added plugins: [
new WebpackPluginCopy([
{ from: '.serverless-secrets.json' }
])
] The Lambda handlerIt's safe to remove this line now secrets.init(require('./.serverless-secrets.json')); The package / deploymentNo more warnings here 🙏 ============================ That's it. |
Hey there! 👋
This is great! Thank you for writing this plugin. 🙏
Well, I'm not able to get past something when using
Serverless Secrets
withServerless Webpack
.I get the following warning from Webpack when bundling my code:
6:16-77 Critical dependency: the request of a dependency is an expression
It's due to the way the
.serverless-secrets.json
config file is required in this line:serverless-secrets/client/index.js
Line 6 in 6cf4af4
And, I get the following error when I run my function:
Error: Cannot find module "."
It's because this is what my
handler.js
contains for the previous line after bundling:Any thoughts on how this can be fixed? 😰
Thanks for your time! 😃
/cc @HyperBrain [Sorry to bother you, but maybe you can help us with this] 😛
The text was updated successfully, but these errors were encountered: