Bird
Raised Fist0
LangChainframework~8 mins

RunnablePassthrough and RunnableLambda in LangChain - Performance & Optimization

Choose your learning style10 modes available

Start learning this pattern below

Jump into concepts and practice - no test required

or
Recommended
Test this pattern10 questions across easy, medium, and hard to know if this pattern is strong
Performance: RunnablePassthrough and RunnableLambda
MEDIUM IMPACT
This concept affects how quickly and efficiently small tasks or functions run within a LangChain workflow, impacting interaction responsiveness and overall execution speed.
Passing data through a chain without modification
LangChain
from langchain.schema.runnables import RunnablePassthrough

passthrough = RunnablePassthrough()
result = passthrough.invoke(data)
RunnablePassthrough is optimized for direct passthrough with minimal overhead.
📈 Performance Gainreduces function call overhead and keeps bundle size minimal
Passing data through a chain without modification
LangChain
class CustomRunnable:
    def invoke(self, input):
        return input

custom = CustomRunnable()
result = custom.invoke(data)
Creating a custom class for simple passthrough adds unnecessary overhead and complexity.
📉 Performance Costadds extra function call overhead and increases bundle size slightly
Performance Comparison
PatternFunction CallsCall Stack DepthExecution OverheadVerdict
Custom class passthrough2+IncreasedHigher due to extra layers[X] Bad
RunnablePassthrough1MinimalMinimal overhead[OK] Good
Wrapped function in class3+IncreasedHigher due to wrapping[X] Bad
RunnableLambda with inline function1MinimalMinimal overhead[OK] Good
Rendering Pipeline
RunnablePassthrough and RunnableLambda affect the execution phase of the LangChain pipeline by minimizing overhead in function invocation, which improves responsiveness and reduces latency.
Execution
Function Invocation
⚠️ BottleneckExcessive function call layers increase call stack depth and slow execution.
Core Web Vital Affected
INP
This concept affects how quickly and efficiently small tasks or functions run within a LangChain workflow, impacting interaction responsiveness and overall execution speed.
Optimization Tips
1Use RunnablePassthrough for direct data passing to avoid overhead.
2Use RunnableLambda for lightweight inline functions to keep call stacks shallow.
3Avoid unnecessary class wrappers around simple functions to reduce execution layers.
Performance Quiz - 3 Questions
Test your performance knowledge
Which Runnable type minimizes overhead when simply passing data through without changes?
ARunnableLambda
BCustom class with invoke method
CRunnablePassthrough
DRunnableSequence
DevTools: Performance
How to check: Record a performance profile while running your LangChain workflow. Look for function call stacks and execution time of your runnables.
What to look for: Check for deep call stacks and long function durations indicating overhead. Shallow stacks with short durations confirm efficient runnables.

Practice

(1/5)
1. What does RunnablePassthrough do with the input it receives?
easy
A. Ignores the input and returns a fixed value
B. Transforms the input using a custom function
C. Returns the input exactly as it is without any changes
D. Throws an error if input is not a string

Solution

  1. Step 1: Understand RunnablePassthrough behavior

    RunnablePassthrough is designed to return whatever input it receives without modifying it.
  2. Step 2: Compare options with behavior

    Only Returns the input exactly as it is without any changes matches this behavior exactly; others describe transformations or errors which RunnablePassthrough does not do.
  3. Final Answer:

    Returns the input exactly as it is without any changes -> Option C
  4. Quick Check:

    RunnablePassthrough returns input unchanged = A [OK]
Hint: RunnablePassthrough just passes input through unchanged [OK]
Common Mistakes:
  • Thinking RunnablePassthrough modifies input
  • Confusing RunnablePassthrough with RunnableLambda
  • Assuming it throws errors on certain inputs
2. Which of the following is the correct way to create a RunnableLambda that doubles a number input?
easy
A. RunnableLambda(lambda x: x + 2)
B. RunnableLambda(lambda x: x * 2)
C. RunnableLambda(lambda x: x / 2)
D. RunnableLambda(lambda x: x - 2)

