1. Introduction to Proxies
Hi, I'm Will Johnston, a developer advocate on WP Engine's DevRel team. Today, I'll discuss using proxies or the JavaScript Proxy API for state management.
♪♪ Hi, I'm Will Johnston. I'm a developer advocate on WP Engine's DevRel team, and I'm here today to talk to you about how you might use proxies or the JavaScript Proxy API for state management. Proxies, if you're unfamiliar, are a way that you can observe some of the underlying operations on objects within JavaScript. So if you're familiar with object.observe, which doesn't exist anymore, but that was kind of the precursor to proxies. And if you're also familiar with property descriptors or object.defineProperty, that's a similar thing to proxies. Proxies just enable a little bit more full features.
2. Introduction to Proxies (continued)
I have been writing code since I was about 10 years old. I got interested in proxies to understand what's happening underneath. Proxy methods like get and set traps are simple and easy to grasp. Reflect is used to forward operations onto the original object and avoid inheritance issues. Let's create a basic demo using proxy to observe changes to an object. We intercept get and set operations and log them. We can use the proxy like a normal object.
A little bit about myself. I have been writing code since I was about 10 years old. In high school, I got a job doing some web programming with Python, and I really fell in love with software at that time and software development, and I've been a constant learner since then. I spent most of my career working on the web and back-end, and over the past few years, probably six to eight years, I've been working mostly with Node in the JavaScript and TypeScript ecosystem.
So, I really got interested in proxies because I've done a fair amount of .NET and C Sharp code, and in C Sharp, you have a concept of getters and setters on an object, and it allows you to stealthily poke into different operations on that object, and proxies function in a similar manner but for JavaScript. So, I got really interested into proxies to try to figure out how I can make that work because I really love when you can just use objects as you normally would, but underneath, you can understand exactly what's happening.
So, there are plenty of proxy methods, and proxy methods are used to set up traps, so you can observe some of the inherent ability for JavaScript to be performing different functions, so that's what a trap is. We're going to use the get and set traps today because those are simple and easy to grasp, and I'm not trying to go too far in depth with the short time that we have.
Something to know, when you're using proxy, you really also need to be using reflect, so the reflect object is used to forward all the operations onto the original object, so if you think of proxy is in front and you're listening to the set method, when somebody tries to set, you want to use the reflect object to eventually forward that operation on and actually set the value on that object. The reason you do this is related to avoiding this in inheritance issues, so if you create a proxy and then you try to inherit from it and you don't use reflect, you're going to get some wonky functionality where you're actually changing the original object, not the inherited object.
So let's just start with a basic demo, let's get a state object up using proxy to observe changes to that object, so we will, I like to create a proxy handler, it's the easiest way, I know that I'm going to eventually have to create a new object, a new proxy, and we will create an object and our handler is gonna be our proxy handler, so we know that our proxy handler is going to be a type proxy handler and that's where we have our get and our set methods, so let's create those now, target property and set target property value. So the get method just returns the target, the property at a certain target and the set method sets the value of a property on a target, so in here, we'll say getting property value value to be in here, we will say setting property oops and we will use the reflect API and this is how we can, it just mimics the proxy API directly. And allows us to not have to worry about what actually happens underneath, we just want to poke in where we care about it and observe the change. There's one additional piece we care about here and that is called the receiver and we want to add that in there. The receiver is going to provide the reflect object with that proper context. So here we have our get, our set, we are just intercepting these operations and logging, pretty simple.
For this, let's create on our state, a person, we'll say first name, Will. And let's go ahead and just add a last name too. Johnston, all right. And then, in order to actually see what's going on here, we will have to console.log, state.person.firstname. So what we expect here is it's going to log, you know, getting person on here. You might expect it to log getting firstname as well. We'll see whether it does in a second. Spoiler alert, it does not. And I'm just going to say here, and I'll call William. So we'll try to set the first name and that's good. All right. So note, when we create this proxy, we can just use state like a normal object. We don't have to care. So if you had, typically you'll use a library that might be using proxy underneath.
3. Observing Properties and Creating Caches
You end up with an object that you use as just a plain old JavaScript object. And that's what's really nice about proxy is it's very stealthy. In order to recursively observe all of the properties on this object, we need to keep track of the properties and the path from the original target. We also need to keep track of whether we've already set a proxy on an object. Let's create a path cache and a proxy cache. In a real-world scenario, we would have to do checks on the value to avoid creating a proxy on a primitive object. If the value is not a string, we get the path and augment it.
You end up with an object that you use as just a plain old JavaScript object. You set values, you get values, and you don't even know if there's a proxy underneath. And that's what's really nice about proxy is it's very stealthy.
So let's see what this does here. Node... Oh, receiver is not defined. Let's add receiver in here. Boy. All right. GettingPerson will settingPerson. So no gettingPerson here. It didn't say gettingPerson, and it didn't say gettingFirstName, and then in here we setPerson. It didn't say settingFirstName, and it didn't say settingLastName.
In order to do that, we would have to have a way to recursively observe all of the properties on this object. So let's look at what that entails here. In order to do that, we would need to know to keep track of all of the different properties, and really the path from the original target to those properties. The other thing we will need to keep track of is whether we've already set a proxy on an object already, because if you try to create a proxy on something that is already a proxy, you'll get an error, and we don't want to deal with that. We need to create a couple of caches.
Let's do a path cache. We'll use a weak map. And a proxy cache. We'll also use a weak map. In here, this is fine. First, let's get the value. Now, in a real-world scenario, we would have to do a lot of checks on this value because you don't want to end up creating a proxy on a primitive object, so like strings, numbers, boolean, et cetera. But for the case of this demo, we're just gonna check type of value is a string. If it's a string, we'll just return value and be done. If it's not a string, we need to get the path. So, if this target is already in our path, which it should be, we will get the path here, and then we need to augment the path. So, first we're gonna say, if the path is empty, so if this is the, if target is our original top-level target, it'll have no path.
4. Building the Proxy
If the path is empty, we add a dot. We create a function to build our proxy, passing in value, path, and property. We focus on the get trap to observe property access. We check if a proxy already exists in the cache. If not, we create a new proxy using the recursive proxy handler. We add the proxy to the cache and return it.
So then, if the path is empty, let's add a dot. This is, you know, this is just dot notation, so we're gonna end up with a path with dot notation. And then we're gonna create a function that will build our proxy, we'll pass in value, and we'll pass path and property to it. So that'll let us build our proxy.
I'm not going to worry about set for the purposes of this demo, because it's a lot more complicated. But let's just worry about get, so we can observe whenever we're trying to get a property. So let's build this build proxy function, build proxy. And it takes in, yeah, value and property. Our property is going to be that path. The value is going to be the target on which we want to create the proxy.
The first thing we're going to do is set at the path cache so that up here we can actually look up all the paths. So we'll always know the path and we'll be able to recursively build this path and this cache of paths. And first we're going to use this target to check if we already have a proxy. So we wanna look in the proxy cache for this value. And if it exists, then we don't need to do anything. So we'll say, if not proxy, we will just check if we want to or really let's be more explicit. If proxy is undefined, first, let's try to create the new proxy. We're going to use the same proxy handler that is our recursive proxy handler. Let's not worry about logging anything in the cache. And then we have this proxy.
We need to add it to the proxy cache. So proxy cache dot set value proxy. There we go. Now we have defined that we're keeping track of this proxy cache. So it's going to have all of our proxies. We don't accidentally overwrite a proxy with another proxy. We have this try catch, just in case something goes wrong. Okay, so we've built our proxy. We have our getter that's recursively calling build proxy. At the end, build proxy returns the proxy, which is going to be the actual value.
5. Observing Changes with Proxy
We build a proxy to observe changes to an object. The recursive getter allows us to observe all of its children. We were able to observe state, person, and firstName. The recursion terminates at firstName because it is a string.
But what we need to do here is instead of new proxy, we need to say build proxy. And for now, let's just initialize it with an empty object. And then we'll say state.person equals firstName, lastName, and yeah, this works. Okay, so we're going to build our proxy here to start. Then we're going to say state.person equals firstName, lastName. We will expect something to be logged here, and then we're going to setPerson again and something will log again. So we got person, that's right here. And then we got firstName. So getting firstName, this is awesome. We have this recursive getter. So every time we get a new property, it builds a proxy with that property so that we can observe all of its children. And so we were first able to observe state. Then we were able to observe person. Then we were able to observe firstName. And it terminated at firstName because firstName is a string. So if we had something off of firstName, that wouldn't work for us.
6. Using Taits Library for State Management
Let's look at a library called taits, a small library for state management using the proxy API. We'll use Taits to fetch posts from a headless WordPress site using Apollo and GraphQL. We'll subscribe to posts and log the length when the posts object is set. We'll also observe all 10 posts in a loop and retrieve the title of each post. Finally, we'll create a function to grab posts.
So let's look at a library that I wrote called taits. So taits is a small library for state management that uses the proxy API in a similar manner to how we're using it here, but it allows you to just create your proxy, it does everything it needs to, and then you can subscribe for changes to paths on that proxy.
So I'll wipe this out. I've already installed all of the dependencies I need, but what we're going to do is use Tait, and fetch posts out of a headless WordPress site. So we'll use Apollo. We have a headless WordPress site set up with GraphQL, and we're just going to pull in posts. I'm gonna copy and paste some stuff in here so I don't type it wrong, but we're just gonna use Apollo and make a GraphQL call to get posts out of our headless WordPress site. And then we will set those posts on state and try to observe the changes.
So the first thing you do with Taits is you get a state object and a subscribe method. And if you're wondering why I call it Taits, I'm not very original, Taits is just state with the S on the end. So let's subscribe to posts. So we want to know when the posts object is set and posts, we expect posts to be an array of posts. So let's just say, if not array.isArrayPosts do nothing. We only care about if we have posts. And then let's log, there are posts.lengthsPosts. Okay. So when we set posts on state, it's going to log the length of the posts. So by default, WP GraphQL pulls down 10 posts and I'm aware that the site we're going to be calling calls 10 posts. So let's take this one step further and let's actually, in a loop, let's observe all 10 of those posts. So ahead of time, notice our state is nothing, we haven't defined posts or anything on our state, but we can observe changes ahead of time. So let's subscribe in a loop and just try to get the title of each post. So, if not title, return. So in order to do this, we want this for loop and then we want posts.title. Taits is able to turn this into a map and understand we need to listen to, you know, post as an array. We wanna listen to property zero through nine in that array, or index zero through nine. Okay, so now let's create our function to grab posts. And I'm just going to copy some of this in here so that I get it right. Yeah, let's do that. Get posts. Because otherwise I will not get this right.
7. Importing Tates and Observing State Changes
If there are errors, we'll just log them. I have a WordPress site set up and ready to get posts using the WPGraphQL API. We make a query to get post nodes title and set the result to posts. After calling get posts, we successfully imported Tates in a real-world scenario, updating state and notifying subscribers. You can find the code on my GitHub and follow me on Twitter for any questions. Thank you, Will, for the talk. Let's see the poll.
So if there are errors, we'll just log them.
Okay, I have a WordPress site set up at this URL. I'm gonna use isomorphic fetch because we're using Node here. And my WordPress site already has a list of posts, so it's ready to go, ready to get posts. And let's, I'm gonna copy the query here. Client.query. So we're making a jql query here to get post nodes title. This is just the WPGraphQL API. And then we will set post to result, or request result.data.Posts.nodes. So assuming we're able to make this request, get the post, we will set the array, which is these nodes, and each node has a title on it, to posts. All right, and the last thing to do here is call get posts. So let's see, let's run this and see what it gives us. It gives us nothing, great. So what did we do wrong? Let's go back through here. I wrote a lot of code without doing much in here. So, what have we done? Ah, we need a wait. There we go. With our wait, it actually waited for the post to return and then we have, there are 10 posts and for each post, we're logging the title. My titles are very original, post 41, 40, all the way down to 32. And there you have it. We have successfully imported Tates in a more real world-ish scenario. We're calling data, we're getting posts from Headless WordPress and we're setting state and it's updating us and our subscribers. What I like about this is we don't have methods specifically for setting. We just set posts directly on here and all of our subscribers were updated. So you don't have to worry about it.
If you like this talk, then you can get all of the code for this out on GitHub. If you go to github.com slash wjohnstow, that's my GitHub and I have it hosted out there. You can follow me on Twitter at wjohnstow and there's gonna be a little Q&A so I can answer any of your questions there as well. Thank you, Will, for this great talk. Now let's see the poll.
8. TypeScript vs JavaScript
TypeScript won by 56% and JavaScript by 44%. TypeScript started in a lead but JavaScript caught up. TypeScript may never fully replace JavaScript, as some people prefer the dynamic nature of JavaScript. However, TypeScript is beneficial for building large applications.
What do you prefer? TypeScript or JavaScript? So TypeScript by 56% and JavaScript by 44%. This is quite interesting. Is that surprising for you, Will? Yeah, actually TypeScript started out in a huge lead and then JavaScript crept back up. I think that, I mean obviously TypeScript won handily here, but as somebody was mentioning in chat, they consider it a loss because TypeScript didn't win 90% to 10%, which I tend to agree. That's funny. I voted, in full disclosure, I voted for TypeScript, so I was trying to move from here a little bit. In full disclosure, I voted for JavaScript. So, we canceled out, nice. Yeah, we canceled out. But do you think, for example, that maybe TypeScript can replace JavaScript? So, I think that TypeScript will probably never fully replace JavaScript because there are people who just love JavaScript and love the dynamic nature, and, I mean, more power to you. I think when I build large applications, I see the real benefit in TypeScript, but sometimes if I'm just trying to write small scripts, JavaScript is good enough. Yeah, yeah, I agree with you. I don't think TypeScript can replace JavaScript in the near future, but it's huge, and I think we'll be keep increasing and getting more adoption, but JavaScript will always be number one, leading.
Performance Implications of Proxies
The performance of proxies is often a concern, but it depends on the use case. If you're trying to optimize for performance, you may want to avoid proxies. However, for normal use, proxies should be fast enough. There is another state management library called MobX that also uses proxies.
Okay, so now, now let's go to the Q&A. If you have questions. So the first question is, what are the performance implications of proxies? Yeah, so a lot of people bring up, when talking about proxies, they bring up the fact that the performance of proxies is slow compared to just normal JavaScript. To that, I would say, it really depends on what you're doing, but if you are working and contemplating something like replacing all your promises with callbacks or something of that nature, because you're trying to get eek out that extra bit of performance, then maybe you should avoid proxies. But proxies should be fast enough for normal use. And somebody mentioned it in the chat, but there is another state management library called MobX, Mobix, I don't know how people pronounce it, but it uses proxies under the hood as well. Yeah, so it doesn't affect performance very much.
Comments