Git Product home page Git Product logo

circonus-gometrics's Introduction

Circonus metrics tracking for Go applications

This library supports named counters, gauges and histograms. It also provides convenience wrappers for registering latency instrumented functions with Go's builtin http server.

Initializing only requires setting an API Token at a minimum.

Options

See OPTIONS.md for information on all of the available cgm options.

Example

Bare bones minimum

A working cut-n-past example. Simply set the required environment variable CIRCONUS_API_TOKEN and run.

package main

import (
    "log"
    "math/rand"
    "os"
    "os/signal"
    "syscall"
    "time"

    cgm "github.com/circonus-labs/circonus-gometrics/v3"
)

func main() {

    logger := log.New(os.Stdout, "", log.LstdFlags)

    logger.Println("Configuring cgm")

    cmc := &cgm.Config{}
    cmc.Debug = false // set to true for debug messages
    cmc.Log = logger

    // Circonus API Token key (https://login.circonus.com/user/tokens)
    cmc.CheckManager.API.TokenKey = os.Getenv("CIRCONUS_API_TOKEN")

    logger.Println("Creating new cgm instance")

    metrics, err := cgm.NewCirconusMetrics(cmc)
    if err != nil {
        logger.Println(err)
        os.Exit(1)
    }

    src := rand.NewSource(time.Now().UnixNano())
    rnd := rand.New(src)

    logger.Println("Adding ctrl-c trap")
    c := make(chan os.Signal, 2)
    signal.Notify(c, os.Interrupt, syscall.SIGTERM)
    go func() {
        <-c
        logger.Println("Received CTRL-C, flushing outstanding metrics before exit")
        metrics.Flush()
        os.Exit(0)
    }()

    logger.Println("Starting to send metrics")

    // number of "sets" of metrics to send
    max := 60

    for i := 1; i < max; i++ {
        logger.Printf("\tmetric set %d of %d", i, 60)
        metrics.Timing("foo", rnd.Float64()*10)
        metrics.Increment("bar")
        metrics.Gauge("baz", 10)
        time.Sleep(time.Second)
    }

    metrics.SetText("fini", "complete")

    logger.Println("Flushing any outstanding metrics manually")
    metrics.Flush()
}

A more complete example

A working, cut-n-paste example with all options available for modification. Also, demonstrates metric tagging.

package main

import (
    "log"
    "math/rand"
    "os"
    "os/signal"
    "syscall"
    "time"

    cgm "github.com/circonus-labs/circonus-gometrics/v3"
)

func main() {

    logger := log.New(os.Stdout, "", log.LstdFlags)

    logger.Println("Configuring cgm")

    cmc := &cgm.Config{}

    // General

    cmc.Interval = "10s"
    cmc.Log = logger
    cmc.Debug = false
    cmc.ResetCounters = "true"
    cmc.ResetGauges = "true"
    cmc.ResetHistograms = "true"
    cmc.ResetText = "true"

    // Circonus API configuration options
    cmc.CheckManager.API.TokenKey = os.Getenv("CIRCONUS_API_TOKEN")
    cmc.CheckManager.API.TokenApp = os.Getenv("CIRCONUS_API_APP")
    cmc.CheckManager.API.URL = os.Getenv("CIRCONUS_API_URL")
    cmc.CheckManager.API.TLSConfig = nil

    // Check configuration options
    cmc.CheckManager.Check.SubmissionURL = os.Getenv("CIRCONUS_SUBMISSION_URL")
    cmc.CheckManager.Check.ID = os.Getenv("CIRCONUS_CHECK_ID")
    cmc.CheckManager.Check.InstanceID = ""
    cmc.CheckManager.Check.DisplayName = ""
    cmc.CheckManager.Check.TargetHost = ""
    //  if hn, err := os.Hostname(); err == nil {
    //      cmc.CheckManager.Check.TargetHost = hn
    //  }
    cmc.CheckManager.Check.SearchTag = ""
    cmc.CheckManager.Check.Secret = ""
    cmc.CheckManager.Check.Tags = ""
    cmc.CheckManager.Check.MaxURLAge = "5m"
    cmc.CheckManager.Check.ForceMetricActivation = "false"

    // Broker configuration options
    cmc.CheckManager.Broker.ID = ""
    cmc.CheckManager.Broker.SelectTag = ""
    cmc.CheckManager.Broker.MaxResponseTime = "500ms"
    cmc.CheckManager.Broker.TLSConfig = nil

    logger.Println("Creating new cgm instance")

    metrics, err := cgm.NewCirconusMetrics(cmc)
    if err != nil {
        logger.Println(err)
        os.Exit(1)
    }

    src := rand.NewSource(time.Now().UnixNano())
    rnd := rand.New(src)

    logger.Println("Adding ctrl-c trap")
    c := make(chan os.Signal, 2)
    signal.Notify(c, os.Interrupt, syscall.SIGTERM)
    go func() {
        <-c
        logger.Println("Received CTRL-C, flushing outstanding metrics before exit")
        metrics.Flush()
        os.Exit(0)
    }()

    // Add metric tags (append to any existing tags on specified metric)
    metrics.AddMetricTags("foo", []string{"cgm:test"})
    metrics.AddMetricTags("baz", []string{"cgm:test"})

    logger.Println("Starting to send metrics")

    // number of "sets" of metrics to send
    max := 60

    for i := 1; i < max; i++ {
        logger.Printf("\tmetric set %d of %d", i, 60)

        metrics.Timing("foo", rnd.Float64()*10)
        metrics.Increment("bar")
        metrics.Gauge("baz", 10)

        if i == 35 {
            // Set metric tags (overwrite current tags on specified metric)
            metrics.SetMetricTags("baz", []string{"cgm:reset_test", "cgm:test2"})
        }

        time.Sleep(time.Second)
    }

    logger.Println("Flushing any outstanding metrics manually")
    metrics.Flush()

}

