Skip to content

Commit

Permalink
Added typeclasses for ReadableLogRecords and ReadWriteLogRecords
Browse files Browse the repository at this point in the history
  • Loading branch information
evanlauer1 committed Jul 4, 2024
1 parent 85583ab commit ec5674e
Show file tree
Hide file tree
Showing 2 changed files with 94 additions and 51 deletions.
68 changes: 54 additions & 14 deletions api/src/OpenTelemetry/Internal/Logging/Types.hs
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
{-# LANGUAGE DeriveFunctor #-}
{-# LANGUAGE InstanceSigs #-}
{-# LANGUAGE NamedFieldPuns #-}

module OpenTelemetry.Internal.Logging.Types (
LoggerProvider (..),
Expand All @@ -18,7 +19,6 @@ module OpenTelemetry.Internal.Logging.Types (
import Data.Function (on)
import qualified Data.HashMap.Strict as H
import Data.IORef (IORef, atomicModifyIORef, modifyIORef, newIORef, readIORef)
import Data.Int (Int64)
import Data.Text (Text)
import OpenTelemetry.Common (Timestamp, TraceFlags)
import OpenTelemetry.Context.Types (Context)
Expand All @@ -31,6 +31,10 @@ import OpenTelemetry.Resource (MaterializedResources)
-- | @Logger@s can be created from @LoggerProvider@s
data LoggerProvider = LoggerProvider
{ loggerProviderResource :: MaterializedResources
-- ^ Describes the source of the log, aka resource. Multiple occurrences of events coming from the same event source can happen across time and they all have the same value of Resource.
-- Can contain for example information about the application that emits the record or about the infrastructure where the application runs. Data formats that represent this data model
-- may be designed in a manner that allows the Resource field to be recorded only once per batch of log records that come from the same source. SHOULD follow OpenTelemetry semantic conventions for Resources.
-- This field is optional.
, loggerProviderAttributeLimits :: AttributeLimits
}
deriving (Show, Eq)
Expand All @@ -49,30 +53,73 @@ data Logger = Logger

{- | This is a data type that can represent logs from various sources: application log files, machine generated events, system logs, etc. [Specification outlined here.](https://opentelemetry.io/docs/specs/otel/logs/data-model/)
Existing log formats can be unambiguously mapped to this data type. Reverse mapping from this data type is also possible to the extent that the target log format has equivalent capabilities.
Uses an IORef under the hood to allow mutability.
-}
newtype LogRecord a = LogRecord (IORef (ImmutableLogRecord a))
data LogRecord a = LogRecord Logger (IORef (ImmutableLogRecord a))


mkLogRecord :: Logger -> ImmutableLogRecord body -> IO (LogRecord body)
mkLogRecord l = fmap (LogRecord l) . newIORef


mkLogRecord :: ImmutableLogRecord body -> IO (LogRecord body)
mkLogRecord = fmap LogRecord . newIORef
{- | This is a typeclass representing @LogRecord@s that can be read from.
A function receiving this as an argument MUST be able to access all the information added to the LogRecord. It MUST also be able to access the Instrumentation Scope and Resource information (implicitly) associated with the LogRecord.
The trace context fields MUST be populated from the resolved Context (either the explicitly passed Context or the current Context) when emitted.
Counts for attributes due to collection limits MUST be available for exporters to report as described in the transformation to non-OTLP formats specification.
-}
class ReadableLogRecord r where
-- | Reads the current state of the @LogRecord@ from its internal @IORef@. The implementation mirrors @readIORef@.
readLogRecord :: r a -> IO (ImmutableLogRecord a)


-- | Reads the @InstrumentationScope@ from the @Logger@ that emitted the @LogRecord@
readLogRecordInstrumentationScope :: r a -> InstrumentationLibrary


-- | Reads the @Resource@ from the @LoggerProvider@ that emitted the @LogRecord@
readLogRecordResource :: r a -> MaterializedResources


{- | This is a typeclass representing @LogRecord@s that can be read from or written to. All @ReadWriteLogRecord@s are @ReadableLogRecord@s.
A function receiving this as an argument MUST additionally be able to modify the following information added to the LogRecord:
- Timestamp
- ObservedTimestamp
- SeverityText
- SeverityNumber
- Body
- Attributes (addition, modification, removal)
- TraceId
- SpanId
- TraceFlags
-}
class (ReadableLogRecord r) => ReadWriteLogRecord r where
-- | Reads the attribute limits from the @LoggerProvider@ that emitted the @LogRecord@. These are needed to add more attributes.
readLogRecordAttributeLimits :: r a -> AttributeLimits


-- | Modifies the @LogRecord@ using its internal @IORef@. This is lazy and is not an atomic operation. The implementation mirrors @modifyIORef@.
modifyLogRecord :: r a -> (ImmutableLogRecord a -> ImmutableLogRecord a) -> IO ()


-- | An atomic version of @modifyLogRecord@. This function is lazy. The implementation mirrors @atomicModifyIORef@.
atomicModifyLogRecord :: r a -> (ImmutableLogRecord a -> (ImmutableLogRecord a, b)) -> IO b


instance ReadableLogRecord LogRecord where
readLogRecord (LogRecord ref) = readIORef ref
readLogRecord (LogRecord _ ref) = readIORef ref
readLogRecordInstrumentationScope (LogRecord (Logger {loggerInstrumentationScope}) _) = loggerInstrumentationScope
readLogRecordResource (LogRecord Logger {loggerProvider = LoggerProvider {loggerProviderResource}} _) = loggerProviderResource


instance ReadWriteLogRecord LogRecord where
modifyLogRecord (LogRecord ref) = modifyIORef ref
atomicModifyLogRecord (LogRecord ref) = atomicModifyIORef ref
readLogRecordAttributeLimits (LogRecord Logger {loggerProvider = LoggerProvider {loggerProviderAttributeLimits}} _) = loggerProviderAttributeLimits
modifyLogRecord (LogRecord _ ref) = modifyIORef ref
atomicModifyLogRecord (LogRecord _ ref) = atomicModifyIORef ref


data ImmutableLogRecord body = ImmutableLogRecord
Expand Down Expand Up @@ -134,17 +181,10 @@ data ImmutableLogRecord body = ImmutableLogRecord
-- - A byte array,
-- - An array (a list) of any values,
-- - A map<string, any>.
, logRecordResource :: MaterializedResources
-- ^ Describes the source of the log, aka resource. Multiple occurrences of events coming from the same event source can happen across time and they all have the same value of Resource.
-- Can contain for example information about the application that emits the record or about the infrastructure where the application runs. Data formats that represent this data model
-- may be designed in a manner that allows the Resource field to be recorded only once per batch of log records that come from the same source. SHOULD follow OpenTelemetry semantic conventions for Resources.
-- This field is optional.
, logRecordInstrumentationScope :: InstrumentationLibrary
, logRecordAttributes :: LogAttributes
-- ^ Additional information about the specific event occurrence. Unlike the Resource field, which is fixed for a particular source, Attributes can vary for each occurrence of the event coming from the same source.
-- Can contain information about the request context (other than Trace Context Fields). The log attribute model MUST support any type, a superset of standard Attribute, to preserve the semantics of structured attributes
-- emitted by the applications. This field is optional.
, logRecordLogger :: Logger
}
deriving (Functor)

Expand Down
77 changes: 40 additions & 37 deletions api/src/OpenTelemetry/Logging/Core.hs
Original file line number Diff line number Diff line change
Expand Up @@ -23,13 +23,13 @@ module OpenTelemetry.Logging.Core (
SeverityNumber (..),
toShortName,
emitLogRecord,
addAttribute,
addAttributes,
logRecordGetAttributes,

-- * Internal logging operations
logDroppedAttributes,
emitOTelLogRecord,
addAttribute,
addAttributes,
logRecordGetAttributes,
) where

import Control.Applicative
Expand Down Expand Up @@ -123,10 +123,10 @@ makeLogger loggerProvider loggerInstrumentationScope = Logger {..}

createImmutableLogRecord
:: (MonadIO m)
=> Logger
=> LA.AttributeLimits
-> LogRecordArguments body
-> m (ImmutableLogRecord body)
createImmutableLogRecord logger@Logger {..} LogRecordArguments {..} = do
createImmutableLogRecord attributeLimits LogRecordArguments {..} = do
currentTimestamp <- getCurrentTimestamp
let logRecordObservedTimestamp = fromMaybe currentTimestamp observedTimestamp

Expand All @@ -138,7 +138,7 @@ createImmutableLogRecord logger@Logger {..} LogRecordArguments {..} = do

let logRecordAttributes =
LA.addAttributes
(loggerProviderAttributeLimits loggerProvider)
attributeLimits
LA.emptyAttributes
attributes

Expand All @@ -152,10 +152,7 @@ createImmutableLogRecord logger@Logger {..} LogRecordArguments {..} = do
, logRecordSeverityNumber = severityNumber
, logRecordSeverityText = severityText <|> (toShortName =<< severityNumber)
, logRecordBody = body
, logRecordResource = loggerProviderResource loggerProvider
, logRecordInstrumentationScope = loggerInstrumentationScope
, logRecordAttributes
, logRecordLogger = logger
}


Expand Down Expand Up @@ -194,12 +191,14 @@ emitLogRecord
-> LogRecordArguments body
-> m (LogRecord body)
emitLogRecord l args = do
ilr <- createImmutableLogRecord l args
liftIO $ mkLogRecord ilr
ilr <- createImmutableLogRecord (loggerProviderAttributeLimits $ loggerProvider l) args
liftIO $ mkLogRecord l ilr


{- | Add an attribute to a @LogRecord@.
This is not an atomic modification
As an application developer when you need to record an attribute first consult existing semantic conventions for Resources, Spans, and Metrics. If an appropriate name does not exists you will need to come up with a new name. To do that consider a few options:
The name is specific to your company and may be possibly used outside the company as well. To avoid clashes with names introduced by other companies (in a distributed system that uses applications from multiple vendors) it is recommended to prefix the new name by your company’s reverse domain name, e.g. 'com.acme.shopname'.
Expand All @@ -218,39 +217,43 @@ Any additions to the 'otel.*' namespace MUST be approved as part of OpenTelemetr
-}
addAttribute :: (ReadWriteLogRecord r, MonadIO m, ToValue a) => r body -> Text -> a -> m ()
addAttribute lr k v =
liftIO $
modifyLogRecord
lr
( \ilr@ImmutableLogRecord {logRecordAttributes, logRecordLogger} ->
ilr
{ logRecordAttributes =
LA.addAttribute
(loggerProviderAttributeLimits $ loggerProvider logRecordLogger)
logRecordAttributes
k
v
}
)
let attributeLimits = readLogRecordAttributeLimits lr
in liftIO $
modifyLogRecord
lr
( \ilr@ImmutableLogRecord {logRecordAttributes} ->
ilr
{ logRecordAttributes =
LA.addAttribute
attributeLimits
logRecordAttributes
k
v
}
)


{- | A convenience function related to 'addAttribute' that adds multiple attributes to a @LogRecord@ at the same time.
This function may be slightly more performant than repeatedly calling 'addAttribute'.
This function may be slightly more performant than repeatedly calling 'addAttribute'.
This is not an atomic modification
-}
addAttributes :: (ReadWriteLogRecord r, MonadIO m, ToValue a) => r body -> HashMap Text a -> m ()
addAttributes lr attrs =
liftIO $
modifyLogRecord
lr
( \ilr@ImmutableLogRecord {logRecordAttributes, logRecordLogger} ->
ilr
{ logRecordAttributes =
LA.addAttributes
(loggerProviderAttributeLimits $ loggerProvider logRecordLogger)
logRecordAttributes
attrs
}
)
let attributeLimits = readLogRecordAttributeLimits lr
in liftIO $
modifyLogRecord
lr
( \ilr@ImmutableLogRecord {logRecordAttributes} ->
ilr
{ logRecordAttributes =
LA.addAttributes
attributeLimits
logRecordAttributes
attrs
}
)


{- | This can be useful for pulling data for attributes and
Expand Down

0 comments on commit ec5674e

Please sign in to comment.