Skip to content

Latest commit

 

History

History
898 lines (734 loc) · 26 KB

README.md

File metadata and controls

898 lines (734 loc) · 26 KB

meowtup-tarantool-2021

Небольшое приложение на Tarantoo, а также golang и html/js.

Мотивация

Представим, что хочется, чтобы котики чувствовали себя свободно, но и их хозяева не переживали. Сделаем сервис с отображением котиков на карте.

Технически

Данные будем хранить и индексировать в Tarantool Обрабатывать запросы будет в golang Рисовать карту на html/js

Начинаем строить приложение

Вот что нам нужно сделать:

  1. Фронтенд на HTML/JS с Leaflet и OpenStreetMap
    • Виджет с картой
    • События пользователя
    • Запросы к Golang-бекенду
  2. Создание Golang приложения
    • Подключение к базе
    • Индексирование
    • Запросы к базе
    • HTTP-сервер
    • HTTP API
  3. Конфигурация базы данных

Фронтенд

Начнем с html

Сделаем файл index.html

touch index.html
<html>
</html>

Подключим фреймворк для рисования карты — leaflet

<head>
    <title>The Map</title>
    <link rel="stylesheet" href="https://unpkg.com/[email protected]/dist/leaflet.css" crossorigin="" />
    <script src="https://unpkg.com/[email protected]/dist/leaflet.js" crossorigin=""></script>
    <script src="https://unpkg.com/[email protected]/leaflet-providers.js" crossorigin=""></script>
</head>

Нарисуем виджет с картой

<body>
    <!-- div для карты -->
    <div id="mapid" style="height:100%"></div>
    <script>
        // Карта
        var mymap = L.map('mapid',
            { 'tap': false })
            .setView([59.95, 30.31], 13)

        // Слой карты с домами, улицами и т.п.
        var osm = L.tileLayer('https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png', {
            maxZoom: 19,
            attribution: '&copy <a href="https://www.openstreetmap.org/copyright">OpenStreetMap</a> contributors',
        }).addTo(mymap)
    </script>
</body>

Добавим слой и добавим на него тестовый маркер просто чтобы убедится

group = L.layerGroup().addTo(mymap)
L.marker([59.95, 30.31]).addTo(group).bindPopup("Hello World")

Должно всё получиться

Создадим функцию, которая будет добавлять объекты на карту по клику мыши

// Обрабатываем нажатие на карту
function onMapClick(e) {
    L.marker(e.latlng).addTo(group).bindPopup("Hello World")
}
mymap.on('click', onMapClick)

Конфигурация базы данных Lua

  • Redis говорит: «Сконфигурируй меня с помощью простых команд в файле»
  • Mongo говорит: «Сконфигурируй меня с помощью YAML файла»
  • Tarantool говорит: «Используй Lua скрипт»

Чтобы запустить Tarantool и подключиться к нему, нам понадобится всего три Lua-функции:

  • box.cfg
  • box.schema.user.create
  • box.schema.user.grant

box.cfg

Функция настраивает весь Tarantool. Часть параметров можно задать только один раз при старте. Другую часть можно менять в любой момент времени.

box.schema.user.create

Функция создаёт пользователя для удаленной работы.

box.schema.user.grant

Функция перечисляет, что можно и что нельзя будет делать пользователю.

Конфигурация Tarantool-а — единственное место, где будет Lua.

Создадим файл init.lua

touch init.lua

init.lua

-- Открываем порт для доступа по iproto
box.cfg({listen="127.0.0.1:3301"})
-- Создаём пользователя для подключения
box.schema.user.create('storage', {password='passw0rd', if_not_exists=true})
-- Даём все-все права
box.schema.user.grant('storage', 'super', nil, nil, {if_not_exists=true})

На одном узле Tarantool находится только одна база данных. Данные складываются в спейсы == таблицы в мире SQL. К данным обязательно строится первичный индекс, а количество вторичных произвольно.

Для хранения маркеров сделаем таблицу:

id coordinates name
string [double, double] string

В поле id хранится уникальный идентификатор, который мы сами сгенерируем. В поле coordinates — координаты маркера (массив из двух double). В поле name — строка с кличкой.