HTTP Handler wrapping

http.HandleFunc("/", metrics.TrackHTTPLatency("/", handler_func))

HTTP latency example

package main

import (
    "os"
    "fmt"
    "net/http"
    cgm "github.com/circonus-labs/circonus-gometrics/v3"
)

func main() {
    cmc := &cgm.Config{}
    cmc.CheckManager.API.TokenKey = os.Getenv("CIRCONUS_API_TOKEN")

    metrics, err := cgm.NewCirconusMetrics(cmc)
    if err != nil {
        panic(err)
    }

    http.HandleFunc("/", metrics.TrackHTTPLatency("/", func(w http.ResponseWriter, r *http.Request) {
        fmt.Fprintf(w, "Hello, %s!", r.URL.Path[1:])
    }))
    http.ListenAndServe(":8080", http.DefaultServeMux)
}

Unless otherwise noted, the source files are distributed under the BSD-style license found in the LICENSE file.

circonus-gometrics's People

Contributors

dhaifley avatar jabley avatar jefferai avatar madsensix avatar maier avatar postwait avatar redhotpenguin avatar sean- avatar yargevad avatar

Stargazers

 avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar

Watchers

 avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar

circonus-gometrics's Issues

Data race on submission URL

I have a unit test that calls WithSubmissionUrl to set a new target, but a previous unit test has already called Start, and there's no way to terminate that goroutine. So, I'm writing over the string at the same time trapCall is reading it.

WARNING: DATA RACE
Write by goroutine 23:
  github.com/go-kit/kit/metrics/circonus.TestGauge()
      /home/travis/gopath/src/github.com/go-kit/kit/metrics/circonus/circonus_test.go:106 +0x295
  testing.tRunner()
      /tmp/workdir/go/src/testing/testing.go:456 +0xdc

Previous read by goroutine 10:
  github.com/circonus-labs/circonus-gometrics.trapCall()
      /home/travis/gopath/src/github.com/circonus-labs/circonus-gometrics/circonus-gometrics.go:340 +0x14a
  github.com/circonus-labs/circonus-gometrics.submit()
      /home/travis/gopath/src/github.com/circonus-labs/circonus-gometrics/circonus-gometrics.go:230 +0x93
  github.com/circonus-labs/circonus-gometrics.Start.func1()
      /home/travis/gopath/src/github.com/circonus-labs/circonus-gometrics/circonus-gometrics.go:396 +0x974

The fastest fix is to wrap all access of package globals with mutexes. The better fix is to stop using package globals :)

