MinUk.Dev
Observability Engineering - minuk dev wiki

Observability Engineering

created : Mon, 20 Feb 2023 22:40:32 +0900
modified : Fri, 05 May 2023 22:42:41 +0900

Part I. The Path to Observability

Chapter 1. What Is Observability?

The Mathematical Definition of Observability

  • Observability : a measure of how well internal states of a system can be inferred from knowledge of its external outputs.

Applying Observability to Software Systems

  • Understand the inner workings of your application
  • Understand any stem state your application may have gotten itself into, even new ones you have never seen before and couldn’t have predicted
  • Understand the innner workings and system state solely by observing and interrogating with external tools
  • Understand the internal state without shipping any new custom code to handle it (because that implies you needed prior knowledge to explain it)

Why Observability Matters Now

  • With monitoring, software developers can’t fully see their systems. They squint at the systems.

Is This Really the Best Way?

  • As an industry, we generally don’t question whether we should do it, but how.
  • More abstract and more complex system

Why Are Metrics and Monitoring Not Enough?

  • Traditional:
    • Monolith
    • 1 stateful (database)
    • low-level permission
    • controllable containers, VMs, bare metals
    • static, long-running
    • only can examine after problems occurred.
    • dashboard & telemetry serve for only ops engineers.
    • block-box applications
    • focus on uptime & failure prevention
    • limited or small number of dimensions
  • But, now:
    • many service
    • polyglot persistence
    • extremely dynamic infrastructure
    • many far-flung and loosely couped services
    • uncontrollable
    • not only ops engineers, but also others
    • automatic instrumentation for complex systems
    • focus on reliability & toleration
    • unlimited number of dimensions

Debugging with Metrics Versus Observability

  • Debugging with metrics requires you to connect dozens of disconnected metrics that were recorded over the course of executing any one particular request, across any number of service or machines, to infer what might have occurred over the various hops needed for its fulfillment.
  • Monitoring is for the known-kunknowns, but observability is for the unknown-unknowns.

The Role of Cardinality

  • Cardinality matters for observability, because high-cardinality information is almost always the most useful in identifying data for debugging or understanding a system.
  • Unfortunately, metrics-based tooling systems can deal with only low-cardinality dimensions at any reasonable scale.

The Role of Dimensionality

  • Cardinality: The uniqueness of the value within your data
  • Dimensionality: The number of keys within that data

Debugging with Observability

  • Observability tools encourage developers to gather rich telemetry for every possible event that could occur, passing along the full context of any given request and storing it for possible use at some point down the line.
  • Observability tools are specifically designed to query aginst high-cardinality, high-dimensionality data.

Observability Is for Modern Systems

Chapter 2. How Debugging Practices Differ Between Observability and Monitoring

  • Traditional monitoring:
    • checking system conditions against known thresholds
    • reactive approach
  • Observability tools:
    • enabling iterative exploratory investigations
    • proactive approch to identifying any failure mode, whether known or unknown

How Monitoring Data Is Used for Debugging

  • Two main consumers of monitoring data:
    • machine: alert
    • human: dashboard

Troubleshooting Behaviors When Using Dashboards

  • Seeing that pattern, you quickly pull up the dashboard for the caching component of your system to confirm your suspicion.

The Limitations of Troubleshooting by Intuition

  • Example 1: Insufficient correlation:

    • Is a particular query scanning fewer rows than before?
    • How often is the new index getting chosen by the query planner, and for which queries?
    • Are write latencies up overall, on average, or at the 95th/99th percentiles?
    • Are queries fatser or slower when they use this index than their previous query plan?
    • What other indexes are also used along with the new index?
    • Has this index made any other indexes obsolete, so we can drop those and reclaim some write capacity?
  • The engineers in the preceding examples might go back and add custom metrics for each query family, for expiration rates per collection, for error rates per shards, etc.

Traditional Monitoring Is Fundamentally Reactive

How Observability Enables Better Debugging

  • Observability lets you explicitly discover the source of any problem, along any dimension or combination of dimensions, without needing to first predict where and how that problem might be happening; this model centers around questioning and understanding.
  • Differences:
    • Relying on institutional knowledge
    • Finding Hidden Issues
    • Having confidence in diagnosing production issues
  • Observability tools pull high-cardinality, high-dimensionality context from telemetry data into a signle location where investigators can easily slice and dice to zoom in, zoom out, or follow breadcrumbs to find definitive answers.

Conclusion

  • Monitoring-based debugging:
    • Metics & dashboard
  • Observability-based debugging:
    • Withouth learning on experience or intimate system knowledge to generate a hunch

Chapter3. Lessons from Scaling Wihtout Observability

  • If you can solve your problem with a LAMP stack (or equivalent), you probably should:
    • When architecting a service, the first rule is to not add unnecessary complexity.
    • Don’t confuse boring with bad.
  • On a technical level, thse shifts include several primary effects:
    • The decomposition of everything, from one to many
    • The need for a variety of data stores, from one database to many storage systems
    • A migration from monolithic applications toward many smaller microservices
    • A variety of infrastructure types away from “big iron” servers toward contatiners, functions, serverless, and other ephemeral, elastic resources