-- создаём таблицу для хранения отзывов на карте
box.schema.space.create('cats', {if_not_exists=true})
box.space.cats:format({
        {name="id", type="string"},
        {name="coordinates", type="array"},
        {name="name", type="string"}
})
-- создаём первичный индекс
box.space.cats:create_index('primary', {
                                parts={{ field="id", type="string" }},
                                type = 'TREE',
                                if_not_exists=true,})
-- создаём индекс для координат
box.space.cats:create_index('spatial', {
                                parts = {{ field="coordinates", type='array'} },
                                type = 'RTREE',
                                unique = false,
                                if_not_exists=true,})

Запущу локальную консоль внутри базы данных, для всяких тестовых команд

require('console').start() os.exit()

На этом Lua закачивается.

Запуск Tarantool

tarantool init.lua

Golang приложение

Создадим файл cats.go

touch cats.go
package main

import (
	"encoding/json"
	"net/http"

    // Генерация уникальных идентификаторов
	"github.com/chilts/sid"
    // Коннектор к Tarantool
	"github.com/tarantool/go-tarantool"
)

// Структура для сериализации гео объектов в/из Tarantool
type CatObject struct {
	Id          string     `json:"id"`
	Coordinates [2]float64 `json:"coordinates"`
	Name     string     `json:"name"`
}

func main() {
	opts := tarantool.Opts{User: "storage", Pass: "passw0rd"}
	conn, err := tarantool.Connect("127.0.0.1:3301", opts)
	if err != nil {
		panic(err)
	}
	defer conn.Close()
}

Запуск

go run ./cats.go
go mod init cats
go get github.com/tarantool/go-tarantool
go run ./cats.go

Добавим хостинг index.html файла

// В корневом эндпоинте отдаём пользователю фронтенд
http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
    http.ServeFile(w, r, "index.html")
})

// Запускаем http сервер на локальном адресе
err = http.ListenAndServe("127.0.0.1:8080", nil)
if err != nil {
    panic(err)
}

Ендпоинт для сохранения кота

// Эндпоинт для сохранения маркера
http.HandleFunc("/put", func(w http.ResponseWriter, r *http.Request) {
    dec := json.NewDecoder(r.Body)
    obj := &CatObject{}
    err := dec.Decode(obj)
    if err != nil {
        http.Error(w, err.Error(), http.StatusInternalServerError)
        return
    }
    // Генерируем уникальный идентификатор маркера
    if obj.Id == "" {
        obj.Id = sid.IdHex()
    }
    var tuples []CatObject
    // Вставляем новый маркер
    err = conn.ReplaceTyped("cats", []interface{}{obj.Id, obj.Coordinates, obj.Name}, &tuples)
    if err != nil {
        http.Error(w, err.Error(), http.StatusInternalServerError)
        return
    }
    enc := json.NewEncoder(w)
    enc.Encode(tuples)
    r.Body.Close()
})
go get github.com/chilts/sid
go run ./cats.go

Отправка котиков из html

function onMapClick(e) {
    var name = window.prompt('Cat name?')
    if (name != null) {
        L.marker(e.latlng).addTo(group).bindPopup(name)

        /*
        * Карта использует систему координат на шаре
        * Tarantool хранит координаты на плоскости
        * Конвертируем из одной системы в другую
        */
        var p = mymap.project(e.latlng, 1)
        
        var cat = {
            "coordinates": [p.x, p.y],
            "name": name,
        }

        fetch("/put", {
            method: "POST",
            body: JSON.stringify(cat)
        })
    }
}

Получение котиков из базы

Фронтенд

В html добавим функцию для загрузки котиков в рамках экрана

function addCat(cat) {
    var l = mymap.unproject(L.point(cat['coordinates']), 1)

    var name = cat['name']
    // Создаем маркер
    L.marker(l).addTo(group).bindPopup(name)
}
// Обрабатываем json пришедший с сервера
function parse(array) {
    array.forEach(addCat)
}
function errorResponse(error) {
    alert('Error: ' + error)
}
function handleListResponse(res) {
    res.json().then(parse).catch(errorResponse)
}
function onMapMove(e) {
    var bounds = mymap.getBounds()
    var northeast = bounds.getNorthEast()
    var southwest = bounds.getSouthWest()
    var ne = mymap.project(northeast, 1)
    var sw = mymap.project(southwest, 1)
    var options = {
        "rect": JSON.stringify([ne.x, ne.y, sw.x, sw.y]),
    }

    // Отправляем запрос на сервер с получением маркеров
    fetch("/list?" + new URLSearchParams(options))
        .then(handleListResponse)
        .catch(errorResponse)
}
mymap.on('move', onMapMove)
onMapMove()