Test failed on Go 1.15

--- FAIL: TestNew (0.00s)
    circonus-gometrics_test.go:65: invalid config (none)
    circonus-gometrics_test.go:74: no API token, no submission URL
    circonus-gometrics_test.go:84: no API token, submission URL only
    circonus-gometrics_test.go:109: no Log, Debug = true
    circonus-gometrics_test.go:121: flush interval [good]
    circonus-gometrics_test.go:132: flush interval [bad]
    circonus-gometrics_test.go:143: Expected parsing flush interval: time: invalid duration thirty seconds got 'parsing flush interval: time: invalid duration "thirty seconds"'
FAIL

Data race

From the Go kit test suite.

WARNING: DATA RACE
Write at 0x00c42010a020 by goroutine 8:
  github.com/circonus-labs/circonus-gometrics/checkmgr.(*CheckManager).Initialize.func1()
      /home/travis/gopath/src/github.com/circonus-labs/circonus-gometrics/checkmgr/checkmgr.go:336 +0x1ba

Previous read at 0x00c42010a020 by goroutine 6:
  github.com/circonus-labs/circonus-gometrics.(*CirconusMetrics).submit()
      /home/travis/gopath/src/github.com/circonus-labs/circonus-gometrics/submit.go:26 +0x7a
  github.com/circonus-labs/circonus-gometrics.(*CirconusMetrics).Flush()
      /home/travis/gopath/src/github.com/circonus-labs/circonus-gometrics/circonus-gometrics.go:313 +0x87a
  github.com/go-kit/kit/metrics/circonus.TestCounter.func2()
      /home/travis/gopath/src/github.com/go-kit/kit/metrics/circonus/circonus_test.go:35 +0x4a
  github.com/go-kit/kit/metrics/teststat.TestCounter()
      /home/travis/gopath/src/github.com/go-kit/kit/metrics/teststat/teststat.go:27 +0x127
  github.com/go-kit/kit/metrics/circonus.TestCounter()
      /home/travis/gopath/src/github.com/go-kit/kit/metrics/circonus/circonus_test.go:38 +0x347
  testing.tRunner()
      /home/travis/.gimme/versions/go1.8beta2.linux.amd64/src/testing/testing.go:656 +0x104

Basically, you're writing the checkmgr.initialized value in the Initialize method, and reading it in the IsReady method (and many others) without any kind of synchronization. You need a mutex or something.

I bet if you run your test suite with the -race flag, you'll see a bunch of these.

server.Certificate undefined

Hello !
When I try to test api (and checkmgr), it fails with

FAIL github.com/circonus-labs/circonus-gometrics/api [build failed]

github.com/circonus-labs/circonus-gometrics/api

./api_test.go:303: server.Certificate undefined (type *httptest.Server has no field or method Certificate)
./api_test.go:337: server.Certificate undefined (type *httptest.Server has no field or method Certificate)

Thank you

client.CheckRetry undefined

Haven't quite figured out why this is happening, getting a test failure with go version go1.7.3 darwin/amd64. All of the go-retryablehttp tests pass. Any pointers welcome.

phred@sisu ~/gocode/src/github.com/circonus-labs/circonus-gometrics $ go test
# github.com/circonus-labs/circonus-gometrics/api
api/api.go:205: client.CheckRetry undefined (type *retryablehttp.Client has no field or method CheckRetry)
FAIL	github.com/circonus-labs/circonus-gometrics [build failed]

Multiple race conditions

==================
WARNING: DATA RACE
Read by goroutine 10:
  github.com/circonus-labs/circonus-gometrics.snapshot()
      /Users/peter/src/github.com/circonus-labs/circonus-gometrics/circonus-gometrics.go:158 +0x7b6
  github.com/circonus-labs/circonus-gometrics.Start.func1()
      /Users/peter/src/github.com/circonus-labs/circonus-gometrics/circonus-gometrics.go:376 +0x10f

