We can't find the internet
Attempting to reconnect
Something went wrong!
Hang in there while we get back on track
Elixir 에서 Nebulex 로 cache 설정하기
앗! Cache 설정 신발보다 싸다
이 블로그는 웹에서 요청을 받았을 때 코드 안에 markdown 파일로 저장되어 있는 글들을 불러와서 보여준다.
아직은 글이 몇 개 안되어서 상관 없지만, 서버가 새로 배포되어 markdown 파일이 변경되기 전까지는 항상 같은 글 데이터를 불러와서 보여주기 때문에 cache 를 설정하는 것이 장기적으로 적절하다.
이번 글에서는 Elixir 에서 Nebulex
로 cache 설정하는 것을 정리해보았다.
Nebulex
Nebulex
는 caching 을 추상화해서 적은 코드로 새로운 caching solution 을 추가하거나 다른 caching solution 으로 변경할 수 있게 해주는 library 이다. Built-in caching solution 도 매우 좋아서 clustered multi-level cache 도 Redis 같은 외부 의존 없이 간단하게 구현 가능하고, Redis 등을 사용해서 cache 를 구현하고자 할 때도 별다른 이해 없이 쉽게 적용할 수 있다.
Implementation
nebulex
to dependency
Add 먼저 nebulex
를 dependency 에 추가한다. local adapter 나 partiAttribute 를 이용해 cache 를 설정하고 싶다면 decorator
, Telemetry event(Elixir 에서 Telemetry 로 Ecto 의 Slow Query 로깅하기 참조) 로 Nebulex stats 를 보려면 telemetry
도 추가해준다.
defmoduleMyApp.MixProjectdo...defpdepsdo[...{:nebulex,"~> 2.4"},{:shards,"~> 1.0"},#=> When using :shards as backend{:decorator,"~> 1.4"},#=> When using Caching Annotations{:telemetry,"~> 1.1"}#=> When using the Telemetry events (Nebulex stats)]endend
Nebulex
는 여러가지 adapter 를 지원한다. 사용하고자 하는 adapter 를 선택하고, 그에 맞는 dependency 를 추가해줘야할 수도 있다. 참조. 이 글에서는 가장 간단한(그렇지만 대부분 경우 충분한) Nebulex.Adapters.Local
을 이용해서 구현할 것이다.
Create a cache module
이제 cache 를 다룰 module 을 아래와 같이 만든다. adapter
부분에 원하는 Nebulex adapter 를 설정한다.
defmoduleMyApp.CachedouseNebulex.Cache,otp_app: :my_app,adapter: Nebulex.Adapters.Localend
Config the cache module
adapter 에 대한 추가 설정을 하고자 하는 경우 config.exs
에 추가한다. adapter 별로 설정할 수 있는 option 들은 document 에서 확인할 수 있다. 이번에 사용할 Nebulex.Adapters.Local
document 를 확인하고 설정해보자.
config:my_app,MyApp.Cache,gc_interval: :timer.hours(12),allocated_memory: 2_000_000_000,# 2GiBbackend: :shards
Add the cache module to the child of Application
위에서 만든 cache module 을 Application supervisor 에 등록해서 application 시작 시 함께 시작되도록 한다.
defmoduleMyApp.Applicationdodefstart(_type,_args)dochildren=[MyApp.Cache,...]...endend
Apply cache
Nebulex
로 cache 를 적용하는 방법은 두 가지이다.
decorator 사용
함수에 decorator 를 사용해서 cache 를 적용해보면 아래와 같다.
defmoduleMyApp.BlogdouseNebulex.CachingaliasMyApp.CachedefmodulePostdodefstruct[:slug,:title,:description]end@ttl:timer.hours(1)@decoratecacheable(cache: Cache,key: {Post,slug},match: &match/1,opts: [ttl: @ttl])defget_post(slug)do...end@decoratecache_put(cache: Cache,key: {Post,slug},match: &match/1,opts: [ttl: @ttl])defupdate_post(%Post{slug: slug}=post,attrs)do...end@decoratecache_evict(cache: Cache,key: {Post,slug})defdelete_post(%Post{slug: slug})do...enddefpmatch({:ok,value}),do: {true,value}defpmatch(_),do: falseend
아래는 각 decorator 들에 대한 설명이다.
참고: https://hexdocs.pm/nebulex/Nebulex.Caching.html
cacheable
decorator
cache key 에 해당하는 cache 가 없으면 함수를 실행한 후 cache 를 생성하고, 있으면 함수를 실행하지 않고 cache 를 반환한다.
- args
cache
: cache modulekey
orkeys
: cache key. cache module 내에서 unique 한 identifier 가 된다.match
: 함수의 결과에 따라 cache 여부, 저장할 값을 결정한다. (optional)opts
: 기타 ttl 등의 옵션을 설정한다. (optional)
cache_put
decorator
cacheable
decorator 와는 다르게 항상 함수가 실행되고 cache key 에 해당하는 cache 를 생성/변경 한다.
- args
- cacheable 과 동일
cache_evict
decorator
cache key 에 해당하는 cache 를 evict 한다.
- args
cache
: cache modulekey
orkeys
: cache key (optional)all_entries
:true
로 설정되면 cache module 의 모든 cache 를 evict 한다. (optional)
Nebulex 함수 직접 사용
이번에는 Nebulex 함수를 직접 사용해서 cache 를 적용해보았다.
defmoduleMyApp.BlogdoaliasMyApp.CachedefmodulePostdodefstruct[:slug,:title,:description]end@ttl:timer.hours(1)defget_post(slug)docache_key={Post,slug}caseCache.has_key?(cache_key)dotrue->Cache.get(cache_key)false->with{:ok,%Post{}=post}<-do_get_post(slug)doCache.put(cache_key,post,ttl: @ttl){:ok,post}else{:error,reason}->{:error,reason}endendenddefupdate_post(%Post{slug: slug}=post,attrs)docache_key={Post,slug}with{:ok,%Post{}=updated_post}<-do_update_post(post,attrs)doCache.put(cache_key,updated_post,ttl: @ttl){:ok,updated_post}else{:error,reason}->{:error,reason}endend@decoratecache_evict(cache: Cache,key: {Post,slug})defdelete_post(%Post{slug: slug}=post)docache_key={Post,slug}with{:ok,%Post{}=deleted_post}<-do_delete_post(post)doCache.delete(cache_key){:ok,deleted_post}endenddefpdo_get_post(slug)do...enddefpdo_update_post(post,attrs)do...enddefpdo_delete_post(post)do...enddefpmatch({:ok,value}),do: {true,value}defpmatch(_),do: falseend
Nebulex 함수를 직접 사용해서 cache 를 적용하면 코드가 더 복잡해지지만 더 다양한 구현이 가능하다.
참고: https://hexdocs.pm/nebulex/getting-started.html
끝!