Skip to content

Commit

Permalink
Merge pull request #7 from teamjorge/feature-add-loader-example
Browse files Browse the repository at this point in the history
Feature add loader example
  • Loading branch information
teamjorge authored Aug 21, 2024
2 parents 4bc4a47 + 71271d7 commit 912f9e8
Show file tree
Hide file tree
Showing 7 changed files with 156 additions and 18 deletions.
17 changes: 3 additions & 14 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -30,25 +30,14 @@ This package will not parse real-time telemetry as that requires opening a memor

## Examples

Please see the [`examples`](https://github.com/teamjorge/ibt/tree/main/examples) folder for detailed usage instructions.
The [Examples](https://github.com/teamjorge/ibt/tree/main/examples) directory houses all of the available examples.

To try the examples locally, please clone to repository:
To try these examples locally, please clone to repository:

```shell
git clone https://github.com/teamjorge/ibt
#or
git clone [email protected]:teamjorge/ibt.git

cd ibt
```

To run the example which summarises the track temperature per lap:

```shell
go run examples/track_temp/main.go

# Or to run it with your own telemetry files

go run examples/track_temp/main.go /path/to/telem/files/*.ibt
```

Please have a look at the instructors in the examples [`README`](./examples/README.md) for details on how to run each example.
4 changes: 2 additions & 2 deletions examples/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,6 @@ All examples allow you to run with the supplied `ibt` file or with your own.

For example:


```shell
go run examples/[Example Folder]/main.go

Expand All @@ -15,4 +14,5 @@ go run examples/[Example Folder]/main.go /path/to/telem/files/*.ibt

Available examples:

* [track temperature](./track_temp/README.md) -
* [track temperature](./track_temp/README.md) - Track temperature per lap summarising
* [loader](./loader/README.md) - Loading of telemetry data to an external destination
25 changes: 25 additions & 0 deletions examples/loader/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
# loader

## Overview

The `loader` example shows a pattern for loading telemetry data to an external destination. This could be loading it into a database, posting to an API, or even storing it in a file.

The practices in this example should not be copied verbatim, but should show how `ibt` can be used to achieve this.

Goals of the example:

* Parse the `ibt` files into groups
* For each group of `ibts` process each tick of telemetry data
* When the threshold of processed telemetry ticks have been reached, perform a bulk load to the storage client
* Add a group number to each telemetry tick to ensure they are easily filtered in the external storage layer
* Store the number of batches loaded and print it after processing each group

## Running

```shell
go run examples/track_temp/*.go

# Or with your own files

go run examples/track_temp/*.go /path/to/telem/files/*.ibt
```
43 changes: 43 additions & 0 deletions examples/loader/main.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
package main

import (
"context"
"log"

"github.com/teamjorge/ibt"
"github.com/teamjorge/ibt/examples"
)

func main() {
// Parse the files into stubs
stubs, err := examples.ParseExampleStubs()
if err != nil {
log.Fatal(err)
}

// Create our storage client
storage := newStorage()
if err := storage.Connect(); err != nil {
log.Fatal(err)
}
// Close it when the application ends
defer storage.Close()

// We group our stubs mainly to be able to identify the batches we are loading
// This might not be necessary on your use case
groups := stubs.Group()

for groupNumber, group := range groups {
// Create a new processor for this group and set the groupNumber.
// It embeds our storage and we set our loading threshold to 100
processor := newLoaderProcessor(storage, groupNumber, 100)

// Process the group
if err := ibt.Process(context.Background(), group, processor); err != nil {
log.Fatalf("failed to process telemetry for stubs %v: %v", stubs, err)
}

// Print the number of batches loaded after each group
log.Printf("%d batches loaded after group %d\n", storage.Loaded(), groupNumber)
}
}
22 changes: 22 additions & 0 deletions examples/loader/mock_storage.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
package main

// This is a mock external storage client.
//
// Think of it as a database, API, or external file
type storage struct {
batchesLoaded int
}

// Simple constructor
func newStorage() *storage { return new(storage) }

func (s *storage) Connect() error { return nil }

func (s *storage) Exec(data []map[string]interface{}) error {
s.batchesLoaded += len(data)
return nil
}

func (s *storage) Close() error { return nil }

func (s *storage) Loaded() int { return s.batchesLoaded }
57 changes: 57 additions & 0 deletions examples/loader/processors.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
package main

import (
"fmt"

"github.com/teamjorge/ibt"
"github.com/teamjorge/ibt/headers"
)

type loaderProcessor struct {
// Our storage client
*storage
// Cache for holding the number of telemetry ticks equal to threshold
cache []map[string]interface{}
// Number
groupNumber int
threshold int
}

// Simple Constructor for creating our processor
func newLoaderProcessor(storage *storage, groupNumber int, threshold int) *loaderProcessor {
return &loaderProcessor{storage, make([]map[string]interface{}, 0), groupNumber, threshold}
}

// Columns we want to parse from telemetry
func (l *loaderProcessor) Whitelist() []string {
return []string{
"Lap", "ThrottleRaw", "BrakeRaw", "ClutchRaw", "LapDistPct", "Lat", "Lon",
}
}

// Our method for processing a single tick of telemetry.
func (l *loaderProcessor) Process(input ibt.Tick, hasNext bool, session *headers.Session) error {
// Add our group number to the tick of telemetry.
// This will be useful to seperate ticks by group in our storage.
input["groupNum"] = l.groupNumber

// Add it to the cache
l.cache = append(l.cache, input)

// If our cache is past the threshold, that means we can now do a bulk load
// to our storage.
if len(l.cache) >= l.threshold {
if err := l.loadBatch(); err != nil {
return fmt.Errorf("failed to load batch - %v", err)
}
// Empty the cache again
l.cache = make([]map[string]interface{}, 0)
}

return nil
}

func (l *loaderProcessor) loadBatch() error {
// Bulk load our batch to storage.
return l.Exec(l.cache)
}
6 changes: 4 additions & 2 deletions examples/track_temp/README.md
Original file line number Diff line number Diff line change
@@ -1,17 +1,19 @@
# track_temp

## Overview

The `track_temp` example shows a simple processor summarising the track temperature on each lap of the provided `ibt` files.

## Running

From the root of the repository:

```shell
go run examples/track_temp/main.go
go run examples/track_temp/*.go

# Or with your own files

go run examples/track_temp/main.go /path/to/telem/files/*.ibt
go run examples/track_temp/*.go /path/to/telem/files/*.ibt
```

Using the included `ibt` file will yield only a single lap and it's value. However, if you have telemetry consisting of a few laps and/or files from a longer session, you should have a nicely summarised per-lap output.

0 comments on commit 912f9e8

Please sign in to comment.