Бекенд

// Отдаём маркеры для указанного в url региона
http.HandleFunc("/list", func(w http.ResponseWriter, r *http.Request) {
    rect, ok := r.URL.Query()["rect"]
    if !ok || len(rect) < 1 {
        http.Error(w, err.Error(), http.StatusInternalServerError)
        return
    }

    var arr []float64
    err := json.Unmarshal([]byte(rect[0]), &arr)
    if err != nil {
        panic(err)
    }

    // Запрашивает 1000 маркеров, которые находятся в регионе rect
    var tuples []CatObject
    err = conn.SelectTyped("cats", "spatial", 0, 1000, tarantool.IterLe,
        arr,
        &tuples)
    if err != nil {
        http.Error(w, err.Error(), http.StatusInternalServerError)
        return
    }
    enc := json.NewEncoder(w)
    enc.Encode(tuples)
    r.Body.Close()
})

Котики должны быть котиками

Добавим файл с котиком

  • index.html
var icon = L.icon({
	iconUrl: "",
	iconSize: [32, 32],
	iconAnchor: [0, 0],
	popupAnchor: [0, 0],
})
L.marker(l, {"icon": icon}).addTo(group).bindPopup(name)

Промежуточный итог

Маркеры на карте создаются. При навигации маркеры загружаются с бека.

Проблема

При навигации маркеры постоянно добавиляются, даже если уже были.

Решение

Выполнять проверку на фронтенде, если маркер уже был отрисован.

var alreadyloaded = {}
function addCat(cat) {
    if (!(cat.id in alreadyloaded)) {
        var l = mymap.unproject(L.point(cat['coordinates']), 1)

        var name = cat['name']
        // Создаем маркер
        L.marker(l, {"icon": icon}).addTo(group).bindPopup(name)
        alreadyloaded[cat.id] = cat
    }
}

А что если коты всегда гуляют сами по себе

Сделаем приложение бота, которое будет рассказывать нам о передвижениях котов. В реальном мире это мог бы быть аггрегатор gps координат с ошейников наших питомцев.

Файл catpaths.go

touch catpaths.go
  • catpaths.go
package main

import (
	"bytes"
	"encoding/json"
	"io"
	"io/ioutil"
	"math/rand"
	"net/http"
	"sync"
	"time"

	"github.com/chilts/sid"
	"github.com/tarantool/go-tarantool"
)

var names = []string{
	"Боня",
	"Ася",
	"Алиса",
	"Багира",
	"Бусинка",
	"Буся",
	"Аврора",
	"Муся",
	"Агния",
	"Ева",
	"Мася",
	"Агата",
	"Василиса",
	"Соня",
	"Агаша",
	"Мурка",
	"Муська",
	"Нюша",
	"Бася",
	"Симка",
	"Абракадабра",
	"Ангел",
	"Багирка",
	"Аба",
	"Анабель",
	"Абби",
	"Сима",
	"Ванесса",
	"Адель",
	"Дымка",
	"Абигель",
	"Бакси",
	"Барселона",
	"Масяня",
	"Абалина",
	"Даша",
	"Гера",
	"Агнесса",
	"Альфа",
	"Бэлла",
	"Амели",
	"Джессика",
	"Айса",
	"Барса",
	"Карамелька",
	"Бан-Ши",
	"Джесси",
	"Ириска",
	"Китти",
	"Агнес",
	"Айрис",
	"Кака",
	"Барсик",
	"Боня",
	"Бакс",
	"Алекс",
	"Бади",
	"Амур",
	"Ебони",
	"Абуссель",
	"Баксик",
	"Жопкинс",
	"Кузя",
	"Персик",
	"Абрек",
	"Абрикос",
	"Тимоша",
	"Авалон",
	"Бабник",
	"Саймон",
	"Бурбузяка",
	"Абу",
	"Марсик",
	"Маркиз",
	"Дымок",
	"Лаки",
	"Симба",
	"Абрамович",
	"Сёма",
	"Пушок",
	"Айс",
	"Бося",
	"Алмаз",
	"Кекс",
	"Басик",
	"Макс",
	"Феликс",
	"Гарфилд",
	"Том",
	"Тиша",
	"Цезарь",
	"Тишка",
	"Мася",
	"Абакан",
	"Лакки",
	"Васька",
	"Адольф",
	"Марсель",
	"Бабасик",
	"Вася",
	"Зевс",
	"Вольт",
	"Адидас",
	"Лео",
}