Solution

  1. Step 1: Identify the doubling function

    Doubling means multiplying the input by 2, so the function should be lambda x: x * 2.
  2. Step 2: Match with options

    RunnableLambda(lambda x: x * 2) matches the doubling function exactly; others perform addition, division, or subtraction.
  3. Final Answer:

    RunnableLambda(lambda x: x * 2) -> Option B
  4. Quick Check:

    Doubling function uses multiplication by 2 = D [OK]
Hint: Doubling means multiply input by 2 in lambda [OK]
Common Mistakes:
  • Using addition instead of multiplication
  • Confusing division or subtraction for doubling
  • Incorrect lambda syntax
3. What will be the output of this code?
passthrough = RunnablePassthrough()
lambda_runner = RunnableLambda(lambda x: x.upper())
result = lambda_runner.invoke(passthrough.invoke('hello'))
print(result)
medium
A. 'HELLO'
B. 'hello'
C. Error: RunnablePassthrough cannot be invoked
D. 'Hello'

Solution

  1. Step 1: Trace RunnablePassthrough output

    Calling passthrough.invoke('hello') returns 'hello' unchanged.
  2. Step 2: Apply RunnableLambda function

    The lambda converts input to uppercase, so 'hello'.upper() returns 'HELLO'.
  3. Final Answer:

    'HELLO' -> Option A
  4. Quick Check:

    Passthrough returns input, lambda uppercases it = C [OK]
Hint: Passthrough returns input, lambda transforms it [OK]
Common Mistakes:
  • Expecting passthrough to modify input
  • Confusing case conversion result
  • Assuming runtime error on invoke
4. Identify the error in this code snippet:
lambda_runner = RunnableLambda(lambda x: x + 1)
result = lambda_runner.invoke('5')
print(result)
medium
A. TypeError because string '5' cannot be added to integer 1
B. SyntaxError in lambda function
C. RunnableLambda cannot be invoked with strings
D. No error; output will be '51'

Solution

  1. Step 1: Analyze lambda operation on input

    The lambda tries to add 1 to input '5', which is a string, causing a type mismatch.
  2. Step 2: Identify error type

    Adding integer 1 to string '5' raises a TypeError in Python.
  3. Final Answer:

    TypeError because string '5' cannot be added to integer 1 -> Option A
  4. Quick Check:

    Adding int to string causes TypeError = B [OK]
Hint: Adding int to string causes TypeError in Python [OK]
Common Mistakes:
  • Assuming implicit string to int conversion
  • Thinking output is string concatenation
  • Confusing error types
5. You want to create a LangChain workflow that takes a list of numbers, passes it unchanged, then doubles each number. Which combination of RunnablePassthrough and RunnableLambda is correct?
hard
A. Use RunnablePassthrough twice, no lambda needed
B. Use RunnableLambda with lambda x: x, then RunnablePassthrough to double numbers
C. Use RunnableLambda with lambda x: x*2, then RunnablePassthrough to pass list
D. Use RunnablePassthrough to pass the list, then RunnableLambda with lambda x: [i*2 for i in x]

Solution

  1. Step 1: Pass list unchanged with RunnablePassthrough

    RunnablePassthrough returns the list as is, so it fits the first step.
  2. Step 2: Double each number with RunnableLambda

    RunnableLambda with lambda x: [i*2 for i in x] correctly doubles each element in the list.
  3. Final Answer:

    Use RunnablePassthrough to pass the list, then RunnableLambda with lambda x: [i*2 for i in x] -> Option D
  4. Quick Check:

    Passthrough passes list, lambda doubles elements = A [OK]
Hint: Passthrough passes input; lambda transforms list elements [OK]
Common Mistakes:
  • Trying to double list with passthrough
  • Using lambda that multiplies list object, not elements
  • Reversing order of passthrough and lambda