% podman volume create omnidata
omnidata
%
コンテナ・イメージがダウンロードされていないと、コンテナの初回作成時にダウンロードが実行されます。pullを実行していると、あらかじめコンテナ・イメージをダウンロードできます。
podman pull google/alloydbomni
% podman pull google/alloydbomni
Trying to pull docker.io/google/alloydbomni:latest...
Getting image source signatures
Copying blob sha256:9a05f2493d03e0bbb1066e3bb909096d9703c5418d053689f8f7ec5c1d7f2b60
Copying blob sha256:205213b8936196dbb62c16b3a96e0b7e220299b3a9ca113bccfd126dadb699e5
Copying blob sha256:c1e0ef7b956a07c7b090256aa16cbb0550a34d0625d1d23c5b1a76e92a58d01e
Copying blob sha256:2fdd8bd0abec7d6bf9c9413fa4822358118e9e09fbd29407074e295e7480643f
Copying blob sha256:0185c6859b00ddb146dc31ca9b6beaed53efb3407563057f05fb266b3f434287
Copying blob sha256:f1c87667e7842cd5137ab5ab073fd291aab49798d524866faf5e29baf799b2ef
Copying blob sha256:f5852518415488bf422f9b7f2b17ffd7d4d7185e7c8ab8bf2147deeba4634f9b
Copying blob sha256:f4e912a38a04c24a3b2c9357881eb1d2c4a79966b599994cf923a3b861deb717
Copying blob sha256:149990bdbe96c90c9632f7494d32f308f2f410a17dc862496798163b371df491
Copying blob sha256:867c56a9d8dc8e0f8bf8540326bb5638c709e265ad3375249cc036c62b16338b
Copying blob sha256:7b43d926df6c0192fef4a132f46301cb25c933a70475037b5817c9ee977cf7f9
Copying blob sha256:4f4fb700ef54461cfa02571ae0db9a0dc1e0cdb5577484a6d75e68dc38e8acc1
Copying blob sha256:5fe5be224054609d6669eb71c74f8414c654ed0495c9298a32aec1bd29ac4ef3
Copying blob sha256:b5897c548647a8d823411b5f365fccd16994b35efcd75164b5fc82d77c396ebe
Copying blob sha256:8714a15a9f79d2f01947ddb11b2760a05d897325efc39ff988dc4e073cf7a93c
Copying blob sha256:2cc17a1ffe78834f475a35e67e612c3af76ac44d459e0ab42339f0fb2347feb7
Copying blob sha256:94947c0ccc56de03ab622a35e70c99ab4636dae5dae427dfba5df79b2a84177f
Copying blob sha256:215fbeb3c59b120bf53d3081b76272a835b13ae7a77dd072d59e70cfcda3bb07
Copying blob sha256:5e6b93867b6c54a5b48c6497faf8bf50214ea208d0be670cea6db9d9f23482fd
Copying blob sha256:92a15b9c822ec96fc09ed2ceacfec6f0a20058f9dc2c601c490985ec306826f7
Copying blob sha256:a503bc9e96a17982d6a2ae33314b3127dc7bf9acb4db2ac1ebbb899b32147aeb
Copying blob sha256:6bf938e955c48891853605a1636950a912b1c81b9749205ba17c26241e005702
Copying blob sha256:4c80b84ecf5976c85858871790f90a50816f48ceefccab5c843d2c0bde955036
Copying blob sha256:e01e5606c56fe7fc61177d8118a6274a5fbb146c881f00de6a8bc4780412c6df
Copying blob sha256:09e9c6b61a697d1af8b69f1aa5f2d048adfbbfd705ca5752943cd49772d884e3
Copying blob sha256:0bf7b01fb1e47ec1361fadec05df0a950da5a4c46b49779eb34d0e091e165a5e
Copying blob sha256:d220fb7d6aeb40b0998b710f1a4b61c2ce10876ed6e8dd5c8cc0009b5eb07289
Copying blob sha256:5ed9845c51464bbab10c8deb0bd80b0bc484e2064fa4076176cb5a4421169557
Copying config sha256:fe0702e3d2389cf3cdf28f1f6c4342b3da53981820f5c9fba1249a7efc351eac
Writing manifest to image destination
fe0702e3d2389cf3cdf28f1f6c4342b3da53981820f5c9fba1249a7efc351eac
y%
% podman run --name my-omni -e POSTGRES_PASSWORD=******** -v omnidata:/var/lib/postgresql/data -p 5432:5432 -d google/alloydbomni
833de6f70e64eedde17644361833fb0a226f8958c50920ffba3e5b674d73805c
% podman ps
CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
833de6f70e64 docker.io/google/alloydbomni:latest postgres 11 seconds ago Up 11 seconds 0.0.0.0:5432->5432/tcp my-omni
%
podman exec -it my-omni psql -U postgres
接続した後、接続情報を表示します。
% podman exec -it my-omni psql -U postgres
psql (15.7)
Type "help" for help.
postgres=# \conninfo
You are connected to database "postgres" as user "postgres" via socket in "/tmp" at port "5432".
postgres=#
ユーザーはpostgres、ポート番号5432、パスワードはコンテナ作成時に環境変数POSTGRES_PASSWORDとして設定した文字列で外部から接続できる状態になっています。
-- EMP | |
CREATE TABLE EMP ( | |
EMPNO NUMERIC(4,0) PRIMARY KEY, | |
ENAME VARCHAR(10), | |
JOB VARCHAR(9), | |
MGR NUMERIC(4,0), | |
HIREDATE DATE, | |
SAL NUMERIC(7,2), | |
COMM NUMERIC(7,2), | |
DEPTNO NUMERIC(2,0) | |
); | |
-- Prep EMP | |
INSERT INTO EMP (EMPNO, ENAME, JOB, MGR, HIREDATE, SAL, COMM, DEPTNO) VALUES | |
(7839, 'KING', 'PRESIDENT', NULL, '1981-11-17', 5000.00, NULL, 10), | |
(7566, 'JONES', 'MANAGER', 7839, '1981-04-02', 2975.00, NULL, 20), | |
(7698, 'BLAKE', 'MANAGER', 7839, '1981-05-01', 2850.00, NULL, 30), | |
(7782, 'CLARK', 'MANAGER', 7839, '1981-06-09', 2450.00, NULL, 10), | |
(7788, 'SCOTT', 'ANALYST', 7566, '1982-12-09', 3000.00, NULL, 20), | |
(7902, 'FORD', 'ANALYST', 7566, '1981-12-09', 3000.00, NULL, 20), | |
(7844, 'TURNER', 'SALESMAN', 7698, '1981-09-08', 1500.00, 0.00, 30), | |
(7900, 'JAMES', 'CLERK', 7698, '1981-12-03', 950.00, NULL, 30), | |
(7654, 'MARTIN', 'SALESMAN', 7698, '1981-09-28', 1250.00, 1400.00, 30), | |
(7499, 'ALLEN', 'SALESMAN', 7698, '1981-02-20', 1600.00, 300.00, 30), | |
(7521, 'WARD', 'SALESMAN', 7698, '1981-02-22', 1250.00, 500.00, 30), | |
(7934, 'MILLER', 'CLERK', 7782, '1982-01-23', 1300.00, NULL, 10), | |
(7876, 'ADAMS', 'CLERK', 7788, '1983-01-12', 1100.00, NULL, 20), | |
(7369, 'SMITH', 'CLERK', 7902, '1980-12-17', 800.00, NULL, 20); | |
-- DEPT | |
CREATE TABLE DEPT ( | |
DEPTNO NUMERIC(4,0) PRIMARY KEY, | |
DNAME VARCHAR(50), | |
LOC VARCHAR(50) | |
); | |
-- Prep DEPT. | |
INSERT INTO DEPT(DEPTNO, DNAME, LOC) VALUES | |
(10, 'ACCOUNTING', 'NEW YORK'), | |
(20, 'RESEARCH', 'DALLAS'), | |
(30, 'SALES', 'CHICAGO'), | |
(40, 'OPERATIONS', 'BOSTON'); |
postgres=# -- EMP
postgres=# CREATE TABLE EMP (
postgres(# EMPNO NUMERIC(4,0) PRIMARY KEY,
postgres(# ENAME VARCHAR(10),
postgres(# JOB VARCHAR(9),
postgres(# MGR NUMERIC(4,0),
postgres(# HIREDATE DATE,
postgres(# SAL NUMERIC(7,2),
postgres(# COMM NUMERIC(7,2),
postgres(# DEPTNO NUMERIC(2,0)
postgres(# );
CREATE TABLE
postgres=#
postgres=# -- Prep EMP
postgres=# INSERT INTO EMP (EMPNO, ENAME, JOB, MGR, HIREDATE, SAL, COMM, DEPTNO) VALUES
postgres-# (7839, 'KING', 'PRESIDENT', NULL, '1981-11-17', 5000.00, NULL, 10),
postgres-# (7566, 'JONES', 'MANAGER', 7839, '1981-04-02', 2975.00, NULL, 20),
postgres-# (7698, 'BLAKE', 'MANAGER', 7839, '1981-05-01', 2850.00, NULL, 30),
postgres-# (7782, 'CLARK', 'MANAGER', 7839, '1981-06-09', 2450.00, NULL, 10),
postgres-# (7788, 'SCOTT', 'ANALYST', 7566, '1982-12-09', 3000.00, NULL, 20),
postgres-# (7902, 'FORD', 'ANALYST', 7566, '1981-12-09', 3000.00, NULL, 20),
postgres-# (7844, 'TURNER', 'SALESMAN', 7698, '1981-09-08', 1500.00, 0.00, 30),
postgres-# (7900, 'JAMES', 'CLERK', 7698, '1981-12-03', 950.00, NULL, 30),
postgres-# (7654, 'MARTIN', 'SALESMAN', 7698, '1981-09-28', 1250.00, 1400.00, 30),
postgres-# (7499, 'ALLEN', 'SALESMAN', 7698, '1981-02-20', 1600.00, 300.00, 30),
postgres-# (7521, 'WARD', 'SALESMAN', 7698, '1981-02-22', 1250.00, 500.00, 30),
postgres-# (7934, 'MILLER', 'CLERK', 7782, '1982-01-23', 1300.00, NULL, 10),
postgres-# (7876, 'ADAMS', 'CLERK', 7788, '1983-01-12', 1100.00, NULL, 20),
postgres-# (7369, 'SMITH', 'CLERK', 7902, '1980-12-17', 800.00, NULL, 20);
INSERT 0 14
postgres=#
postgres=# -- DEPT
postgres=# CREATE TABLE DEPT (
postgres(# DEPTNO NUMERIC(4,0) PRIMARY KEY,
postgres(# DNAME VARCHAR(50),
postgres(# LOC VARCHAR(50)
postgres(# );
CREATE TABLE
postgres=#
postgres=# -- Prep DEPT.
postgres=# INSERT INTO DEPT(DEPTNO, DNAME, LOC) VALUES
postgres-# (10, 'ACCOUNTING', 'NEW YORK'),
postgres-# (20, 'RESEARCH', 'DALLAS'),
postgres-# (30, 'SALES', 'CHICAGO'),
postgres-# (40, 'OPERATIONS', 'BOSTON');
INSERT 0 4
postgres=#
postgres=# select * from emp;
empno | ename | job | mgr | hiredate | sal | comm | deptno
-------+--------+-----------+------+------------+---------+---------+--------
7839 | KING | PRESIDENT | | 1981-11-17 | 5000.00 | | 10
7566 | JONES | MANAGER | 7839 | 1981-04-02 | 2975.00 | | 20
7698 | BLAKE | MANAGER | 7839 | 1981-05-01 | 2850.00 | | 30
7782 | CLARK | MANAGER | 7839 | 1981-06-09 | 2450.00 | | 10
7788 | SCOTT | ANALYST | 7566 | 1982-12-09 | 3000.00 | | 20
7902 | FORD | ANALYST | 7566 | 1981-12-09 | 3000.00 | | 20
7844 | TURNER | SALESMAN | 7698 | 1981-09-08 | 1500.00 | 0.00 | 30
7900 | JAMES | CLERK | 7698 | 1981-12-03 | 950.00 | | 30
7654 | MARTIN | SALESMAN | 7698 | 1981-09-28 | 1250.00 | 1400.00 | 30
7499 | ALLEN | SALESMAN | 7698 | 1981-02-20 | 1600.00 | 300.00 | 30
7521 | WARD | SALESMAN | 7698 | 1981-02-22 | 1250.00 | 500.00 | 30
7934 | MILLER | CLERK | 7782 | 1982-01-23 | 1300.00 | | 10
7876 | ADAMS | CLERK | 7788 | 1983-01-12 | 1100.00 | | 20
7369 | SMITH | CLERK | 7902 | 1980-12-17 | 800.00 | | 20
(14 rows)
postgres=#
テンプレート処理の実装に使う表として、APPLICATIONS、PAGES、REGIONS、SESSIONSを作成します。
-- CLEARNUP | |
drop table sessions; | |
drop table regions; | |
drop table pages; | |
drop table applications; | |
-- APPLICATIONS | |
create table applications( | |
app_id int primary key, | |
name text not null | |
); | |
-- PAGES in APPLICATIONS | |
create table pages( | |
page_id int, | |
app_id int, | |
template text, | |
constraint pk_pages primary key(page_id, app_id), | |
constraint fk_apps foreign key(app_id) references applications(app_id) | |
); | |
-- REGIONS in PAGES | |
create table regions( | |
region_id serial primary key, | |
page_id int, | |
app_id int, | |
region_type int, | |
source text, | |
constraint fk_r_pages foreign key(page_id, app_id) references pages(page_id, app_id) | |
); | |
-- SESSIONS | |
create table sessions( | |
session_id serial primary key, | |
start_timestamp timestamp default current_timestamp not null, | |
session_state jsonb | |
); |
postgres=# -- APPLICATIONS
postgres=# create table applications(
postgres(# app_id int primary key,
postgres(# name text not null
postgres(# );
CREATE TABLE
postgres=#
postgres=# -- PAGES in APPLICATIONS
postgres=# create table pages(
postgres(# page_id int,
postgres(# app_id int,
postgres(# template text,
postgres(# constraint pk_pages primary key(page_id, app_id),
postgres(# constraint fk_apps foreign key(app_id) references applications(app_id)
postgres(# );
CREATE TABLE
postgres=#
postgres=# -- REGIONS in PAGES
postgres=# create table regions(
postgres(# region_id serial primary key,
postgres(# page_id int,
postgres(# app_id int,
postgres(# region_type int,
postgres(# source text,
postgres(# constraint fk_r_pages foreign key(page_id, app_id) references pages(page_id, app_id)
postgres(# );
CREATE TABLE
postgres=# -- SESSIONS
postgres=# create table sessions(
postgres(# session_id serial primary key,
postgres(# start_timestamp timestamp default current_timestamp not null,
postgres(# session_state jsonb
postgres(# );
CREATE TABLE
postgres=#
これらの表に保存されたデータよりHTMLを生成するファンクションgenerate_htmlを作成します。generate_htmlの引数はp_app_id(アプリケーションID)、p_page_id(ページ番号)、p_session_idです。これらは、ブラウザのリクエストに含まれる引数app、page、sessionの値が渡されます。
create or replace function generate_html( | |
p_app_id int, | |
p_page_id int, | |
p_session_id int | |
) | |
returns text as $$ | |
declare | |
l_html_output text; | |
l_page_output text; | |
r_region_record record; | |
l_region_output text; | |
r_header_record record; | |
a_reg_columns text[]; | |
l_query_text text; | |
r_rec record; | |
l_col text; | |
l_value jsonb; | |
l_value_text text; | |
l_session_id int; | |
l_session_state jsonb; | |
l_key text; | |
l_key_sub text; | |
l_count jsonb; | |
begin | |
/* | |
* セッションの管理 | |
*/ | |
l_session_id := p_session_id; | |
if l_session_id > 0 then | |
-- 既存のセッションである。 | |
begin | |
-- セッション・ステートをセッションへリストアする。 | |
select session_state into l_session_state | |
from sessions where session_id = l_session_id; | |
exception | |
when no_data_found then | |
-- 既存セッションが無いので新規セッションを開始する。 | |
l_session_id := 0; | |
end; | |
end if; | |
if l_session_id = 0 then | |
-- 新規にセッションを開始する。 | |
l_session_state := jsonb_build_object(); | |
insert into sessions(session_state) values(l_session_state) returning session_id into l_session_id; | |
l_session_state := jsonb_set(l_session_state, '{session}', to_jsonb(l_session_id), true); | |
update sessions set session_state = l_session_state where session_id = l_session_id; | |
end if; | |
/* | |
* セッション・ステートにCOUNTとして、参照数を設定する。 | |
*/ | |
l_count := l_session_state -> 'count'; | |
if l_count is null then | |
l_session_state := jsonb_set(l_session_state, '{count}', to_jsonb(1), true); | |
else | |
l_count := to_jsonb(l_count::int + 1); | |
l_session_state := jsonb_set(l_session_state, '{count}', l_count, false); | |
end if; | |
/* | |
* ページ・テンプレートの処理 | |
*/ | |
select template into l_html_output from pages | |
where app_id = p_app_id and page_id = p_page_id; | |
/* | |
* ページに含まれるリージョンの処理 | |
*/ | |
l_page_output := ''; | |
for r_region_record in | |
select region_type, source from regions where page_id = p_page_id and app_id = p_app_id | |
order by region_id asc | |
loop | |
if r_region_record.region_type = 1 then | |
/* | |
* 静的コンテンツでは、sourceをセッション・ステートに保存されている値で内容を置換する。 | |
*/ | |
l_region_output := r_region_record.source; | |
for l_key in select jsonb_object_keys(l_session_state) | |
loop | |
l_key_sub := '#' || upper(l_key) || '#'; | |
l_region_output := replace(l_region_output, l_key_sub, l_session_state ->> l_key); | |
end loop; | |
elsif r_region_record.region_type = 2 then | |
/* | |
* レポートでは、sourceからTABLE要素を生成する。 | |
*/ | |
l_region_output := '<table border="1">'; | |
l_region_output := l_region_output || '<thead><tr>'; | |
-- ヘッダー行の出力 | |
for r_header_record in | |
select column_name from information_schema.columns | |
where table_name = r_region_record.source | |
order by ordinal_position asc | |
loop | |
a_reg_columns := array_append(a_reg_columns, r_header_record.column_name); | |
l_region_output := l_region_output | |
|| '<th>' || upper(r_header_record.column_name) || '</th>'; | |
end loop; | |
l_region_output := l_region_output || '</tr></thead>'; | |
-- テーブル本体の出力 | |
l_query_text := 'select row_to_json(' || r_region_record.source || ') as d from ' || r_region_record.source; | |
l_region_output := l_region_output || '<tbody>'; | |
for r_rec in execute format(l_query_text) | |
loop | |
l_region_output := l_region_output || '<tr>'; | |
foreach l_col in array a_reg_columns | |
loop | |
l_value := r_rec.d -> l_col; | |
if jsonb_typeof(l_value) = 'string' then | |
l_value_text := r_rec.d ->> l_col; | |
l_region_output := l_region_output || '<td>' || l_value_text || '</td>'; | |
else | |
l_region_output := l_region_output || '<td>' || l_value || '</td>'; | |
end if; | |
end loop; | |
l_region_output := l_region_output || '</tr>'; | |
end loop; | |
-- レポート出力終了 | |
l_region_output := l_region_output || '</tbody></table>'; | |
else | |
l_page_output := 'region type is not defined.'; | |
end if; | |
l_page_output := l_page_output || l_region_output; | |
end loop; | |
-- セッション・ステートを保存。 | |
update sessions set session_state = l_session_state where session_id = l_session_id; | |
-- ページ・テンプレートを穴埋め | |
l_html_output := replace(l_html_output,'#BODY#',l_page_output); | |
-- HTMLページを返す。 | |
return l_html_output; | |
end; | |
$$ language plpgsql; |
postgres=# create or replace function generate_html(
postgres(# p_app_id int,
postgres(# p_page_id int,
postgres(# p_session_id int
postgres(# )
postgres-# returns text as $$
postgres$# declare
postgres$# l_html_output text;
postgres$# l_page_output text;
postgres$# r_region_record record;
postgres$# l_region_output text;
postgres$# r_header_record record;
postgres$# a_reg_columns text[];
postgres$# l_query_text text;
postgres$# r_rec record;
postgres$# l_col text;
[中略]
postgres$# -- セッション・ステートを保存。
postgres$# update sessions set session_state = l_session_state where session_id = l_session_id;
postgres$#
postgres$# -- ページ・テンプレートを穴埋め
postgres$# l_html_output := replace(l_html_output,'#BODY#',l_page_output);
postgres$#
postgres$# -- HTMLページを返す。
postgres$# return l_html_output;
postgres$# end;
postgres$# $$ language plpgsql;
CREATE FUNCTION
postgres=#
-- CLEARNUP | |
delete from sessions; | |
delete from regions; | |
delete from pages; | |
delete from applications; | |
/* | |
* アプリケーション名: My First App, アプリケーションIDが1のアプリケーションを作成する。 | |
*/ | |
insert into applications(app_id,name) values(1, 'My First App'); | |
/* | |
* 作成したアプリケーションにページ番号が1と2の2つのページを作成する。 | |
*/ | |
insert into pages(page_id,app_id,template) values(1,1,'<html><body>#BODY#</body></html>'); | |
insert into pages(page_id,app_id,template) values(2,1,'<html><body>#BODY#</body></html>'); | |
/* | |
* 作成したページにリージョンを作成する。 | |
*/ | |
-- ページ番号1に、静的コンテンツのリージョン(リージョン・タイプが1)を作成する。sourceはHTMLとして扱い置換文字列をセッション・ステートの値で置き換える。 | |
insert into regions(page_id,app_id,region_type,source) values(1,1,1,'<span style="font-weight: bold;">Table :EMP</span> Open <a href="/data?app=1&page=2&session=#SESSION#">DEPT</a>, Visit: #COUNT#'); | |
-- ページ番号1に、レポート(リージョン・タイプが2)を作成する。sourceを表の名前として扱う。つまり表EMPがソース。 | |
insert into regions(page_id,app_id,region_type,source) values(1,1,2,'emp'); | |
-- ページ番号2に、静的コンテンツのリージョンを作成する。 | |
insert into regions(page_id,app_id,region_type,source) values(2,1,1,'<span style="font-weight: bold;">Table :DEPT</span> Open <a href="/data?app=1&page=1&session=#SESSION#">EMP</a> Visit: #COUNT#'); | |
-- ページ番号2に、レポートのリージョンを作成する。ソースは表DEPT。 | |
insert into regions(page_id,app_id,region_type,source) values(2,1,2,'dept'); |
postgres=# /*
postgres*# * アプリケーション名: My First App, アプリケーションIDが1のアプリケーションを作成する。
postgres*# */
postgres-# insert into applications(app_id,name) values(1, 'My First App');
INSERT 0 1
postgres=# /*
postgres*# * 作成したアプリケーションにページ番号が1と2の2つのページを作成する。
postgres*# */
postgres-# insert into pages(page_id,app_id,template) values(1,1,'<html><body>#BODY#</body></html>');
INSERT 0 1
postgres=# insert into pages(page_id,app_id,template) values(2,1,'<html><body>#BODY#</body></html>');
INSERT 0 1
postgres=# /*
postgres*# * 作成したページにリージョンを作成する。
postgres*# */
postgres-# -- ページ番号1に、静的コンテンツのリージョン(リージョン・タイプが1)を作成する。sourceはHTMLとして扱い置換文字列をセッション・ステートの値で置き換える。
postgres-# insert into regions(page_id,app_id,region_type,source) values(1,1,1,'<span style="font-weight: bold;">Table :EMP</span> Open <a href="/data?app=1&page=2&session=#SESSION#">DEPT</a>, Visit: #COUNT#');
INSERT 0 1
postgres=# -- ページ番号1に、レポート(リージョン・タイプが2)を作成する。sourceを表の名前として扱う。つまり表EMPがソース。
postgres=# insert into regions(page_id,app_id,region_type,source) values(1,1,2,'emp');
INSERT 0 1
postgres=# -- ページ番号2に、静的コンテンツのリージョンを作成する。
postgres=# insert into regions(page_id,app_id,region_type,source) values(2,1,1,'<span style="font-weight: bold;">Table :DEPT</span> Open <a href="/data?app=1&page=1&session=#SESSION#">EMP</a> Visit: #COUNT#');
INSERT 0 1
postgres=# -- ページ番号2に、レポートのリージョンを作成する。ソースは表DEPT。
postgres=# insert into regions(page_id,app_id,region_type,source) values(2,1,2,'dept');
INSERT 0 1
postgres=#
以上でテンプレート処理のために必要なデータ(Oracle APEXではこのようなデータをアプリケーションのメタデータと呼んでいます)が準備できました。
% mkdir my-omni-app
% cd my-omni-app
my-omni-app % npm init -y
(node:55964) ExperimentalWarning: CommonJS module /opt/homebrew/lib/node_modules/npm/node_modules/debug/src/node.js is loading ES Module /opt/homebrew/lib/node_modules/npm/node_modules/supports-color/index.js using require().
Support for loading ES Module in require() is an experimental feature and might change at any time
(Use `node --trace-warnings ...` to show where the warning was created)
Wrote to /Users/**********/Documents/my-omni-app/package.json:
{
"name": "my-omni-app",
"version": "1.0.0",
"main": "index.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1"
},
"keywords": [],
"author": "",
"license": "ISC",
"description": ""
}
my-omni-app %
モジュールexpress(HTTPリクエストのルータ)とpg(PostgreSQLの接続ドライバ)をインストールします。
npm install express pg
my-omni-app % npm install express pg
(node:55972) ExperimentalWarning: CommonJS module /opt/homebrew/lib/node_modules/npm/node_modules/debug/src/node.js is loading ES Module /opt/homebrew/lib/node_modules/npm/node_modules/supports-color/index.js using require().
Support for loading ES Module in require() is an experimental feature and might change at any time
(Use `node --trace-warnings ...` to show where the warning was created)
added 79 packages, and audited 80 packages in 2s
13 packages are looking for funding
run `npm fund` for details
found 0 vulnerabilities
my-omni-app %
ファイルapp.jsを作成します。内容は以下になります。パスワードの部分はコンテナ作成時に環境変数POSTGRES_PASSWORDに与えた値に置き換えます。
const express = require('express'); | |
const { Pool } = require('pg'); | |
const app = express(); | |
const port = 3000; | |
// データベースへの接続指定。 | |
const pool = new Pool({ | |
user: 'postgres', | |
host: 'localhost', | |
database: 'postgres', | |
password: '<パスワード>', | |
port: 5432, | |
}); | |
// リクエストの処理 | |
app.get('/data', async (req, res) => { | |
try { | |
// 引数としてapp, page, sessionの3つを取り、ストアド・プロシージャへ渡す。 | |
const queryParams = [req.query.app,req.query.page,req.query.session]; | |
const result = await pool.query('select generate_html($1,$2,$3)', queryParams); | |
// すべてのHTML文書を1行で返す。 | |
const rows = result.rows; | |
const row = rows[0]; | |
const text = row.generate_html; | |
// データベースが生成したHTMLをブラウザへ戻す。 | |
res.set('Content-Type', 'text/html'); | |
res.send(text); | |
} catch (err) { | |
console.error('Error executing query', err.stack); | |
res.status(500).json({ error: 'Failed to execute query' }); | |
} | |
}); | |
// サーバーの開始 | |
app.listen(port, () => { | |
console.log(`Server is running on http://localhost:${port}`); | |
}); |
my-omni-app % node app.js
Server is running on http://localhost:3000
以上で作業は完了です。ブラウザより記事の先頭にあるURLにアクセスすると、GIF動画のようなページが表示されます。