func randName() string {
	return names[rand.Intn(len(names))]
}

func randFloat(min float64, max float64) float64 {
	return min + rand.Float64()*(max-min)
}

func main() {
	opts := tarantool.Opts{User: "storage", Pass: "passw0rd"}
	conn, err := tarantool.Connect("127.0.0.1:3301", opts)
	if err != nil {
		panic(err)
	}
	defer conn.Close()

	conn.Call("box.space.geo:truncate", []interface{}{})

	tr := &http.Transport{
		MaxConnsPerHost: 50,
	}
	defer tr.CloseIdleConnections()
	cl := &http.Client{
		Transport: tr,
	}

    // Питерские координаты
	bounds := []float64{298.02, 148.52, 299.64, 149.20}

	centerx := 299.12
	centery := 148.80

	rand.Seed(time.Now().UnixNano())

	data := make(map[string]map[string]interface{})
	for i := 0; i < 1e3; i++ {
		id := sid.IdHex()
		item := map[string]interface{}{
			"id":          id,
			"coordinates": []float64{randFloat(bounds[0], bounds[2]), randFloat(bounds[1], bounds[3])},
			"name":        randName(),
		}
		data[id] = item

		bytes := new(bytes.Buffer)
		json.NewEncoder(bytes).Encode(item)
		resp, err := cl.Post("http://127.0.0.1:8080/put", "application/json", bytes)
		if err != nil {
			panic(err)
		}
		resp.Body.Close()
	}

	var wg sync.WaitGroup

	parralel := 0
	for step := 0; step < 1e6; step++ {
		for _, source := range data {

			value := make(map[string]interface{})
			value["coordinates"] = source["coordinates"]
			value["id"] = source["id"]
			value["name"] = source["name"]
			coords := value["coordinates"].([]float64)

			parralel = parralel + 1
			wg.Add(1)
			go func() {
				if rand.Int31n(4) == 0 {
					coords[0] = coords[0] + 0.0003
				} else {
					coords[0] = coords[0] - 0.0003
				}
				if rand.Int31n(4) == 0 {
					coords[1] = coords[1] + 0.0003
				} else {
					coords[1] = coords[1] - 0.0003
				}

				if coords[0] > centerx {
					coords[0] = coords[0] - 0.0003
				} else {
					coords[0] = coords[0] + 0.0003
				}
				if coords[1] > centery {
					coords[1] = coords[1] - 0.0003
				} else {
					coords[1] = coords[1] + 0.0003
				}

				if coords[0] < bounds[0] {
					coords[0] = bounds[0]
				}
				if coords[0] > bounds[2] {
					coords[0] = bounds[2]
				}
				if coords[1] < bounds[1] {
					coords[1] = bounds[1]
				}
				if coords[1] > bounds[3] {
					coords[1] = bounds[3]
				}

				value["coordinates"] = coords

				bytes := new(bytes.Buffer)
				json.NewEncoder(bytes).Encode(value)
				resp, err := cl.Post("http://127.0.0.1:8080/put", "application/json", bytes)
				if err != nil {
					panic(err)
				}
				io.Copy(ioutil.Discard, resp.Body)
				resp.Body.Close()
				wg.Done()
			}()
			if parralel == 1 {
				parralel = 0
				wg.Wait()
			}
		}
		time.Sleep(10 * time.Millisecond)
	}
}

Запуск

go run ./cat_paths.go

Теперь если обновлять страницу, то маркеры будут двигаться.

Автоматическое обновление страницы

Добавим автоматические обновление страницы.

redraw = setInterval(onMapMove, 1)

