frontpage.
newsnewestaskshowjobs

Made with ♥ by @iamnishanth

Open Source @Github

Open in hackernews

JavaScript helper function for you to use

4•EGreg•8h ago
Got a function that fetches some values asynchronously from somewhere? Consider wrapping it in this and making it a super-function.

Here is what it does for all the functions you wrap with it. In my experience, these are very helpful and also gives you a place you can even hook into and add more later (such as handling batching transparencly, etc):

Memoizes async getters: Call a function with the same arguments and it returns the cached result instead of recomputing.

Handles in-flight deduping: If multiple parts of your app call the same getter while it's still working, only one request is sent. The rest wait on the same promise.

Throttles concurrency: You can limit how many calls to your getter run in parallel. Useful for APIs, disk I/O, or anything rate-sensitive.

Supports custom caching backends: Pass any object with get, set, delete, and has. Works with Map, LRU, or your own cache logic.

Optional LRU eviction: If you pass a plain Map, it upgrades it to an LRU with a max size. Least recently used items are evicted when full.

Handles callbacks and Promises: Wraps traditional callback-style async functions, but gives you a modern Promise-based interface.

Smart-ish keying: Builds a cache key by stringifying non-function arguments. Works well for most everyday use cases.

Supports manual eviction: Call getter.forget(...args) to remove specific entries or getter.force(...args) to bypass the cache for one call.

Allows custom preparation logic: You can pass a prepare() function to clone or process cached results before using them.

  function createGetter(fn, {
    cache = new Map(),
    cacheSize = 100, // Used only if cache is a Map
    throttleSize = Infinity,
    prepare,
    callbackIndex,
    resolveWithFirstArgument = false
  } = {}) {
    const inFlight = new Map();
    let activeCount = 0;
    const queue = [];
  
    // Wrap Map in a simple LRU if needed
    if (cache instanceof Map) {
      const rawMap = cache;
      const lru = new Map();
  
      cache = {
        get(key) {
          if (!rawMap.has(key)) return undefined;
          const value = rawMap.get(key);
          lru.delete(key);
          lru.set(key, true); // Mark as most recently used
          return value;
        },
        set(key, value) {
          rawMap.set(key, value);
          lru.set(key, true);
          if (rawMap.size > cacheSize) {
            const oldest = lru.keys().next().value;
            rawMap.delete(oldest);
            lru.delete(oldest);
          }
        },
        delete(key) {
          rawMap.delete(key);
          lru.delete(key);
        },
        has(key) {
          return rawMap.has(key);
        }
      };
    }
  
    function makeKey(args) {
      return JSON.stringify(args.map(arg => (typeof arg === 'function' ? 'ƒ' : arg)));
    }
  
    function execute(context, args, key, resolve, reject) {
      const callback = (err, result) => {
        if (err) return reject(err);
        cache.set(key, [context, arguments]);
        if (prepare) prepare.call(null, context, arguments);
        resolve(resolveWithFirstArgument && context !== undefined ? context : result);
        processNext();
      };
  
      if (callbackIndex != null) args.splice(callbackIndex, 0, callback);
      else args.push(callback);
  
      if (fn.apply(context, args) === false) {
        cache.delete(key); // opt-out of cache
      }
    }
  
    function processNext() {
      activeCount--;
      if (queue.length && activeCount < throttleSize) {
        const next = queue.shift();
        activeCount++;
        execute(...next);
      }
    }
  
    const getter = function (...args) {
      return new Promise((resolve, reject) => {
        const context = this;
        const key = makeKey(args);
  
        if (cache.has(key)) {
          const [cachedContext, cachedArgs] = cache.get(key);
          if (prepare) prepare.call(null, cachedContext, cachedArgs);
          return resolve(resolveWithFirstArgument && cachedContext !== undefined ? cachedContext : cachedArgs[1]);
        }
  
        if (inFlight.has(key)) {
          return inFlight.get(key).then(resolve, reject);
        }
  
        const promise = new Promise((res, rej) => {
          if (activeCount < throttleSize) {
            activeCount++;
            execute(context, args.slice(), key, res, rej);
          } else {
            queue.push([context, args.slice(), key, res, rej]);
          }
        });
  
        inFlight.set(key, promise);
        promise.finally(() => {
          inFlight.delete(key);
        });
  
        promise.then(resolve, reject);
      });
    };
  
    getter.forget = (...args) => {
      const key = makeKey(args);
      inFlight.delete(key);
      return cache.delete(key);
    };
  
    getter.force = function (...args) {
      getter.forget(...args);
      return getter.apply(this, args);
    };
  
    return getter;
  }

Pocket LLM Server Just Like a Pocket WiFi

2•itstomo•1h ago•0 comments

Ask HN: People who work different timezones than your company. How sched?

11•tetris11•10h ago•16 comments

Ask HN: What are some cool or underrated tech companies based in Australia?

3•hao_liu•4h ago•0 comments

Ask HN: How did Soham Parekh get so many jobs?

290•jshchnz•6d ago•401 comments

Ask HN: How are you making money on the side?

16•arkensaw•15h ago•25 comments

Ask HN: What are some cool or underrated tech companies based in Canada?

79•pedrodelfino•8h ago•48 comments

Ask HN: How is the tech scene in LA?

14•asdev•2d ago•21 comments

Ask HN: Has anyone else learned English just by reading tech posts (like HN)?

10•FerkiHN•22h ago•5 comments

Ask HN: What's the verdict on GPT wrapper companies these days?

12•NewUser76312•1d ago•14 comments

Agentic terminology doesn't make any sense

4•mathewpregasen•1d ago•1 comments

Ask HN: What Are You Working On? (June 2025)

439•david927•1w ago•1383 comments

Ask HN: Any resources for finding non-smart appliances?

155•everyone•1d ago•130 comments

Ask HN: Freelancer? Seeking freelancer? (July 2025)

84•whoishiring•1w ago•211 comments

N8n AI Workflows – 3,400 Workflows and an LLM Prototype

11•sayedev•1d ago•1 comments

Ask HN: Do you use LLM for HTML translations?

2•Mooty•22h ago•2 comments

Are there any noteworthy LinkedIn alternatives?

4•junaidkhalid•22h ago•9 comments

Ask HN: Who is hiring? (July 2025)

271•whoishiring•1w ago•406 comments

Ask HN: Worth leaving position over push to adopt vibe coding?

74•NotAnOtter•4d ago•90 comments

Slack is just the worst – and I've used a BBS and 14.4k modem

11•sincethebbsera•8h ago•5 comments

Ask HN: What inspires you to persevere through adversity?

14•justanything•1d ago•17 comments

Ask HN: Took a break after burnout – what now?

8•BugsBunny1991•1d ago•11 comments

Ask HN: Who wants to be hired? (July 2025)

129•whoishiring•1w ago•388 comments

Ask HN: Advice for Starting a Hacker Space?

35•pkdpic•3d ago•36 comments

Ask HN: What's the 2025 stack for a self-hosted photo library with local AI?

228•jamesxv7•1w ago•120 comments

Ask HN: What happened to W3C's PROV initiative to add provenance to the Web?

3•azhenley•1d ago•0 comments

Proposal: GUI-first, text-based mechanical CAD inspired by software engineering

5•thinkmachyx•3d ago•15 comments

Ask HN: Brick and Mortar Dev Agency

2•takklz•1d ago•1 comments

Which email clients work well with keyboard shortcuts?

3•tenzo•1d ago•5 comments

Ask HN: HN was much more interesting a year ago

14•fud101•15h ago•12 comments

Ask HN: How do you deal with data backups in servers?

4•atomicnature•1d ago•4 comments