README.md

# ExBitset

Fast implementation for dealing with immutable bitarray sets.

## Installation

Add ExBitset to your `mix.exs` dependencies:

```elixir
def deps do
  [
    {:ex_bitset, "~> 0.1.0"}
  ]
end
```

## Usage

To define a bitset type you need to use the `defbitset` function inside the
Elixir module (similar to how you would define structs).

```elixir
defmodule Roles do
  import ExBitset, only: [defbitset: 1]

  defbitset [:admin, :owner, :writer, :viewer, :guest]
end
```

You can then create new bitsets using the previously defined structure and
perform operations on it:


```elixir
roles = ExBitset.new(Roles, [:admin, :owner])

assert Enum.member?(roles, :admin)
assert :owner in Enum.to_list(roles)

bin_roles = ExBitset.to_binary(roles)
int_roles = ExBitset.to_int(roles)

assert Roles
  |> ExBitset.from_binary(bin_roles)
  |> Enum.member?(:admin)
assert Roles
  |> ExBitset.from_int(int_roles)
  |> Enum.member?(:admin)
```

### Usage with Ecto

To store one of these bitsets on the DB using, you can define a new Ecto type.
The type being stored can be an integer or binary, we're going to use integer in
this example because it would be more efficient for small sets, but you could
use any.

```elixir
defmodule MyApp.EctoTypes.Roles do
  import ExBitset, only: [defbitset: 1]
  use Ecto.Type

  defbitset [:admin, :owner, :writer, :viewer, :guest]

  def type, do: :integer

  def cast(roles) when is_list(roles) do
    {:ok, ExBitset.new(__MODULE__, roles)}
  end

  def cast(roles) when is_struct(roles, ExBitset) do
    {:ok, roles}
  end

  def cast(_roles), do: :error

  def load(roles) when is_integer(roles) do
    {:ok, ExBitset.from_int(__MODULE__, roles)}
  end

  def dump(roles) when is_struct(roles, ExBitset) do
    {:ok, ExBitset.to_int(roles)}
  end

  def dump(_roles), do: :error
end
```