Сделаем изменение координат маркеров.

var alreadyloaded = {}
var popups = {}
function addCat(cat) {
    if (!(cat.id in alreadyloaded)) {
        var l = mymap.unproject(L.point(cat['coordinates']), 1)

        var name = cat['name']
        // Создаем маркер
        popups[cat.id] = L.marker(l, {"icon": icon}).addTo(group).bindPopup(name)
        alreadyloaded[cat.id] = cat
    } else {
        var l = mymap.unproject(L.point(cat['coordinates']), 1)
        popups[cat.id].setLatLng(l)
    }
}

Тестирование нагрузочное

Сделаем файл cats_test.go c тестами и проверим

touch cats_test.go
package main

import (
	"bytes"
	"encoding/json"
	"io/ioutil"
	"math/rand"
	"net/http"
	"testing"
)

const letterBytes = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"

func RandStringBytes(n int) string {
	b := make([]byte, n)
	for i := range b {
		b[i] = letterBytes[rand.Intn(len(letterBytes))]
	}
	return string(b)
}

func BenchmarkWrite(b *testing.B) {
	tr := &http.Transport{}
	defer tr.CloseIdleConnections()
	cl := &http.Client{
		Transport: tr,
	}

	var x = rand.Float64()
	var y = rand.Float64()
	var data = map[string]interface{}{
		"coordinates": []float64{x, y},
		"name":        RandStringBytes(100),
	}

	for i := 0; i < b.N; i++ {
		data["coordinates"] = []float64{float64(rand.Int31n(1000)) + rand.Float64(), float64(rand.Int31n(1000)) + rand.Float64()}
		bytes := new(bytes.Buffer)
		json.NewEncoder(bytes).Encode(data)
		res, err := cl.Post("http://127.0.0.1:8080/put", "application/json", bytes)
		if err != nil {
			b.Fatal("Post:", err)
		}
		_, err = ioutil.ReadAll(res.Body)
		if err != nil {
			b.Fatal("ReadAll:", err)
		}
	}
}

func BenchmarkRead(b *testing.B) {
	tr := &http.Transport{}
	defer tr.CloseIdleConnections()
	cl := &http.Client{
		Transport: tr,
	}

	for i := 0; i < b.N; i++ {
		bytes, err := json.Marshal([]float64{float64(rand.Int31n(1000)) + rand.Float64(),
            float64(rand.Int31n(1000)) + rand.Float64(),
			float64(rand.Int31n(360)) + rand.Float64(),
            float64(rand.Int31n(360)) + rand.Float64()})
		if err != nil {
			b.Fatal(err)
		}

		res, err := cl.Get("http://127.0.0.1:8080/list?rect=" + string(bytes))
		if err != nil {
			b.Fatal(err)
		}

		var objects []CatObject
		dec := json.NewDecoder(res.Body)
		err = dec.Decode(&objects)
		if err != nil {
			b.Fatal(err)
		}
		res.Body.Close()
	}
}

Запуск тестов

go test -benchmem -benchtime 10s -bench BenchmarkWrite

Масштабирование

Подключим реплику на чтение В golang писать будем в мастер, читать будем с реплики

replica.lua

touch replica.lua
box.cfg{work_dir="replica", listen=3302, replication="storage:[email protected]:3301"}

Создадим рабочую директорию для реплики

mkdir replica

Запустим реплику

tarantool replica.lua

Добавим в файл cats.go подключение к readonly реплике

opts = tarantool.Opts{User: "storage", Pass: "passw0rd"}
readconn, err := tarantool.Connect("127.0.0.1:3302", opts)
if err != nil {
	panic(err)
}
defer readconn.Close()
err = readconn.SelectTyped("cats", "spatial", 0, 1000, tarantool.IterLe,
			arr,
			&tuples)

Запустим котиков. Потому запустим нагрузочные тесты.

В заключение

Пользуйтесь Тарантулом, он данные хранит и вам быстро отдаёт. Tarantool это масштабируемый OLTP. Clickhouse это масштабируемый OLAP.

Можно писать микросервисы на Tarantool рядом с данными.

Что осталось за кадром:

  • Шардирование
  • Автоматический фейловер
  • Синхронная репликация

Вопросы?