The Evolution Toward Modern Practices

  • Userexperience can no longer be generalized as being the same for all service users. In the new model, different users of a service may be routed through the system in different ways, using different components, providing experience s that can vary widely.

  • Monitoring alerts that look for edge cases in production that have system conditions exceeding known thresholds generate a tremendous number of false positives, false negatives, and meaningless noise. Alerting has shifted to a model in which fewer alerts are triggered, by focusing only on symptoms that directly impact user experience.

  • Debuggers can no longer be attached to one specific runtime. Fullfilling service requests now requires hopping across a network, spanning multiple runtimes, often multiple times per individual request.

  • Known recurring failures that require manual remediation and can be defined in a runbook are no longer the norm. Service failures have shifted from that model toward one in which known recurring fialures can be recovered automatically. Failures that cannot be automatcially recovered, and therefore trigger an alert, likely mean the responding engineer wil lbe facing a novel problem.

  • Tips to diagnose problems:

    • Microservices and requests that spanned multiple runtimes.
    • Polyglot storage systems, without requiring expertise in each.
    • Multitenancy and running obht server-side code and queries; we could easily drill down into individual user experiences to see exactly what was happening.

Chapter 4. How Observability Relates to DevOps, SRE, and Cloud Native

Cloud Native, DevOps, and SRE in a Nutshell

  • Loose coupling unlocks several key bueniness benefits
  • CNCF defines:
    • cloud native: building and running scalable applications in modern, dynamic environments…
    • resilient, manageable, and observable.
    • make high-impact changes frequently and predictably with minimal toil

Observability: Deubbing Then Versus Now

  • Goal: agnostic in terms of how it’s accomplished.
  • Immutable infrastructure means that it’s no longer feasible to ssh into a host for debugging.

Observability Empowers DevOps and SRE Practices

  • Chaos engineering and continuous verification
  • Feature flagging
  • Progressive release patterns
  • Incident analysis and blameless postmortem

Conclusion

  • The shift toward DevOps, SRE, and cloud native practices have created a need for a solution like observability.

Part II. Fundamentals of Observability

Chatper 5. Structured Events Are the Building Blocks of Observability

  • Metrics do not retain the context of the event, simply an aggregate measurement of what occurred at a particular time.

Debugging with Structured Events

  • event : a record of everything that occurred while one particular request interacted with your service.
  • Details:
    • unique IDs, variable values, headers, every parameter passed by the request, the execution time, any calls made to remote services, the execution time of those remote calls, or any other bit of context
    • key value pairs
  • Arbitariliy wide events

The Limitations of Metrics as a Building Block

  • pre-aggregated measure
  • As aggregate numerical representations of predefined relations over predefined periods of time

The Limitations of Traditional Logs as a Building Block

Unstructured Logs

  • Traditional logs are unstructured
  • hug volumes of noisy data : slow & clunky
  • Hard to handle in nowadays systems.

Structured Logs

  • Many logs are only portions of events
  • Dozon log lines
  • Log data can be useful when redesigned to resemble a structured event.

Properties of Events That Are Useful in Debugging

  • unknown-unknowns
  • An observability tool must be able to support high-cardinality queries to be useful to an investigator.
  • High cardinality and high dimensionality are the capabilities that enable you to find very fine-grained needles in deeply complex distributed system haystacks

Conclusion

  • The fundamental building block of observability is the arbitrarily wide structured event.
  • Metrics aggregate system state over a predefined period of time.
  • Unstructured logs are human redable but computationally difficult to use.
  • Structured logs are machine parsable and can be useful for the goals of observability.

Chapter 6. Stitchign Events into Traces

  • Distributed traces are simply an interrelated series of events.

Distributed Tracing and Why It Matters Now

  • Distirbuted tracing is a method of tracking the progression of a single request as it is handled by various services that make up an application.
  • Solution
    • Twitter’s Zipkin (2012)
    • Uber’s Jaeger (2017)
    • Honycomb, Lightstep

The Components of Tracing

  • TraceID: unique identifier
  • SpanID: unique identifier for each individual span created.
  • ParentID: A Parent ID is absent in the root span.
  • Timestamp: Each span must indicate when its work began
  • Duration: Each span must also record how long that work took to finish.

Instrumenting a Trace the Hard Way

  • traceID: UUID
func rootHandler(r *http.Request, w http.ResponseWriter) {
  traceData := make(map[string]any)
  traceData["trace_id"] = uuid.String()
  traceData["span_id"] = uuid.String()

  startTime := time.Now()
  traceData["timestamp"] = startTime.Unix()

  authorized := callAuthService(r)
  name := call NameService(r)

  if authorized {
    w.Write([]byte(fmt.Sprintf(`{"message": "Waddup %s"}`, name)))
  } else {
    w.Write([]byte(`{"message": "Not cool dawg"}`))
  }

  traceData["duration_ms"] = time.Now().Sub(startTime)
  sendSpan(traceData)
}
  • W3C
  • X-B3-TraceId: Contains the trace ID for the entire trace
  • X-B3-ParentSpanId: Contains the current span ID, which will be set as the parent ID in the child’s generated span

Adding Custom Fields into Trace Spans

func rootHandler(r *http.Request, w http.ResponseWriter) {
  traceData := make(map[string]any)
  traceData["tags"] = make(map[string]any)

  hostname, _ := os.Hostname()
  traceData["tags"]["hostname"] = hostname
  //
}

