proxy-pattern

With a Proxy object, we get more control over the interactions with certain objects. A proxy object can determine the behavior whenever we're interacting with the object, for example when we're getting a value, or setting a value.

Safety Notice

This listing is imported from skills.sh public index metadata. Review upstream SKILL.md and repository scripts before running.

Copy this and send it to your AI assistant to learn

Install skill "proxy-pattern" with this command: npx skills add patternsdev/skills/patternsdev-skills-proxy-pattern

Proxy Pattern

With a Proxy object, we get more control over the interactions with certain objects. A proxy object can determine the behavior whenever we're interacting with the object, for example when we're getting a value, or setting a value.

When to Use

  • Use this when you need to add validation, formatting, notifications, or debugging to object access

  • This is helpful for controlling and intercepting property gets and sets on objects

When NOT to Use

  • In performance-critical hot paths where Proxy overhead on every property access matters

  • When simple getter/setter methods or Object.defineProperty achieve the same validation with less indirection

  • When the target objects are rarely accessed and the interception logic isn't needed

Instructions

  • Create a Proxy with a handler object defining get and set traps

  • Use the Reflect object within handlers for cleaner property access and modification

  • Add validation logic in the set trap to ensure data integrity

  • Avoid using proxies in performance-critical code paths as they add overhead

Details

Generally speaking, a proxy means a stand-in for someone else. Instead of speaking to that person directly, you'll speak to the proxy person who will represent the person you were trying to reach. The same happens in JavaScript: instead of interacting with the target object directly, we'll interact with the Proxy object.

Let's create a person object, that represents John Doe.

const person = { name: "John Doe", age: 42, nationality: "American", };

Instead of interacting with this object directly, we want to interact with a proxy object. In JavaScript, we can easily create a new proxy by creating a new instance of Proxy .

const person = { name: "John Doe", age: 42, nationality: "American", };

const personProxy = new Proxy(person, {});

The second argument of Proxy is an object that represents the handler. In the handler object, we can define specific behavior based on the type of interaction. Although there are many methods that you can add to the Proxy handler, the two most common ones are get and set :

  • get : Gets invoked when trying to access a property

  • set : Gets invoked when trying to modify a property

Instead of interacting with the person object directly, we'll be interacting with the personProxy .

Let's add handlers to the personProxy Proxy. When trying to modify a property, thus invoking the set method on the Proxy , we want the proxy to log the previous value and the new value of the property. When trying to access a property, thus invoking the get method on the Proxy , we want the proxy to log a more readable sentence that contains the key and value of the property.

const personProxy = new Proxy(person, { get: (obj, prop) => { console.log(The value of ${prop} is ${obj[prop]}); return obj[prop]; }, set: (obj, prop, value) => { console.log(Changed ${prop} from ${obj[prop]} to ${value}); obj[prop] = value; return true; }, });

When accessing the name property, the Proxy returned a better sounding sentence: The value of name is John Doe .

When modifying the age property, the Proxy returned the previous and new value of this property: Changed age from 42 to 43 .

A proxy can be useful to add validation. A user shouldn't be able to change person 's age to a string value, or give them an empty name. Or if the user is trying to access a property on the object that doesn't exist, we should let the user know.

const personProxy = new Proxy(person, { get: (obj, prop) => { if (!obj[prop]) { console.log( Hmm.. this property doesn't seem to exist on the target object ); } else { console.log(The value of ${prop} is ${obj[prop]}); } return obj[prop]; }, set: (obj, prop, value) => { if (prop === "age" && typeof value !== "number") { console.log(Sorry, you can only pass numeric values for age.); } else if (prop === "name" && value.length < 2) { console.log(You need to provide a valid name.); } else { console.log(Changed ${prop} from ${obj[prop]} to ${value}.); obj[prop] = value; } return true; }, });

The proxy makes sure that we aren't modifying the person object with faulty values, which helps us keep our data pure!

Reflect

JavaScript provides a built-in object called Reflect , which makes it easier for us to manipulate the target object when working with proxies.

Previously, we tried to modify and access properties on the target object within the proxy through directly getting or setting the values with bracket notation. Instead, we can use the Reflect object. The methods on the Reflect object have the same name as the methods on the handler object.

Instead of accessing properties through obj[prop] or setting properties through obj[prop] = value , we can access or modify properties on the target object through Reflect.get() and Reflect.set() . The methods receive the same arguments as the methods on the handler object.

const personProxy = new Proxy(person, { get: (obj, prop) => { console.log(The value of ${prop} is ${Reflect.get(obj, prop)}); return Reflect.get(obj, prop); }, set: (obj, prop, value) => { console.log(Changed ${prop} from ${obj[prop]} to ${value}); return Reflect.set(obj, prop, value); }, });

We can access and modify the properties on the target object easily with the Reflect object.

Tradeoffs

Proxies are a powerful way to add control over the behavior of an object. A proxy can have various use-cases: it can help with validation, formatting, notifications, or debugging.

Overusing the Proxy object or performing heavy operations on each handler method invocation can easily affect the performance of your application negatively. It's best to not use proxies for performance-critical code.

Source

  • patterns.dev/vanilla/proxy-pattern

References

  • Proxy - MDN

  • JavaScript Proxy - David Walsh

  • Awesome ES2015 Proxy - GitHub @mikaelbr

  • Thoughts on ES6 Proxies Performance - Valeri Karpov

Source Transparency

This detail page is rendered from real SKILL.md content. Trust labels are metadata-based hints, not a safety guarantee.

Related Skills

Related by shared tags or category signals.

Coding

react-2026

No summary provided by upstream source.

Repository SourceNeeds Review
Coding

hooks-pattern

No summary provided by upstream source.

Repository SourceNeeds Review
Coding

react-composition-2026

No summary provided by upstream source.

Repository SourceNeeds Review
Coding

ai-ui-patterns

No summary provided by upstream source.

Repository SourceNeeds Review