Previous write by goroutine 33:
  github.com/circonus-labs/circonusllhist.(*Histogram).InsertBin()
      /Users/peter/src/github.com/circonus-labs/circonusllhist/circonusllhist.go:359 +0x5c8
  github.com/circonus-labs/circonusllhist.(*Histogram).RecordValues()
      /Users/peter/src/github.com/circonus-labs/circonusllhist/circonusllhist.go:380 +0x77
  github.com/circonus-labs/circonusllhist.(*Histogram).RecordValue()
      /Users/peter/src/github.com/circonus-labs/circonusllhist/circonusllhist.go:273 +0x4f
  github.com/circonus-labs/circonus-gometrics.(*Histogram).RecordValue()
      /Users/peter/src/github.com/circonus-labs/circonus-gometrics/circonus-gometrics.go:207 +0xa8
  github.com/go-kit/kit/metrics/circonus.histogram.Observe()
      /Users/peter/src/github.com/go-kit/kit/metrics/circonus/circonus.go:98 +0x3e
  github.com/go-kit/kit/metrics/teststat.PopulateNormalHistogram()
      /Users/peter/src/github.com/go-kit/kit/metrics/teststat/common.go:24 +0xef
  github.com/go-kit/kit/metrics/circonus.TestHistogram()
      /Users/peter/src/github.com/go-kit/kit/metrics/circonus/circonus_test.go:169 +0x3f5
  testing.tRunner()
      /usr/local/Cellar/go/1.6.2/libexec/src/testing/testing.go:473 +0xdc

Goroutine 10 (running) created at:
  github.com/circonus-labs/circonus-gometrics.Start()
      /Users/peter/src/github.com/circonus-labs/circonus-gometrics/circonus-gometrics.go:398 +0x38
  github.com/go-kit/kit/metrics/circonus.TestCounter.func2()
      /Users/peter/src/github.com/go-kit/kit/metrics/circonus/circonus_test.go:71 +0x25
  sync.(*Once).Do()
      /usr/local/Cellar/go/1.6.2/libexec/src/sync/once.go:44 +0xf6
  github.com/go-kit/kit/metrics/circonus.TestCounter()
      /Users/peter/src/github.com/go-kit/kit/metrics/circonus/circonus_test.go:71 +0x529
  testing.tRunner()
      /usr/local/Cellar/go/1.6.2/libexec/src/testing/testing.go:473 +0xdc

Goroutine 33 (running) created at:
  testing.RunTests()
      /usr/local/Cellar/go/1.6.2/libexec/src/testing/testing.go:582 +0xae2
  testing.(*M).Run()
      /usr/local/Cellar/go/1.6.2/libexec/src/testing/testing.go:515 +0x11d
  main.main()
      github.com/go-kit/kit/metrics/circonus/_test/_testmain.go:58 +0x210
==================
==================
WARNING: DATA RACE
Write by goroutine 10:
  github.com/circonus-labs/circonus-gometrics.snapshot()
      /Users/peter/src/github.com/circonus-labs/circonus-gometrics/circonus-gometrics.go:158 +0x8a1
  github.com/circonus-labs/circonus-gometrics.Start.func1()
      /Users/peter/src/github.com/circonus-labs/circonus-gometrics/circonus-gometrics.go:376 +0x10f

Previous read by goroutine 33:
  github.com/circonus-labs/circonusllhist.(*Histogram).InternalFind()
      /Users/peter/src/github.com/circonus-labs/circonusllhist/circonusllhist.go:313 +0xd0
  github.com/circonus-labs/circonusllhist.(*Histogram).InsertBin()
      /Users/peter/src/github.com/circonus-labs/circonusllhist/circonusllhist.go:341 +0x74
  github.com/circonus-labs/circonusllhist.(*Histogram).RecordValues()
      /Users/peter/src/github.com/circonus-labs/circonusllhist/circonusllhist.go:380 +0x77
  github.com/circonus-labs/circonusllhist.(*Histogram).RecordValue()
      /Users/peter/src/github.com/circonus-labs/circonusllhist/circonusllhist.go:273 +0x4f
  github.com/circonus-labs/circonus-gometrics.(*Histogram).RecordValue()
      /Users/peter/src/github.com/circonus-labs/circonus-gometrics/circonus-gometrics.go:207 +0xa8
  github.com/go-kit/kit/metrics/circonus.histogram.Observe()
      /Users/peter/src/github.com/go-kit/kit/metrics/circonus/circonus.go:98 +0x3e
  github.com/go-kit/kit/metrics/teststat.PopulateNormalHistogram()
      /Users/peter/src/github.com/go-kit/kit/metrics/teststat/common.go:24 +0xef
  github.com/go-kit/kit/metrics/circonus.TestHistogram()
      /Users/peter/src/github.com/go-kit/kit/metrics/circonus/circonus_test.go:169 +0x3f5
  testing.tRunner()
      /usr/local/Cellar/go/1.6.2/libexec/src/testing/testing.go:473 +0xdc