Stitching Events into Traces

  • Tracing doesn’t have to be limited to service-to-service calls.

Conclusion

  • Events are the building blocks of observability, and traces are simply an interrelated series of events.

Chapter 7. Instrumentation with OpenTelemetry

  • OpenTelemetry standard

A Brief Introduction to Instrumentation

  • Before otel, we may have installed instrumentation libraries or agents in our applications

Open Instrumentation Standards

  • OpenTelemetry(OTel) : OpenTracing + OpenCensus
  • OTel has become the single open source standard for application instrumentation among observability solutions.

Instrumentation Using Code-Based Examples

  • OpenTelemetry:
    • API
    • SDK
    • Tracer
    • Meter
    • Context propagation
    • Exporter
    • Collector

Start with Automatic Instrumentation

  • go:

    • http

      import "go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp"
      
      mux.Handle("/rout", otelhttp.NewHandler(otelhttp.WithRoutTag("/route", http.HandlerFunc(h)), "handler_span_name"))
    • gprc

      import (
        "go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgprc"
      )
      
      s := grpc.NewServer(
        grpc.UnaryInterceptor(otelgrpc.UnaryServerInterceptor()),
        gprc.StreamInterceptor(otelgrpc.StreamServerInterceptor()),
      )
  • java:

    @Configuration
    public class ObservationConfiguration {
      @Bean
      public MicrometerObservationCapability micrometerObservationCapability(ObservationRegistry registry) {
          return new MicrometerObservationCapability(registry);
      }
    }

Add Custom Instrumentation

  • Starting and finishing trace spans
import "go.opentelemetry.io/otel"

var tr = otel.Tracer("module_name")

func funcName(ctx context.Context) {
  sp := tr.Start(ctx, "span_name")
  defer sp.End()

  // do work here
}
  • Adding wide fields to an event
import "go.opentelemetry.io/otel/attribute"

sp.SetAttributes(attribute.Int("http.code", resp.ResponseCode))
sp.SetAttributes(attribute.String("app.user", username))

Send Instrumentation Data to a Backend System

import (
  x "github.com/my/backend/exporter"
  "go.opentelemetry.io/otel"
  sdktrace "go.opentelemetry.io/otel/sdk/trace"
)

func main() {
  exporterX := x.NewExporter(...)
  exporterY := y.NewExporter(...)
  tp, err := sdktrace.NewTracerProvider(
    sdktrace.WithSampler(sdktrace.AlwaysSample()),
    sdktrace.WithSyncer(exporterX),
    sdktrace.WithBatcher(exporterY),
  )
  otel.SetTracerProvider(tp)
}

Conclusion

  • OTel is an open source standard.

Chapter 8. Analyzing Events to Achieve Observability

  • Hypothesis-driven debugging: form hypotheses and explorer the data to confirm or deny them. replying on intuition and pattern matching.

Debugging from Known Conditions

  • Prepare runbook to identify and solve evry possible “root cause”

  • Modern systems rarely fail in precisely the same way twice.:

    • The time for creating runbooks & dashboards is largely wasted
    • Technical debt
    • Wrong documentation is perhaps more dangerous than no documentation
  • Be debugging from known conditions, is not collection event data for unknown conditions.:

    • e.g. tail -f and grep with unstructured logs, or finding a spike on one dashboard

Debugging from First Principles

  • The real power of observability is that you shouldn’t have to know so much in advance of debugging an issue.

Using the Core Analysis Loop

  • Four stages of the core analysis loop:
    • What are you trying to understand?
    • Visualize telemetry data to find relevant performance anomalies
    • Search for common dimensions within the anomalous area by grouping of filtering for different attributes in your wide events
    • Has this isolated likely dimensions that identify potential sources of the anomaly?

Automating the Brute-Force Portion of the Core Analysis Loop

  • The core analysis loop is a method to objectively find a signal that matters within a sea of otherwise normal system noise.

This Misleading Promise of AIOps

  • artificial intelligence for operations (AIOps)

Conclusion

  • The core analysis loop is an effective technique for fast fault localization.

Chapter 9. How Observability and Monitoring Come Together

Where Monitoring Fits

  • RRDs(round-robin databases) to TSDB(time series databases)
  • The optimization of monitoring systems to find known-unknowns means that it’s a best fit for understanding the state of your systems, which change much less frequently and in more predictable ways than your application code.
  • Over time, metrics have also been adapted to creep into monitoring application-level concerns.:
    • In the role of a warning signal, aggregate measures like metrics work well.

Where Observability Fits

  • The optimization of observability to find unknown-unknowns means it’s a best fit for understanding the state of the code you write, which changes much more frequently than your systems (typically, every day) and in far less predictable ways.
  • Monitoring and observabitliy tools have different best practices and different implementations, and they serve different purposes.

System Versus Software Considerations

  • Software: the code you are actively developing that runs a production service delivering value to your customers.

  • System: an umbrella for everything else about the underlying infrastructure and runtime that is necessary to run that service.:

    • databases(e.g., MySQL or MongoDB)
    • compute and storage (e.g. containers or virtual machines)
    • others (kafka, Postfix, HAProxy, Memcached, or even something like Jira)
  • Factors that vary between systems and software

