Skip to content

Commit 95e2fff

Browse files
authored
Add native bitstring support to Postgres(#577)
1 parent 629b663 commit 95e2fff

File tree

8 files changed

+47
-12
lines changed

8 files changed

+47
-12
lines changed

Earthfile

+11-10
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
VERSION 0.6
22

33
all:
4-
ARG ELIXIR_BASE=1.15.6-erlang-25.3.2.6-alpine-3.17.4
4+
ARG ELIXIR_BASE=1.15.6-erlang-25.3.2.6-alpine-3.18.4
55
BUILD \
66
--build-arg POSTGRES=15.0 \
77
--build-arg POSTGRES=11.11 \
@@ -20,7 +20,7 @@ all:
2020
+integration-test-mssql
2121

2222
setup-base:
23-
ARG ELIXIR_BASE=1.15.6-erlang-25.3.2.6-alpine-3.17.4
23+
ARG ELIXIR_BASE=1.15.6-erlang-25.3.2.6-alpine-3.18.4
2424
FROM hexpm/elixir:$ELIXIR_BASE
2525
RUN apk add --no-progress --update git build-base
2626
ENV ELIXIR_ASSERT_TIMEOUT=10000
@@ -62,7 +62,7 @@ integration-test-postgres:
6262

6363
# then run the tests
6464
WITH DOCKER \
65-
--pull "postgres:$POSTGRES"
65+
--pull "postgres:$POSTGRES" --platform linux/amd64
6666
RUN set -e; \
6767
timeout=$(expr $(date +%s) + 30); \
6868
docker run --name pg --network=host -d -e POSTGRES_USER=postgres -e POSTGRES_PASSWORD=postgres -e POSTGRES_DB=postgres "postgres:$POSTGRES"; \
@@ -84,7 +84,7 @@ integration-test-mysql:
8484

8585
ARG MYSQL="5.7"
8686
WITH DOCKER \
87-
--pull "mysql:$MYSQL"
87+
--pull "mysql:$MYSQL" --platform linux/amd64
8888
RUN set -e; \
8989
timeout=$(expr $(date +%s) + 30); \
9090
docker run --name mysql --network=host -d -e MYSQL_ROOT_PASSWORD=root "mysql:$MYSQL" \
@@ -103,25 +103,26 @@ integration-test-mysql:
103103

104104

105105
integration-test-mssql:
106+
ARG TARGETARCH
106107
FROM +setup-base
107108

108109
RUN apk add --no-cache curl gnupg --virtual .build-dependencies -- && \
109-
curl -O https://download.microsoft.com/download/e/4/e/e4e67866-dffd-428c-aac7-8d28ddafb39b/msodbcsql17_17.5.2.1-1_amd64.apk && \
110-
curl -O https://download.microsoft.com/download/e/4/e/e4e67866-dffd-428c-aac7-8d28ddafb39b/mssql-tools_17.5.2.1-1_amd64.apk && \
111-
echo y | apk add --allow-untrusted msodbcsql17_17.5.2.1-1_amd64.apk mssql-tools_17.5.2.1-1_amd64.apk && \
110+
curl -O https://download.microsoft.com/download/3/5/5/355d7943-a338-41a7-858d-53b259ea33f5/msodbcsql18_18.3.2.1-1_${TARGETARCH}.apk && \
111+
curl -O https://download.microsoft.com/download/3/5/5/355d7943-a338-41a7-858d-53b259ea33f5/mssql-tools18_18.3.1.1-1_${TARGETARCH}.apk && \
112+
echo y | apk add --allow-untrusted msodbcsql18_18.3.2.1-1_${TARGETARCH}.apk mssql-tools18_18.3.1.1-1_${TARGETARCH}.apk && \
112113
apk del .build-dependencies && rm -f msodbcsql*.sig mssql-tools*.apk
113-
ENV PATH="/opt/mssql-tools/bin:${PATH}"
114+
ENV PATH="/opt/mssql-tools18/bin:${PATH}"
114115

115116
DO +COMMON_SETUP_AND_MIX
116117

117118
ARG MSSQL="2017"
118119
WITH DOCKER \
119-
--pull "mcr.microsoft.com/mssql/server:$MSSQL-latest"
120+
--pull "mcr.microsoft.com/mssql/server:$MSSQL-latest" --platform linux/amd64
120121
RUN set -e; \
121122
timeout=$(expr $(date +%s) + 30); \
122123
docker run -d -p 1433:1433 --name mssql -e 'ACCEPT_EULA=Y' -e 'MSSQL_SA_PASSWORD=some!Password' "mcr.microsoft.com/mssql/server:$MSSQL-latest"; \
123124
# wait for mssql to start
124-
while ! sqlcmd -S tcp:127.0.0.1,1433 -U sa -P 'some!Password' -Q "SELECT 1" >/dev/null 2>&1; do \
125+
while ! sqlcmd -C -S tcp:127.0.0.1,1433 -U sa -P 'some!Password' -Q "SELECT 1" >/dev/null 2>&1; do \
125126
test "$(date +%s)" -le "$timeout" || (echo "timed out waiting for mssql"; exit 1); \
126127
echo "waiting for mssql"; \
127128
sleep 1; \

integration_test/myxql/test_helper.exs

+2
Original file line numberDiff line numberDiff line change
@@ -93,6 +93,8 @@ version =
9393
end
9494

9595
excludes = [
96+
# not sure how to support this yet
97+
:bitstring_type,
9698
# MySQL does not have an array type
9799
:array_type,
98100
# The next two features rely on RETURNING, which MySQL does not support

integration_test/support/migration.exs

+8
Original file line numberDiff line numberDiff line change
@@ -108,6 +108,14 @@ defmodule Ecto.Integration.Migration do
108108
end
109109
end
110110

111+
unless :bitstring_type in ExUnit.configuration()[:exclude] do
112+
create table(:bitstrings) do
113+
add :bs, :bitstring
114+
add :bs_with_default, :bitstring, default: <<42::6>>
115+
add :bs_with_size, :bitstring, size: 10
116+
end
117+
end
118+
111119
create table(:composite_pk, primary_key: false) do
112120
add :a, :integer, primary_key: true
113121
add :b, :integer, primary_key: true

integration_test/tds/test_helper.exs

+1
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ ExUnit.start(
44
exclude: [
55
# not sure how to support this yet
66
:aggregate_filters,
7+
:bitstring_type,
78
# subquery contains ORDER BY and that is not supported
89
:subquery_aggregates,
910
# sql don't have array type

lib/ecto/adapters/postgres/connection.ex

+17
Original file line numberDiff line numberDiff line change
@@ -1015,6 +1015,11 @@ if Code.ensure_loaded?(Postgrex) do
10151015
["'\\x", Base.encode16(binary, case: :lower) | "'::bytea"]
10161016
end
10171017

1018+
defp expr(%Ecto.Query.Tagged{value: bitstring, type: :bitstring}, _sources, _query)
1019+
when is_bitstring(bitstring) do
1020+
bitstring_literal(bitstring)
1021+
end
1022+
10181023
defp expr(%Ecto.Query.Tagged{value: other, type: type}, sources, query) do
10191024
[maybe_paren(other, sources, query), ?:, ?: | tagged_to_db(type)]
10201025
end
@@ -1580,6 +1585,10 @@ if Code.ensure_loaded?(Postgrex) do
15801585
end
15811586
end
15821587

1588+
defp default_type(literal, _type) when is_bitstring(literal) do
1589+
bitstring_literal(literal)
1590+
end
1591+
15831592
defp default_type(literal, _type) when is_number(literal), do: to_string(literal)
15841593
defp default_type(literal, _type) when is_boolean(literal), do: to_string(literal)
15851594

@@ -1824,6 +1833,13 @@ if Code.ensure_loaded?(Postgrex) do
18241833

18251834
defp single_quote(value), do: [?', escape_string(value), ?']
18261835

1836+
defp bitstring_literal(value) do
1837+
size = bit_size(value)
1838+
<<val::size(size)>> = value
1839+
1840+
[?b, ?', val |> Integer.to_string(2) |> String.pad_leading(size, ["0"]), ?']
1841+
end
1842+
18271843
defp intersperse_reduce(list, separator, user_acc, reducer, acc \\ [])
18281844

18291845
defp intersperse_reduce([], _separator, user_acc, _reducer, acc),
@@ -1870,6 +1886,7 @@ if Code.ensure_loaded?(Postgrex) do
18701886
defp ecto_to_db(:bigserial), do: "bigserial"
18711887
defp ecto_to_db(:binary_id), do: "uuid"
18721888
defp ecto_to_db(:string), do: "varchar"
1889+
defp ecto_to_db(:bitstring), do: "varbit"
18731890
defp ecto_to_db(:binary), do: "bytea"
18741891
defp ecto_to_db(:map), do: Application.fetch_env!(:ecto_sql, :postgres_map_type)
18751892
defp ecto_to_db({:map, _}), do: Application.fetch_env!(:ecto_sql, :postgres_map_type)

mix.exs

+1-1
Original file line numberDiff line numberDiff line change
@@ -76,7 +76,7 @@ defmodule EctoSQL.MixProject do
7676
if path = System.get_env("ECTO_PATH") do
7777
{:ecto, path: path}
7878
else
79-
{:ecto, "~> 3.11.0"}
79+
{:ecto, github: "elixir-ecto/ecto"}
8080
end
8181
end
8282

mix.lock

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44
"decimal": {:hex, :decimal, "2.1.1", "5611dca5d4b2c3dd497dec8f68751f1f1a54755e8ed2a966c2633cf885973ad6", [:mix], [], "hexpm", "53cfe5f497ed0e7771ae1a475575603d77425099ba5faef9394932b35020ffcc"},
55
"deep_merge": {:hex, :deep_merge, "1.0.0", "b4aa1a0d1acac393bdf38b2291af38cb1d4a52806cf7a4906f718e1feb5ee961", [:mix], [], "hexpm", "ce708e5f094b9cd4e8f2be4f00d2f4250c4095be93f8cd6d018c753894885430"},
66
"earmark_parser": {:hex, :earmark_parser, "1.4.39", "424642f8335b05bb9eb611aa1564c148a8ee35c9c8a8bba6e129d51a3e3c6769", [:mix], [], "hexpm", "06553a88d1f1846da9ef066b87b57c6f605552cfbe40d20bd8d59cc6bde41944"},
7-
"ecto": {:hex, :ecto, "3.11.0", "ff8614b4e70a774f9d39af809c426def80852048440e8785d93a6e91f48fec00", [:mix], [{:decimal, "~> 2.0", [hex: :decimal, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "7769dad267ef967310d6e988e92d772659b11b09a0c015f101ce0fff81ce1f81"},
7+
"ecto": {:git, "https://github.com/elixir-ecto/ecto.git", "bed81b9a69f3425147fb57df1b3dd5fb4c95792c", []},
88
"ex_doc": {:hex, :ex_doc, "0.30.9", "d691453495c47434c0f2052b08dd91cc32bc4e1a218f86884563448ee2502dd2", [:mix], [{:earmark_parser, "~> 1.4.31", [hex: :earmark_parser, repo: "hexpm", optional: false]}, {:makeup_elixir, "~> 0.14", [hex: :makeup_elixir, repo: "hexpm", optional: false]}, {:makeup_erlang, "~> 0.1", [hex: :makeup_erlang, repo: "hexpm", optional: false]}], "hexpm", "d7aaaf21e95dc5cddabf89063327e96867d00013963eadf2c6ad135506a8bc10"},
99
"jason": {:hex, :jason, "1.4.1", "af1504e35f629ddcdd6addb3513c3853991f694921b1b9368b0bd32beb9f1b63", [:mix], [{:decimal, "~> 1.0 or ~> 2.0", [hex: :decimal, repo: "hexpm", optional: true]}], "hexpm", "fbb01ecdfd565b56261302f7e1fcc27c4fb8f32d56eab74db621fc154604a7a1"},
1010
"makeup": {:hex, :makeup, "1.1.1", "fa0bc768698053b2b3869fa8a62616501ff9d11a562f3ce39580d60860c3a55e", [:mix], [{:nimble_parsec, "~> 1.2.2 or ~> 1.3", [hex: :nimble_parsec, repo: "hexpm", optional: false]}], "hexpm", "5dc62fbdd0de44de194898b6710692490be74baa02d9d108bc29f007783b0b48"},

test/ecto/adapters/postgres_test.exs

+6
Original file line numberDiff line numberDiff line change
@@ -1990,6 +1990,9 @@ defmodule Ecto.Adapters.PostgresTest do
19901990
{:add, :on_hand, :integer, [default: 0, null: true]},
19911991
{:add, :published_at, :"time without time zone", [null: true]},
19921992
{:add, :is_active, :boolean, [default: true]},
1993+
{:add, :flags, :bitstring, [null: false]},
1994+
{:add, :flags_with_default, :bitstring, [default: <<42::10>>]},
1995+
{:add, :flags_with_size, :bitstring, [size: 10]},
19931996
{:add, :tags, {:array, :string}, [default: []]},
19941997
{:add, :languages, {:array, :string}, [default: ["pt", "es"]]},
19951998
{:add, :limits, {:array, :integer}, [default: [100, 30_000]]}
@@ -2002,6 +2005,9 @@ defmodule Ecto.Adapters.PostgresTest do
20022005
"on_hand" integer DEFAULT 0 NULL,
20032006
"published_at" time without time zone NULL,
20042007
"is_active" boolean DEFAULT true,
2008+
"flags" varbit NOT NULL,
2009+
"flags_with_default" varbit DEFAULT b'0000101010',
2010+
"flags_with_size" varbit(10),
20052011
"tags" varchar(255)[] DEFAULT ARRAY[]::varchar[],
20062012
"languages" varchar(255)[] DEFAULT ARRAY['pt','es']::varchar[],
20072013
"limits" integer[] DEFAULT ARRAY[100,30000]::integer[])

0 commit comments

Comments
 (0)