Goroutine 10 (running) created at:
  github.com/circonus-labs/circonus-gometrics.Start()
      /Users/peter/src/github.com/circonus-labs/circonus-gometrics/circonus-gometrics.go:398 +0x38
  github.com/go-kit/kit/metrics/circonus.TestCounter.func2()
      /Users/peter/src/github.com/go-kit/kit/metrics/circonus/circonus_test.go:71 +0x25
  sync.(*Once).Do()
      /usr/local/Cellar/go/1.6.2/libexec/src/sync/once.go:44 +0xf6
  github.com/go-kit/kit/metrics/circonus.TestCounter()
      /Users/peter/src/github.com/go-kit/kit/metrics/circonus/circonus_test.go:71 +0x529
  testing.tRunner()
      /usr/local/Cellar/go/1.6.2/libexec/src/testing/testing.go:473 +0xdc

Goroutine 33 (running) created at:
  testing.RunTests()
      /usr/local/Cellar/go/1.6.2/libexec/src/testing/testing.go:582 +0xae2
  testing.(*M).Run()
      /usr/local/Cellar/go/1.6.2/libexec/src/testing/testing.go:515 +0x11d
  main.main()
      github.com/go-kit/kit/metrics/circonus/_test/_testmain.go:58 +0x210
==================
==================
WARNING: DATA RACE
Read by goroutine 10:
  github.com/circonus-labs/circonusllhist.(*Histogram).DecStrings()
      /Users/peter/src/github.com/circonus-labs/circonusllhist/circonusllhist.go:524 +0x19d
  github.com/circonus-labs/circonus-gometrics.Start.func1()
      /Users/peter/src/github.com/circonus-labs/circonus-gometrics/circonus-gometrics.go:393 +0x787

Previous write by goroutine 33:
  github.com/circonus-labs/circonusllhist.(*Histogram).InsertBin()
      /Users/peter/src/github.com/circonus-labs/circonusllhist/circonusllhist.go:371 +0xa04
  github.com/circonus-labs/circonusllhist.(*Histogram).RecordValues()
      /Users/peter/src/github.com/circonus-labs/circonusllhist/circonusllhist.go:380 +0x77
  github.com/circonus-labs/circonusllhist.(*Histogram).RecordValue()
      /Users/peter/src/github.com/circonus-labs/circonusllhist/circonusllhist.go:273 +0x4f
  github.com/circonus-labs/circonus-gometrics.(*Histogram).RecordValue()
      /Users/peter/src/github.com/circonus-labs/circonus-gometrics/circonus-gometrics.go:207 +0xa8
  github.com/go-kit/kit/metrics/circonus.histogram.Observe()
      /Users/peter/src/github.com/go-kit/kit/metrics/circonus/circonus.go:98 +0x3e
  github.com/go-kit/kit/metrics/teststat.PopulateNormalHistogram()
      /Users/peter/src/github.com/go-kit/kit/metrics/teststat/common.go:24 +0xef
  github.com/go-kit/kit/metrics/circonus.TestHistogram()
      /Users/peter/src/github.com/go-kit/kit/metrics/circonus/circonus_test.go:169 +0x3f5
  testing.tRunner()
      /usr/local/Cellar/go/1.6.2/libexec/src/testing/testing.go:473 +0xdc

Goroutine 10 (running) created at:
  github.com/circonus-labs/circonus-gometrics.Start()
      /Users/peter/src/github.com/circonus-labs/circonus-gometrics/circonus-gometrics.go:398 +0x38
  github.com/go-kit/kit/metrics/circonus.TestCounter.func2()
      /Users/peter/src/github.com/go-kit/kit/metrics/circonus/circonus_test.go:71 +0x25
  sync.(*Once).Do()
      /usr/local/Cellar/go/1.6.2/libexec/src/sync/once.go:44 +0xf6
  github.com/go-kit/kit/metrics/circonus.TestCounter()
      /Users/peter/src/github.com/go-kit/kit/metrics/circonus/circonus_test.go:71 +0x529
  testing.tRunner()
      /usr/local/Cellar/go/1.6.2/libexec/src/testing/testing.go:473 +0xdc