| Factor | Your systems | Your software | | Rate of change | Package updates (monthly) | Repo commits (daily) | | Predictability | High (stable) | Low (many new features) | | Value to your business | Low (cost center) | High (revenue generator) | | Number of users | Few (internal teams) | Many (your customers) | | Core concern | Is the system or service healthy? | Can each request acquire the resources it needs for end-to-end execution in a timely and reliable manner? | | Evaluation perspective | The system | Your customers | | Evaluation criteria | Low-level kernel and hardware device drivers | Variables and API endpoint | | Functional responsibility | Infrastructure operations | Software development | | Method for understanding | Monitoring | Observability |

Accessing Your Organizational Needs

  • Observability will help you deeply understand how software you develop and ship is performing when serving your customers.
  • Metrics-based monitoring tools and their associated alerts help you see when capacity limits or known error conditions of underlying systems are being reached.:
    • low level DNS counters, disk statistics, ethernet ports, statistics, system firmware, etc.
  • Today, your mileage may vary, depending on the robustness of your cloud provider.

Exceptions: Infrastructure Monitoring That Can’t Be Ignored.

  • higher-order infrastructure metircs like CPU usage, memory consumption, and disk activity are indicative of physical performance limitations.

Real-World Examples

  • skip to summarize

Conclusion

  • Monitoring is best suited to evaluating the health of your systems.
  • The most notable exceptions to that neat dividing line are higher-order infrastructure metrics on physical devices that directly impact software performance.

Part III. Observability for Teams

Chapter 10. Applying Observability Practices in Your Team

Join a Community Group

Start with the Biggest Pain Points

  • The fastest way to drive adoption is to solve the biggest pain points for teams responsible for managing their production services. Target those pains. Resist the urge to start small.

Buy Instead of Build

  • ROI
  • Your best way to do that is to instrument your applications by using OpenTelemetry.
  • ELK stack (Elasticsearch, Logstash, and Kibana)
  • Prometheus
  • Jaeger

Flesh Out Your Instrumentation Iteratively

  • You don’t need a fully developed set of instrumentation to get immediate value with observability

Look for Opportunities to Leverage Existing Efforts

  • Individuals and organizations commit the sunk-cost fallacy when they continue a behavior or endeavor as a result of previously invested time, money, or effort.
  • Examples:
    • If you’re using an ELK stack - or even just the Logstash part - it’s trivial to add a snippet of code to fork the output of a source stream to your observability tool.
    • If you’re already using structured logs, all you need to do is add a unique ID to log events as they propagate throughout your entire stack. You can keep those logs in your existing log analysis tool, while also sending them as trace events to your observability tool.
    • Try running observability instrumentation (for example, Honeycomb’s Beelines or OTel) alongside your existing APM solution.
    • If you’re using Ganglia, you can leverage that data by parsing the Extensible Markup Language (XML) dump it puts into /var/tmp with a once-a-minute cronjob that that shovels that data into your observability tool as events. That’s a less than optimal use of observabilty, but it certainly creates familiarity for Ganglia users.
    • Re-create the most useful of your old monitoring dashbaords as easily referenceable queries within your new observability tool. While dashbaords certainly have their shortcomings, this gives new users a landing spot where they can understand the system performance they care about at glance, and also gives them an opportunity to explore and know more.

Prepare for the Hardest Last Push

  • The goal of a complete implementation is to have built a reliable go-to debugging solution that can be used to fully understand the state of your production applications whenever anything goest wrong.

Conclusion

  • moving fast, demonstrating high value and ROI, and tackling work iteratively.

Chapter 11. Observability-Driven Development

Test-Driven Development

  • TDD is particularly powerful because tests run the same way every time.
  • Observability can help you write and ship better code even before it lands in source control

Observability in the Development Cycle

  • While the original intent is still fresh in the original author’s head.

Determining Where to Debug

  • Observability is not for debugging your code logic.
  • Observability is for figuring out where in your systems to find the code you need to debug.

Debugging in the Time of Microservices

  • It is likely still incredibly unclear according to your monitoring tooling whether that slowness is being caused by any of the collowing:
    • A bug in your code
    • A particular user chaning their usage pattern
    • A database overflowing its capacity
    • Network connection limits
    • A misconfigured load balancer
    • Issues with service registration or service discovery
    • Some combination of the preceding factors

How Instrumentation Drives Observability

  • Good instrumentation drives observability.
  • Every engineer should be expected to instrument their code such that they can answer these questions as soon as it’s deployed:
    • Is your code doing what your expected it to do?
    • How does it compare to the previous version?
    • Are users actively using your code?
    • Are any abnormal condtions emerging?

Shifting Observability Left

  • observability-driven development

Using Observability to Speed Up Software Delivery

  • Observability-driven development in tandem with feature flags and progressive delivery patterns can equip engineering teams with the tools they need to stop instinctively rolling back deployments and instead dig in to further investigate what’s really happening whenever issues occur during relase of a new feature.

Conclusion

  • Observability can, and should, be used early in the software development life cycel.
  • Test-driven development is a useful tool for examining how your code runs against a defined specification.
  • Observability-driven development is a useful tool for examining how your code behaves in the chaotic and turbulent world of production.

Chapter 12. Using Service-Level Objectives for Reliability

Traditional Monitoring Approaches Create Dangerous Alert Fatigue

  • Problem: normalization of deviance
  • alert fatigue
  • Distributed systems today require an alternative approach.

Threshold Alert Is for Known-Unknowns Only

  • failure is inevitable.
  • Distributed systems design for resilience with loosely coupled components.
  • Failures that get automatically remediated should not trigger alarms.
  • Alerting criteria:
    • reliable indicator with service’s UX
    • alert must be solvable

User Experience Is a North Star

  • Static thresholds are too rigid and coarse to reliably indicate degreaded user experience in a dynamic environment.

What Is a Service-Level Objective?

  • Service-level objectives(SLOs)

Reliable Alerting with SLOs

  • Two kinds of SLIs exist:
    • time-based measures
    • event-based measures: must be actionable

Conclusion

  • SLOs decouple the “what” and “why” behind incident alerting. Focusing on symptom-of-pain-based alerts means that SLOs can be reliable indicators of customer experience.

Chatper 13. Acting on and Debugging SLO-Based ALerts

Alerting Before Your Error Budget Is Empty

  • error budge: maximum amount of system unavailability that your business is willing to tolerate

Framing Time as a Sliding Window

  • The correct first choice to make when calculating burn trajectories is to frame time as a sliding window, rather than a statci fixed window.
  • Otherwise, there isn’t enough data after a window reset to make meaningful decisions

Forecasting to Create a Predictive Burn Alert

  • The Lookahead Window
  • Extrapolating the future from current burn rate
  • Short-term burn alerts
  • Context-aware burn alerts

The Baseline Window

Acting on SLO Burn Alerts

Using Observability Data for SLOs Versus Time-Series data

  • Observability data that traces actual user experience with your services is a more accurate representation of system state than coarsely aggregated time-series data.

Conclusion

  • SLOs are a modern form of monitoring that solve many of the problems with noisy monitoring we outlined before.
  • SLOs are not specific to observability.

Chapter 14. Observability and the Software Supply Chain

  • software supply chain

Part IV. Observability at Scale

Chatper 15. Build Versus Buy and Return on Investment

How to Anlyze the ROI of Observability

The Real Costs of Building Your Own

  • The Hidden Costs of Using “Free” Software
  • The Benefits of Building Your Own
  • The Risks of Building Your Own

The Real Costs of Buying Software

  • The Hidden Financial Costs of Commercial Software
  • The Hidden Nonfinancial Costs of Commercial Software

Buy Versus Build Is Not a Binary Choice

Chatper 16. Efficient Data Storage

The Functional Requirements for Observability

  • Time-Series Databases Are Inadequate for Observability:
    • Traditionally, a time-series database(TSDB) is used to store aggregated metrics.
    • Problem: Cardinality Explosion
  • Other Possible Data Stores:
    • NoSQL
    • Apache Cassandra, Elasticsearch/OpenSearch, ScyllaDB, and InfluxDB
    • ClickHouse, Apache Druid
  • Data Storage Strategies:
    • Row-based Storage:
      • Bigtable
    • Column-based Storage

Chapter 17. Cheap and AccurateEnough: Sampling

Sampling to Refine Your Data Collection

  • At scale, the need to refine your data set to optimize for resource costs becomes critical. But even at a smaller scale, where the need to shave resources is less pressing, refining the data you decide to keep can still provide valuable cost savings.

Using Different Approaches to Sampling

Constant-Probability Sampling

  • Constant sampling is not effective in the following circumstances:
    • You care a lot about error cases and not very much about success cases.
    • Some customers send orders of magnitude more traffic than others, and you want all customers to have a good experience.
    • You want to ensure that a huge increase in traffic on your servers can’t overwhelm your analytics backend.

Sampling on Recent Traffic Volume

Sampling Based on Event Content (Keys)

  • Events with errors are more important than those with successes.

  • Events for newly placed orders are more important than those checking on order status.

  • Events affecting paying customers are more imporant to keep than those for cusomters using the free tier.

  • Others:

    • HTTP method

Combining per Key and Historical Methods

Choosing Dynamic Sampling Options

When to Make a Sampling Decision for Traces

  • tail-based sampling

Translating Sampling Strategies into Code

  • The Base Case
func handler(resp http.ResponseWriter, req *http.Request) {
  start := time.Now()
  i, err := callAnotherService()
  resp.Write(i)
  RecordEvent(req, start, err)
}
  • Fixed-Rate Sampling
var sampleRate = flag.Int("sampleRate", 1000, "Static sample rate")
func handler(resp http.ResponseWriter, req *http.Request) {
  start := time.Now()
  i, err := callAnotherService()
  resp.Write(i)

  r := rand.Float64()
  if r < 1.0 / *sampleRate {
    RecordEvent(req, start, err)
  }
}
  • Recording the Sample Rate
var sampleRate = flag.Int("sampleRate", 1000, "Service's sample rate")

fun handler(resp http.ResposneWriter, req *http.Request) {
  start := time.Now()
  i, err := callAnotherService()
  resp.Write(i)

  r := rand.Float64()
  if r < 1.0 / *sampleRate {
    RecordEvent(req, *sampleRate, start, err)
  }
}
  • Consistent Sampling:
    • Any full end-to-end trace