Goroutine 33 (running) created at:
  testing.RunTests()
      /usr/local/Cellar/go/1.6.2/libexec/src/testing/testing.go:582 +0xae2
  testing.(*M).Run()
      /usr/local/Cellar/go/1.6.2/libexec/src/testing/testing.go:515 +0x11d
  main.main()
      github.com/go-kit/kit/metrics/circonus/_test/_testmain.go:58 +0x210
==================

Test error with Go 1.14beta1

Fedora Rawhide x86_64, with Go 1.14beta1, and circonus-gometrics 3.0.0 beta4:

Testing    in: /builddir/build/BUILD/circonus-gometrics-c6f494a5b139fc62376f353711f5e4abce77f2b1/_build/src
         PATH: /builddir/build/BUILD/circonus-gometrics-c6f494a5b139fc62376f353711f5e4abce77f2b1/_build/bin:/usr/bin:/bin:/usr/sbin:/sbin:/usr/local/sbin
       GOPATH: /builddir/build/BUILD/circonus-gometrics-c6f494a5b139fc62376f353711f5e4abce77f2b1/_build:/usr/share/gocode
  GO111MODULE: off
      command: go test -buildmode pie -compiler gc -ldflags "-X github.com/circonus-labs/circonus-gometrics/version=3.0.0 -X github.com/circonus-labs/circonus-gometrics/version.commit=c6f494a5b139fc62376f353711f5e4abce77f2b1 -extldflags '-Wl,-z,relro -Wl,--as-needed  -Wl,-z,now -specs=/usr/lib/rpm/redhat/redhat-hardened-ld '"
      testing: github.com/circonus-labs/circonus-gometrics
github.com/circonus-labs/circonus-gometrics
PASS
ok  	github.com/circonus-labs/circonus-gometrics	0.015s
github.com/circonus-labs/circonus-gometrics/api
2020/01/28 23:50:32 circonus-gometrics/api is DEPRECATED and will be removed - switch to github.com/circonus-labs/go-apiclient
--- FAIL: TestNew (0.00s)
    api_test.go:70: invalid config [nil]
    api_test.go:82: invalid config [blank]
    api_test.go:95: API Token, no API App, no API URL
    api_test.go:106: API Token, API App, no API URL
    api_test.go:118: API Token, API App, API Account ID, no API URL
    api_test.go:131: API Token, API App, API URL [host]
    api_test.go:144: API Token, API App, API URL [trailing '/']
    api_test.go:157: API Token, API App, API URL [w/o trailing '/']
    api_test.go:170: API Token, API App, API URL [invalid]
    api_test.go:183: Expected an '&errors.errorString{s:"parse http://something.somewhere.com\\somepath$: invalid character \"\\\\\" in host name"}' error, got '&url.Error{Op:"parse", URL:"http://something.somewhere.com\\somepath$", Err:"\\"}'
    api_test.go:187: Debug true, no log.Logger
FAIL
exit status 1
FAIL	github.com/circonus-labs/circonus-gometrics/api	29.162s

Refactor makes some things very difficult

Previously:

import "github.com/circonus-labs/circonus-gometrics"

m := circonusgometrics.NewCirconusMetrics()
m.SubmissionUrl = s.URL

Now:

import (
    "github.com/circonus-labs/circonus-gometrics"
    "github.com/circonus-labs/circonus-gometrics/checkmgr"
)

m, _ := circonusgometrics.NewCirconusMetrics(&circonusgometrics.Config{
    CheckManager: checkmgr.Config{
        Check: checkmgr.CheckConfig{
            SubmissionUrl: url,
        },
    },
})

Are you sure this is a good idea? :)

edit: got the initialization wrong

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    ๐Ÿ–– Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. ๐Ÿ“Š๐Ÿ“ˆ๐ŸŽ‰

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google โค๏ธ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.