var sampleRate = flag.Int("sampleRate", 1000, "Service's sample rate")

func handler(resp http.ResponseWriter, req *http.Request) {
  // Use an upstream-generated random sampling ID if it exists.
  // otherwise we're a root span. generate & pass down a random ID.
  var r float64
  if r, err := floatFromHexBytes(req.Header.Get("Sampling-ID")); err != nil {
    r = rand.Float64()
  }

  start := time.Now()
  // Propagate the Sampling-ID when creating a child span
  i, err := callAnotherService(r)
  resp.Write(i)

  if r < 1.0 / *sampleRate {
    RecordEvent(req, *sampleRate, start, err)
  }
}
  • Target Rate Sampling
var targetEventsPerSec = flag.Int("targetEventsPerSec", 5,
  "The Target number of requests per second to sample from this ervice.")

var sampleRate float64 = 1.0
var requestsInPastMinute *int

func main() {
  // Initialize counters.
  rc := 0
  requestsInPastMinute = &rc

  go func() {
    for {
      time.Sleep(time.Minute)
      newSampleRate = *requestsInPastMinute / (60 * *targetEventsPerSec)
      if newSampleRate < 1 {
        sampleRate = 1.0
      } else {
        sampleRate = newSampleRate
      }
      newRequestCounter := 0
      requestsInPastMinute = &newRequestCounter
    }
  }()
  http.Handler("/", handler)
}

fun handler(resp http.ResponseWriter, req *http.Request) {
  var r float64
  if r, err := floatFromHexBytes(req.Header.Get("Sampling-ID")); err != nil {
    r = rand.Float64()
  }

  start := time.Now()
  *requestsInPastMinute ++
  i, err := call AnotherService(r)
  resp.Write(i)

  if r < 1.0 / sampleRate {
    RecordEvent(req, sampleRate, start, err)
  }
}
  • Having More Than One Static Sample Rate
var sampleRate = flag.Int("sampleRate", 1000, "Service's sample rate")

func handler(resp http.ResponseWriter, req *http.Request) {
  start := time.Now()
  i, err := callAnotherService(r)
  resp.Write(i)

  r := rand.Float64()
  if err != nil || time.Since(start) > 500*time.Millisecond {
    if r < 1.0 / *outlierSampleRate {
      RecordEvent(req, *outlierSampleRate, start, err)
    }
  } else {
    if r < 1.0 / *sampleRate {
      RecordEvent(req, *sampleRate, start, err)
    }
  }
}
  • Sampling by Key and Target Rate
  • Sampling with Dynamic Rates on Arbitrarily Many Keys
  • Putting It All togheter: Head and Tail per Key Target Rate Sampling

Chapter 18. Telemetry Mangement with Pipelines

Attributes of Telemetry Pipelines

  • Routing
  • Security and Compliance
  • Workload Isolation: to protect the reliability and availability of data sets in critical scenarios.
  • Data Buffering
  • Capacity Management:
    • Rate limiting
    • Sampling
    • Queuing
  • Data Filtering and Augmentation
  • Data Transformation
  • Ensuring Data Quality and Consistency:
    • To use logs as an example:
      • Convert unstructured logs to structured data by extracting sepcific fields
      • Detect and redact or filter any PII or sensitive data in the log data
      • Convert IP addresses to geographic latitude/longitude fields through the use of geolocation databases such as MaxMind
      • Ensure the schema of the log data, to ensure that the expected data exists and that specific fields are of specific types
      • Filter low-value logs from being sent to the backend

Managing a Telemetry Pipeline: Anatomy

  • Key components of a telemetry pipeline:

    • A receiver collects data from a source:
      • Prometheus scraper
    • A buffers is a store for the data, often for a short period of time:
      • Kafka
    • A processor often takes the data from a buffer, applices a transformation to it, and then persists the data back to a buffer.
    • An exporter is a component that acts as a sink for the telemetry data. An exporter often takes the data from a buffer and writes that data to a telemetry backend
  • Roles of the receiver, processor, and exporter in a pipeline:

    • Trace data:
      • Receiver:
        • Gather trace data in different formats(Zipkin, Jaeger, AWS X-Ray, Otel)
        • Gather data from different services ( from all Slack mobile clients)
      • Exporter or processor:
        • Ingest data into various trace backends
        • Perform tail sampling of the data
        • Drop low-value traces
        • Extract logs from traces
        • Route trace data to various backends for compliance needs
        • Filter data
    • Metrics data:
      • Receiver:
        • Identify and scrape targets
      • Exporter or processor:
        • Relabel metrics
        • Downample metrics
        • Aggregate metrics
        • Push data to multiple backends
        • Detect high-cardinality tags or time series
        • Filter high-cardinality tags or metrics
    • Logs data:
      • Receiver:
        • Gather data from different services
        • Endpoitn for collecting logs pushed from different services
      • Exporter or processor:
        • Parse log data into semistructured or sturctured logs
        • Filter PII and sensitive data from logs
        • Push data to multiple backends for GDPR resons
        • Route infrequently queried or audit logs to flat files, and high-value logs or frequent queries to an indexed system

Challenges When Managing a Telemetry Pipeline

  • Performance
  • Correctness
  • Availability
  • Reliability
  • Isolation
  • Data Freshness

Use Case: Telemetry Management at Slack

Open Source Alternatives

  • Facebook’s Scribe
  • Logstash, Fluentd
  • Prometheus Pushgateway, M3 Aggregator

Part V. Spreading Observability Culture

Chapter 19. The business Case for Observability

The Reactive Approach to Introducing Change

  • When to outage occurs
  • Dysfunction is no longer tolerated.

The Return on Investment of Observability

  • Observability’s four important wyas:
    • Higher incremental revenu:
      • Observability tools helps teams improve uptime and performance, leading to increased incremental revenue directly as a result of improving code quality.
    • Cost savings from faster incident response:
      • Observability significantly reduces labor costs via faster mean time and detect (MTTD) and mean time to resolve(MTTR), improved query response times, the ability to find bottlenecks quicker, reduction of time spent on call, and time saved by avoiding rollbacks
    • Cost savings from avoided incidents:
      • Observability tools enable developers to find causes of problems before they become critical and long-lasting, which helps prevent incidents.
    • Cost savings from decreased employee churn:
      • Implementing observability results in improved job satisfaction and decrease in developer burnout, alert and on-call fatigue, and turnover

The Proactive Approach to Introducing Change

Introducing Observability as a Practice

  • The goal of observability is to provide engineering teams the capability to develop, operate, throughly debug, and report on their systems.
  • With a blameless culture in practice, business leaders should also ensure that a clear scope of work exists when introducing obserability.

Using the Approriate Tools

Instrumentation

  • OTel is the merging standard. No longer lock into one specific vendor.

Data Storage and Analytics

  • Popular opensource data storage layers: Cassandra, Elasticsearch, M3, and InfluxDB

Rolling Out Tools to Your Teams

Knowing When You Have Enough Observability

Conclusion

  • The need for observability is recognized within teams for a variety of reasons.
  • Similar to security and testability, observability must be approached as an ongoing practice.
  • Observability requires onging care and maintenance.

Chapter 20. Observability’s Stakeholders and Allies

Recognizing Nonengineering Observability Needs

  • Understanding the adoption of new features. Which customers are using your newly shipped feature? Does that match the list of customers who expressed interest in it? In what ways do useage patterns of active feature users differ from those who tried it but later abandoned the experience?

  • Finding successful product usage trends for new customers. Does the sales team understand which combination of features seems to reasonate with prospects who go on to become customers? Do you understand the product usage commonalities in users that failed to activate your product? Do those point to friction that needs to be eroded somehow?

  • Accurately relaying service availability information to both customers and internal support teams via up-to-date service status pages. Can you provide templated queires so that supprt teams can self-serve when users report outages?

  • Understanding both short-term and long-term reliabili8ty trends. Is reliability improving for users of your software? Dose the shape of that reliability graph match other sources of data, like customer complaints? Are you experiencing fewer outages or more? Are those outages recovering more slowly?

  • Resolving issues proactively. Are you able to find and resolve customer-impacting issues before they are reported by a critical mass of cusomers via support tickets? Are you proactively resolving issues, or are you relying on cusomers to find them for you?

  • Shipping features to cusomters more quickly and reliably. Are deploymenets to production being closely watched to spot performance anomalies and fix them? Can you decouple deployments from releases so that shipping new features is less likely to cause widespread outages?

  • The best way to further your organization-wide observability adoption initiative is to reach out to adjacent teams that could benefit from this knowledge.

Creating Observability Allies in Practice

  • skip

Using Observability Versus Business Intelligence Tools

  • Query Execution Time
  • Accuracy
  • Recency
  • Structure
  • Time Windows
  • Ephemerality

Chapter 21. An Observability Maturity Model

A Note About Maturity Models

  • CMM(Capability Maturity Model):
    • Level 1: Initial(chaotic, ad hoc, individual heroics) - the starting point for use of a new or undocumented repeat process
    • Level 2: Repeatable - the process is at least documented sufficiently such that repeating the same steps may be attempted.
    • Level 3: Defined - the process is defined/confirmed as a standard business process
    • Level 4: Capable - the process is quantitatively managed in accordance with agreed-upon metrics
    • Level 5: Efficient - process management includes deliberate process optimization/improvement
  • Maturaity models can help you identify and quantify tangible and measurable objectives that are useful in driving long-term initiatives.
  • Those hypotheses, and the maturity model itself, should be continuously improved over time as more data becomes avaiable.

Why Observability Needs a Maturity Model

  • Production software systems present sociotechinal challenges

About the Obserability Maturity Model

  • OMM(Observability Maturity Model):
    • Sustainable sysems and quality of life for engineers
    • Delivering against business needs by increasing customer satisfaction
  • Observability is not aproperty of the computer system alone or the people alone.

Capabilities Refernced in the OMM

  • There is no singular and correct order or prescriptive way of doing these things.

Respond to System Failure with Resilience

  • If your team is doign well:

    • System uptime meets your business goals and is improving.
    • On-call resposne to alerts is efficient, and alerts are not ignored.
    • On-call dutiy is not excessively stressful, and engineers are not hesitant to take additional shifts as needed.
    • Engineers can handle incident workload without working extra hours or feeling unduly stressed.
  • If you team is doing poeerly:

    • The organization is spending a lot of additional time and money staffing on-call rotations.
    • Incidents are frequent and prolonged.
    • Those on call suffer from alert fatigue or aren’t alerted about real failures
    • Incident responders cannot easily diagnose issues.
    • Some team members are disproportionately pulled into emergencies.
  • How observability is related:

    • Democratization of troubleshooting techiniques by easily sharing past investigation paths helps distribute incident resolution skills across a team, so anyone can effectively respond to incidents as they occur.

Deliver High-Quality Code

  • If your team is doing well:

    • Code is stable, fewer burgs are discoverd in production, and fewer outages occur.
    • After code is deployed to production, your team focuses on customer solutions rather than support.
    • Engineers find it intutive to debug problems at any stage, from writing code in development to troubleshooting incidents in production at full release scale.
    • Isolated issues that occur can typically be fixed withotu triggering cascading failures.
  • If your team is doing poeerly:

    • Customer support costs are high.
    • A high percentage of engineering time is spent fixing bugs instead of working on new features.
    • Team members are often reluctant to deploy new features because of perceived risk.
    • It takes a long time to identify an issue, construct a way to reproduce the failure case, and reparie it.
    • Developers have low confidence in their code’s reliability after it has shipped.
  • How observability is related:

    • Well-monitored and tracked code makes it easy to see when and how a process is failing, and easy to identify and fix vulnerable spots.
    • Be alerted rapidly, fix before visible to users

Manage Complexity and Technical Debt

  • If your team is doing well:

    • Engineers spend the majority of their time making forward progress on core business goals.
    • Bug fixing and other reactive work takes up a minority of the team’s time.
    • Engineers spend very little time disoriented or trying to find where in the codebase they need to plumb through changes.
  • If your team is doing poorly:

    • Engineering time is wated rebuilding thigns when their scaling limits are reached or edge cases are hit.
    • Teams are distracted by fixing the wrong thing or picking the wrong way to fix something.
    • Engineers frequently experience uncontrollable ripple effects from a localized change.
    • People are afraid to make changes to the code, aka the “haunted graveyard” effect.
  • How observability is related:

    • Observability enables teams to understand the end-to-end performance of their systems and debug failures and slowness without wasting time.

Release on a Predictable Cadence

  • If your team is doing well:

    • The release cadence matches business needs and customer expectations.
    • Code gest into production shortly after being written. Engieers can trigger deployment of their own code after it’s been peer reviewed, satisfies controls, and is checked in.
    • Code paths can be enabled or disabled instantly, without needing a deployment.
    • Deployments and rollbacks are fast.
  • If your team is doing poorly:

    • Releases are infrequent and require lots of human intervention.
    • Lots of changes are shipped at once.
    • Releases have to happen in a particular order.
    • The sales team has to gate promises on a particular release train.
    • Teams avoid deploying on certain days or times of year. They are hesitant because poorly managed release cycles have frequently interfered with quality of life during nonbusiness hours.
  • How observability is related:

    • Observability is how your understand the build pipeline as well as production.
    • Observability and instrumentation are also how you gain confidence in your release.

Understand User Behavior

  • If your team is doing well:

    • Instrumentation is easy to add and augment.
    • Developers have easy access to key performance indicators (KIPs) for customer outcomes and system utilization/cost, and can visualize them side by side.
    • Feature flagging or similar makes it possible to iterate rapidly with a small subset of users before fully launching.
    • Product managers cna get a useful view of customer feedback and behavior.
    • Product-market fit is easier to achieve.
  • If your team is doing poorly:

    • Product managers don’t have enought data to make good decisions about what to build next.
    • Developers feel that their work doesn’t have impact.
    • Product features grow to excessive scope, are designed by committe, or don’t receive customer feedback until late in the cycle.
    • Product-market fit is not achieved.
  • How observability is related:

    • Effective product management requires access to relavant data.

Using the OMM for Your Organization

  • To remember that creatign a mature observability practice is not a linear progression and that these capabilities do not exist in a vaccum.

Conclusion

  • How they respond to system failure with resilience
  • How easily they can deliver high-quality code
  • How well they manage complexity and technical debt
  • How predictable their software release cadence is
  • How well they can understand user behavior

Chatper 22. Where to Go from Here

  • Observability for software systems is a measure of how well you can understand and explain any state your system can get into, no matter how novel or bizarre. You must be able to comparatively debug that bizarre or novel state across all dimensions of system state data, and combinations of dimensions, in an ad hoc iterative investigation, without being required to define or predict those debugging needs in advance. If you can understand any bizarre or novel state without needing to ship new code, you have observability.

Observability, Then Versus Now

Additional Resource

  • Site Reliability Engineering
  • Implementing Service Level Objectives
  • Cloud Native Observability with OpenTelemetry
  • Distributed Tracing in Practice
  • Honycomb’s blog

Predictions for Where Observability Is Going

  • Otel
  • RUM(real user monitoring):
    • colelction and recording web traffic
    • without impacting code performance
  • synthetic monitoring:
    • allow proactive testing
    • catch performance issues before they can possibly create impacts to real users.