diff --git a/.dockerignore b/.dockerignore
new file mode 100644
index 0000000000000000000000000000000000000000..b512c09d476623ff4bf8d0d63c29b784925dbdf8
--- /dev/null
+++ b/.dockerignore
@@ -0,0 +1 @@
+node_modules
\ No newline at end of file
diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml
index e35789def0cfe075414a549041e95c960b8a2c40..3a29f10fe025a48e2859a3b0f067bf9061d59475 100644
--- a/.gitlab-ci.yml
+++ b/.gitlab-ci.yml
@@ -2,6 +2,8 @@ image: node:16-alpine
 stages:
   - setup
   - test
+  - build
+  - dockerize
 
 install-dependencies:
   stage: setup
@@ -33,14 +35,40 @@ install-dependencies:
     paths:
       - node_modules/.cache/nx
 
-build:
-  stage: test
-  extends: .distributed
-  script:
-    - yarn nx affected --base=HEAD~1 --target=build --parallel --max-parallel=3
-
 test:
   stage: test
   extends: .distributed
   script:
     - yarn nx affected --base=HEAD~1 --target=test --parallel --max-parallel=2
+
+build:
+  stage: build
+  only:
+    - main
+  needs:
+    - install-dependencies
+  artifacts:
+    paths:
+      - node_modules/.cache/nx
+      - dist/apps/web-graphpolaris
+  script:
+    # - yarn nx affected --base=HEAD~1 --target=build --parallel --max-parallel=3
+    # only build web-graphpolaris
+    - yarn nx build web-graphpolaris --prod
+
+build-docker:
+  image: docker:stable
+  stage: dockerize
+  tags:
+    - docker
+  only:
+    - main
+  script:
+    - docker build --progress plain -t datastropheregistry.azurecr.io/frontend:latest:latest .
+  # after_script:
+  #   - docker login datastropheregistry.azurecr.io -u $REGISTRY_USERNAME -p $REGISTRY_PASSWORD
+  #   - if [[ ! -z $CI_COMMIT_BRANCH+x ]]; then DOCKER_TAG=$CI_COMMIT_BRANCH; else DOCKER_TAG=$CI_MERGE_REQUEST_TARGET_BRANCH_NAME; fi
+  #   - docker tag $CI_PROJECT_NAME-webserver-service datastropheregistry.azurecr.io/$CI_PROJECT_NAME-webserver-service:$DOCKER_TAG
+  #   - docker push datastropheregistry.azurecr.io/$CI_PROJECT_NAME-webserver-service:$DOCKER_TAG
+  dependencies:
+    - build
diff --git a/.vscode/settings.json b/.vscode/settings.json
index 6324a78d9f129a028e3518b3bc3ccb4be511a062..0c27507c7c9016a219a2527fa8d4b61b0302bfad 100644
--- a/.vscode/settings.json
+++ b/.vscode/settings.json
@@ -24,5 +24,14 @@
     "store",
     "libs"
   ],
-  "jest.jestCommandLine": "nx affected:test"
+  "jest.jestCommandLine": "nx affected:test",
+
+  "jsonColorToken.languages": [
+    "json",
+    "jsonc",
+    "javascript",
+    "javascriptreact",
+    "typescript",
+    "typescriptreact"
+  ]
 }
diff --git a/Dockerfile b/Dockerfile
index 5ae50ee3abb63bbc32251cc96e7094c28d330d66..39c70241463b4eebd6f935ad20eb0dbee962c535 100644
--- a/Dockerfile
+++ b/Dockerfile
@@ -1,13 +1,12 @@
-# # Prepare nginx
-# FROM nginx:1.19-alpine
-# WORKDIR /app
+# Prepare nginx
+FROM nginx:1.19-alpine
+WORKDIR /app
 
-# # ! This copy source needs to be changed to reflect the actual app name
-# COPY ./dist/apps/frontend /usr/share/nginx/html
+COPY ./dist/apps/web-graphpolaris /usr/share/nginx/html
 
-# RUN rm /etc/nginx/conf.d/default.conf
-# COPY nginx/nginx.conf /etc/nginx/conf.d
+RUN rm /etc/nginx/conf.d/default.conf
+COPY nginx/nginx.conf /etc/nginx/conf.d
 
-# # Fire up nginx
-# EXPOSE 80
-# CMD ["nginx", "-g", "daemon off;"]
+# Fire up nginx
+EXPOSE 80
+CMD ["nginx", "-g", "daemon off;"]
diff --git a/README.md b/README.md
index 9f79b01a9bf341c469a185d07e24283bc938c6cc..ebcaddb86091a83d3a8e2df8a5d1defc53f03b89 100644
--- a/README.md
+++ b/README.md
@@ -7,23 +7,23 @@ Due to the way auth works (using a sameSite cookie), the procedure for running l
 ### MacOS / Linux
 
 1. `sudo vim /etc/hosts` open the hosts file with your prefered text editor as root
-2. Add a new row containing `127.0.0.1 local.datastrophe.science.uu.nl`, this will route traffic from `local.datastrophe.science.uu.nl` to `127.0.0.1`
+2. Add a new row containing `127.0.0.1 local.graphpolaris.com`, this will route traffic from `local.graphpolaris.com` to `127.0.0.1`
 3. `brew install mkcert` install mkcert utility
 4. `mkcert -install` generate local CA (certificate authority)
 5. Move into the /certs folder at the project root using `cd`
-6. `mkcert --key-file local-key.pem --cert-file local-cert.pem local.datastrophe.science.uu.nl` create certificates for local SSL
+6. `mkcert --key-file local-key.pem --cert-file local-cert.pem local.graphpolaris.com` create certificates for local SSL
 
 ### Windows
 
 1. Open the `hosts` file under `C:\Windows\System32\drivers\etc` using a text editor, as administrator
-2. Add a new row containing `127.0.0.1 local.datastrophe.science.uu.nl`, this will route traffic from `local.datastrophe.science.uu.nl` to `127.0.0.1`
+2. Add a new row containing `127.0.0.1 local.graphpolaris.com`, this will route traffic from `local.graphpolaris.com` to `127.0.0.1`
 3. Install mkcert using any of the ways described [here](https://github.com/FiloSottile/mkcert#windows)
 4. Open an elevated Powershell or CMD session
 5. Move into the /certs folder at the project root using `cd`
-6. `mkcert --key-file local-key.pem --cert-file local-cert.pem local.datastrophe.science.uu.nl` create certificates for local SSL
+6. `mkcert --key-file local-key.pem --cert-file local-cert.pem local.graphpolaris.com` create certificates for local SSL
 
 > No idea if the Windows steps work
 
 ## Running Locally
 
-To run the application using SSL (with these keys) simply run `nx run web-graphpolaris:dev`, or `yarn nx run web-graphpolaris:dev` if `nx` is not installed globally. This should open a window to `https://local.datastrophe.science.uu.nl:4200/` automatically.
+To run the application using SSL (with these keys) simply run `nx run web-graphpolaris:dev`, or `yarn nx run web-graphpolaris:dev` if `nx` is not installed globally. This should open a window to `https://local.graphpolaris.com:4200/` automatically.
diff --git a/apps/web-graphpolaris/project.json b/apps/web-graphpolaris/project.json
index dc9c691acc30a77b498ee4ee9c7abf79928cff15..86403889595b911f74d0f30838cc3bc428eb2251 100644
--- a/apps/web-graphpolaris/project.json
+++ b/apps/web-graphpolaris/project.json
@@ -7,7 +7,7 @@
       "executor": "@nrwl/web:dev-server",
       "options": {
         "buildTarget": "web-graphpolaris:build",
-        "host": "local.datastrophe.science.uu.nl",
+        "host": "local.graphpolaris.com",
         "port": 4200,
         "watch": true,
         "hmr": true,
@@ -19,7 +19,9 @@
     },
     "build": {
       "executor": "@nrwl/web:webpack",
-      "outputs": ["{options.outputPath}"],
+      "outputs": [
+        "{options.outputPath}"
+      ],
       "defaultConfiguration": "production",
       "options": {
         "compiler": "babel",
@@ -33,18 +35,18 @@
           "apps/web-graphpolaris/src/favicon.ico",
           "apps/web-graphpolaris/src/assets"
         ],
-        "styles": ["apps/web-graphpolaris/src/styles.scss"],
+        "styles": [
+          "apps/web-graphpolaris/src/styles.scss"
+        ],
         "scripts": [],
         "webpackConfig": "@nrwl/react/plugins/webpack"
       },
       "configurations": {
         "production": {
-          "fileReplacements": [
-            {
-              "replace": "apps/graphpolaris/src/environments/environment.ts",
-              "with": "apps/graphpolaris/src/environments/environment.prod.ts"
-            }
-          ],
+          "fileReplacements": [{
+            "replace": "apps/graphpolaris/src/environments/environment.ts",
+            "with": "apps/graphpolaris/src/environments/environment.prod.ts"
+          }],
           "optimization": true,
           "outputHashing": "all",
           "sourceMap": false,
@@ -69,14 +71,20 @@
     },
     "lint": {
       "executor": "@nrwl/linter:eslint",
-      "outputs": ["{options.outputFile}"],
+      "outputs": [
+        "{options.outputFile}"
+      ],
       "options": {
-        "lintFilePatterns": ["apps/graphpolaris/**/*.{ts,tsx,js,jsx}"]
+        "lintFilePatterns": [
+          "apps/graphpolaris/**/*.{ts,tsx,js,jsx}"
+        ]
       }
     },
     "test": {
       "executor": "@nrwl/jest:jest",
-      "outputs": ["coverage/apps/graphpolaris"],
+      "outputs": [
+        "coverage/apps/graphpolaris"
+      ],
       "options": {
         "jestConfig": "apps/web-graphpolaris/jest.config.js",
         "passWithNoTests": true
@@ -105,7 +113,9 @@
     },
     "build-storybook": {
       "executor": "@nrwl/storybook:build",
-      "outputs": ["{options.outputPath}"],
+      "outputs": [
+        "{options.outputPath}"
+      ],
       "options": {
         "uiFramework": "@storybook/react",
         "outputPath": "dist/storybook/graphpolaris",
@@ -121,4 +131,4 @@
     }
   },
   "tags": []
-}
+}
\ No newline at end of file
diff --git a/apps/web-graphpolaris/src/app/app.tsx b/apps/web-graphpolaris/src/app/app.tsx
index a9e7eb790222d8840f14d1c89271f8719bc191a4..6be79b2f6674c5f19dc0e18e83851ed4f48d9bb8 100644
--- a/apps/web-graphpolaris/src/app/app.tsx
+++ b/apps/web-graphpolaris/src/app/app.tsx
@@ -1,16 +1,18 @@
 // eslint-disable-next-line @typescript-eslint/no-unused-vars
-import { GetUserInfo } from '@graphpolaris/shared/data-access/api';
-import { AuthorizationHandler } from '@graphpolaris/shared/data-access/authorization';
-import {
-  assignNewGraphQueryResult,
-  useAppDispatch,
-} from '@graphpolaris/shared/data-access/store';
 import { useEffect, useState } from 'react';
 import GridLayout from 'react-grid-layout';
 import LoginScreen from '../components/login/loginScreen';
 import Panel from '../components/panels/panel';
 import { RawJSONVis } from '../components/visualisations/rawjsonvis/rawjsonvis';
 import SemanticSubstrates from '../components/visualisations/semanticsubstrates/semanticsubstrates';
+import Schema from '../components/schema/schema';
+import { GetUserInfo } from '@graphpolaris/shared/data-access/api';
+import QueryBuilder from '../components/querybuilder/querybuilder';
+import {
+  assignNewGraphQueryResult,
+  useAppDispatch,
+} from '@graphpolaris/shared/data-access/store';
+import { AuthorizationHandler } from '@graphpolaris/shared/data-access/authorization';
 
 function useIsAuthorized() {
   const [userAuthorized, setUserAuthorized] = useState(false);
@@ -61,7 +63,9 @@ export function App() {
           key="query-panel"
           data-grid={{ x: 3, y: 20, w: 5, h: 10, maxH: 20, isDraggable: false }}
         >
-          <Panel content="Query Panel" color="blue"></Panel>
+          <Panel content="Query Panel" color="blue">
+            <QueryBuilder />
+          </Panel>
         </div>
         <div
           key="visualisation-panel"
diff --git a/apps/web-graphpolaris/src/components/login/loginScreen.tsx b/apps/web-graphpolaris/src/components/login/loginScreen.tsx
index 93339edaadd13653822233ecf9aac3468f9f4435..fc1cf4a89f19194a218396348dbcc5cf4dc1e608 100644
--- a/apps/web-graphpolaris/src/components/login/loginScreen.tsx
+++ b/apps/web-graphpolaris/src/components/login/loginScreen.tsx
@@ -102,7 +102,7 @@ const LoginScreen = () => {
         <img
           onClick={() =>
             openSignInWindow(
-              'https://datastrophe.science.uu.nl/user/sign-in?provider=1'
+              'https://api.graphpolaris.com/user/sign-in?provider=1'
             )
           }
           src="assets/login-screen/google.png"
@@ -111,7 +111,7 @@ const LoginScreen = () => {
         <img
           onClick={() =>
             openSignInWindow(
-              'https://datastrophe.science.uu.nl/user/sign-in?provider=2'
+              'https://api.graphpolaris.com/user/sign-in?provider=2'
             )
           }
           src="assets/login-screen/github.png"
@@ -119,9 +119,7 @@ const LoginScreen = () => {
         />
         <p
           onClick={() =>
-            openSignInWindow(
-              'https://datastrophe.science.uu.nl/user/create_free/'
-            )
+            openSignInWindow('https://api.graphpolaris.com/user/create_free/')
           }
         >
           Developer
diff --git a/apps/web-graphpolaris/src/components/querybuilder/customFlowLines/connection.tsx b/apps/web-graphpolaris/src/components/querybuilder/customFlowLines/connection.tsx
new file mode 100644
index 0000000000000000000000000000000000000000..618f67b6bc755f3e1766712d77f3e72dd0981291
--- /dev/null
+++ b/apps/web-graphpolaris/src/components/querybuilder/customFlowLines/connection.tsx
@@ -0,0 +1,73 @@
+import { handles } from '@graphpolaris/querybuilder/usecases';
+import React from 'react';
+import { EdgeProps, getSmoothStepPath, Position } from 'react-flow-renderer';
+
+/**
+ * A custom query element edge line component.
+ * @param {EdgeProps} param0 The coordinates for the start and end point, the id and the style.
+ */
+export default function ConnectionLine({
+  id,
+  sourceX,
+  sourceY,
+  targetX,
+  targetY,
+  style,
+  sourceHandleId,
+  targetHandleId,
+}: EdgeProps) {
+  //Centering the line
+  sourceY -= 3;
+  targetY -= 3;
+
+  // Correct line positions with hardcoded numbers, because react flow lacks this functionality
+  // if (sourceHandleId == ) sourceX += 2;
+
+  // if (targetHandleId == Handles.ToAttributeHandle) targetX += 2;
+
+  let spos: Position = Position.Bottom;
+  if (sourceHandleId == handles.relation.fromEntity) {
+    spos = Position.Left;
+    sourceX += 7;
+    sourceY += 3;
+  } else if (sourceHandleId == handles.relation.toEntity) {
+    spos = Position.Right;
+    sourceX -= 2;
+    sourceY -= 3;
+  } else if (
+    sourceHandleId !== undefined &&
+    sourceHandleId !== null &&
+    sourceHandleId.includes('functionHandle')
+  ) {
+    spos = Position.Top;
+    sourceX -= 4;
+    sourceY += 3;
+  }
+
+  let tpos: Position = Position.Bottom;
+  if (targetHandleId == handles.relation.fromEntity) {
+    tpos = Position.Left;
+    targetX += 7;
+    targetY += 3;
+  } else if (targetHandleId == handles.relation.toEntity) {
+    tpos = Position.Right;
+    targetX -= 2;
+    targetY -= 3;
+  }
+
+  // Create smoothstep line
+  const path = getSmoothStepPath({
+    sourceX: sourceX,
+    sourceY: sourceY,
+    sourcePosition: spos,
+    targetX: targetX,
+    targetY: targetY,
+    targetPosition: tpos,
+  });
+
+  return (
+    <g stroke="#2e2e2e">
+      <path id={id} fill="none" strokeWidth={3} style={style} d={path} />
+    </g>
+  );
+}
diff --git a/apps/web-graphpolaris/src/components/querybuilder/customFlowLines/connectionDrag.tsx b/apps/web-graphpolaris/src/components/querybuilder/customFlowLines/connectionDrag.tsx
new file mode 100644
index 0000000000000000000000000000000000000000..9932aecdd154e7aa1af622afb9ddbf14fb170c77
--- /dev/null
+++ b/apps/web-graphpolaris/src/components/querybuilder/customFlowLines/connectionDrag.tsx
@@ -0,0 +1,41 @@
+import React from 'react';
+import { ConnectionLineComponentProps } from 'react-flow-renderer';
+
+/**
+ * A custom query element to render the line when connecting flow elements.
+ * @param {ConnectionLineComponentProps} param0 Source and target coordinates of the edges.
+ */
+export default function ConnectionDragLine({
+  sourceX,
+  sourceY,
+  targetX,
+  targetY,
+}: ConnectionLineComponentProps) {
+  return (
+    <g>
+      <path
+        fill="none"
+        stroke="#222"
+        strokeWidth={2.5}
+        className="animated"
+        d={`M${sourceX},${sourceY}L ${targetX},${targetY}`}
+      />
+      <circle
+        cx={sourceX}
+        cy={sourceY}
+        fill="#fff"
+        r={3}
+        stroke="#222"
+        strokeWidth={1.5}
+      />
+      <circle
+        cx={targetX}
+        cy={targetY}
+        fill="#fff"
+        r={3}
+        stroke="#222"
+        strokeWidth={1.5}
+      />
+    </g>
+  );
+}
diff --git a/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/attributepill/attributepill.module.scss b/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/attributepill/attributepill.module.scss
new file mode 100644
index 0000000000000000000000000000000000000000..9ba5ba22c49c279617faef06e92d4256e49a36fb
--- /dev/null
+++ b/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/attributepill/attributepill.module.scss
@@ -0,0 +1,60 @@
+@use './variables.module.scss';
+
+.attribute {
+  display: flex;
+  font-family: monospace;
+  font-weight: bold;
+  font-size: variables.$fontsize;
+  border-radius: 2px;
+}
+
+// .handle {
+//   border: 0px;
+//   border-radius: 10px;
+//   left: 12px;
+//   width: 7px;
+//   height: 7px;
+//   margin-bottom: 11px;
+//   background: rgba(255, 255, 255, 0.6);
+//   box-shadow: 0 0 0 1px rgba(0, 0, 0, 0.3);
+//   transform-origin: center;
+// }
+
+.contentWrapper {
+  display: flex;
+  align-items: center;
+
+  .content {
+    padding: variables.$ypad 0 variables.$ypad 1ch;
+    max-width: 15ch;
+    text-overflow: ellipsis;
+    overflow: hidden;
+    white-space: nowrap;
+  }
+}
+
+.attributeInput {
+  float: right;
+  padding: 0 1ch 0 0;
+  display: flex;
+  align-items: center;
+
+  input {
+    background-color: rgba(100, 100, 100, 0.1);
+    font-family: monospace;
+    font-size: variables.$fontsize;
+    border: 1px solid rgba(100, 100, 100, 0.3);
+    border-radius: 2px;
+    height: variables.$height;
+    outline: none;
+    transition: border 0.3s;
+    color: black;
+    &::placeholder {
+      color: black;
+    }
+
+    &:focus {
+      border: 1px solid rgba(0, 0, 0, 0.3);
+    }
+  }
+}
diff --git a/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/attributepill/attributepill.tsx b/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/attributepill/attributepill.tsx
new file mode 100644
index 0000000000000000000000000000000000000000..41b5803cc251e5f345702e9a364ee9eac4c6b113
--- /dev/null
+++ b/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/attributepill/attributepill.tsx
@@ -0,0 +1,96 @@
+import {
+  CheckDatatypeConstraint,
+  GetAttributeBoolOperators,
+} from '@graphpolaris/querybuilder/usecases';
+import {
+  updateQBAttributeOperator,
+  updateQBAttributeValue,
+  useAppDispatch,
+} from '@graphpolaris/shared/data-access/store';
+import { useTheme } from '@mui/material';
+import React, { useMemo, useState } from 'react';
+import styles from './attributepill.module.scss';
+import AttributeOperatorSelect from './operatorselect';
+
+/**
+ * Component to render an attribute flow element
+ * @param {FlowElement<EntityData>)} param0 The data of an entity flow element.
+ */
+export const AttributeRFPill = React.memo(
+  ({ id, data }: { id: string; data: any }) => {
+    const theme = useTheme();
+    const dispatch = useAppDispatch();
+    const [value, setValue] = useState(data?.value || '');
+
+    const onChange = (e: any) => {
+      setValue(e.target.value);
+    };
+    const validateInput = () => {
+      const newValue = CheckDatatypeConstraint(data.datatype, value);
+      setValue(newValue);
+      dispatch(updateQBAttributeValue({ id, value: newValue }));
+    };
+
+    // Calculates the size of the input
+    const getInputWidth = () => {
+      if (value == '') return 1;
+      else if (value.length > 10) return 10;
+      return value.length;
+    };
+
+    const boolOperators = useMemo(
+      () => GetAttributeBoolOperators(data?.datatype),
+      [data?.datatype]
+    );
+
+    // Determine the backgroundcolor based on if the attribute is connected to a entity or relation
+    let bgcolor;
+    if (data?.attributeOfA == 'entity')
+      bgcolor = theme.palette.queryBuilder.entity.lighterbg;
+    else if (data?.attributeOfA == 'relation')
+      bgcolor = theme.palette.queryBuilder.relation.lighterbg;
+    else bgcolor = theme.palette.queryBuilder.attribute.background;
+
+    return (
+      <div
+        className={styles.attribute}
+        style={{
+          background: bgcolor,
+          color: theme.palette.queryBuilder.text,
+        }}
+      >
+        {/* <Handle
+        id={Handles.Attribute}
+        type="source"
+        position={Position.Bottom}
+        className={styles.handle}
+      /> */}
+        <div className={styles.contentWrapper}>
+          <span className={styles.content} title={data.name}>
+            {data.name}
+          </span>
+          <AttributeOperatorSelect
+            selected={data?.operator}
+            options={boolOperators}
+            changed={(o) =>
+              dispatch(updateQBAttributeOperator({ id, operator: o.value }))
+            }
+          />
+          <span className={styles.attributeInput}>
+            <input
+              style={{ maxWidth: `${getInputWidth()}ch` }}
+              type="string"
+              placeholder={'?'}
+              value={value}
+              onChange={onChange}
+              onBlur={validateInput}
+              onKeyDown={(e) => e.key == 'Enter' && validateInput()}
+            ></input>
+          </span>
+        </div>
+      </div>
+    );
+  }
+);
+
+export default AttributeRFPill;
diff --git a/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/attributepill/operatorselect.module.scss b/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/attributepill/operatorselect.module.scss
new file mode 100644
index 0000000000000000000000000000000000000000..1c31d2744f029e77c53efbfac2aae8983640770a
--- /dev/null
+++ b/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/attributepill/operatorselect.module.scss
@@ -0,0 +1,70 @@
+@use './variables.module.scss';
+
+.container {
+  position: relative;
+  vertical-align: baseline;
+  margin: 0 1ch;
+  font-weight: normal;
+  font-size: 7px;
+}
+
+.valueContainer {
+  color: #6a6a6a;
+  border: 1px solid rgba(0, 0, 0, 0);
+  border-radius: 2px;
+  background-color: transparent;
+
+  transition: border-color 0.2s;
+
+  height: variables.$height;
+  align-items: center;
+  display: flex;
+  padding: 0 1px 1px 1px;
+
+  &.highlighted,
+  &:hover {
+    border-color: rgba(0, 0, 0, 0.4);
+  }
+}
+
+.listbox {
+  font-size: 10px;
+  box-sizing: border-box;
+  padding: 5px;
+  margin: 5px 0 0 0;
+  list-style: none;
+  position: absolute;
+  height: auto;
+  box-shadow: 0 5px 13px -3px #e0e3e7;
+  background: white;
+  border: 1px solid #cdd2d7;
+  border-radius: 0.75em;
+  color: #1a2027;
+  overflow: auto;
+  z-index: 1;
+  outline: 0px;
+  left: -8px;
+
+  &.hidden {
+    opacity: 0;
+    visibility: hidden;
+    transition: opacity 0.4s 0.1s ease, visibility 0.4s 0.1s step-end;
+  }
+
+  & > li {
+    padding: 1px 4px;
+    border-radius: 2px;
+
+    &.selected {
+      background: #f1f1f1;
+    }
+
+    &:hover {
+      background: #e7ebf0;
+    }
+
+    &[aria-selected='true'] {
+      background: #e0e3e7;
+    }
+  }
+}
diff --git a/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/attributepill/operatorselect.tsx b/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/attributepill/operatorselect.tsx
new file mode 100644
index 0000000000000000000000000000000000000000..470c87618fdbd49fd8d70b3de926e0d3605cd12a
--- /dev/null
+++ b/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/attributepill/operatorselect.tsx
@@ -0,0 +1,85 @@
+import * as React from 'react';
+import { SelectOption } from '@mui/base';
+import styles from './operatorselect.module.scss';
+import { useRef, useState } from 'react';
+
+// const grey = {
+//   100: '#E7EBF0',
+//   200: '#E0E3E7',
+//   300: '#CDD2D7',
+//   400: '#B2BAC2',
+//   500: '#A0AAB4',
+//   600: '#6F7E8C',
+//   700: '#3E5060',
+//   800: '#2D3843',
+//   900: '#1A2027',
+// };
+
+interface Props {
+  options: SelectOption<string>[];
+  selected: string;
+  changed?: (newSelected: SelectOption<string>) => void;
+}
+
+function AttributeOperatorSelect({
+  options,
+  selected,
+  changed = () => {},
+}: Props) {
+  const listboxRef = useRef<HTMLUListElement>(null);
+  const [listboxVisible, setListboxVisible] = useState(false);
+  const [currSelected, setCurrSelected] = useState(
+    options.find((o) => o.value == selected)?.label || options[0].label
+  );
+
+  React.useEffect(() => {
+    if (listboxVisible) {
+      listboxRef.current?.focus();
+    }
+  }, [listboxVisible]);
+
+  const changeSelected = (option: SelectOption<string>) => {
+    if (option.label != currSelected) {
+      setCurrSelected(option.label);
+      changed(option);
+    }
+  };
+
+  return (
+    <div
+      className={styles.container}
+      // onMouseOver={() => setListboxVisible(true)}
+      onMouseOut={() => setListboxVisible(false)}
+      onClick={() => setListboxVisible(true)}
+      onFocus={() => setListboxVisible(true)}
+      onBlur={() => setListboxVisible(false)}
+    >
+      <div
+        className={
+          styles.valueContainer + ' ' + (listboxVisible && styles.highlighted)
+        }
+      >
+        <span>{currSelected}</span>
+      </div>
+      {options.length > 1 && (
+        <ul
+          className={styles.listbox + ' ' + (!listboxVisible && styles.hidden)}
+          ref={listboxRef}
+          onMouseOver={() => setListboxVisible(true)}
+        >
+          {options.map((option) => (
+            <li
+              className={option.label == currSelected ? styles.selected : ''}
+              key={option.value}
+              onClick={() => changeSelected(option)}
+            >
+              {option.label}
+            </li>
+          ))}
+        </ul>
+      )}
+    </div>
+  );
+}
+
+export default AttributeOperatorSelect;
diff --git a/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/attributepill/variables.module.scss b/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/attributepill/variables.module.scss
new file mode 100644
index 0000000000000000000000000000000000000000..08bc31bb67c70a043d0114880fceada930fe14b1
--- /dev/null
+++ b/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/attributepill/variables.module.scss
@@ -0,0 +1,3 @@
+$height: 5px;
+$fontsize: 6px;
+$ypad: 2px;
diff --git a/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/entitypill/entitypill.module.scss b/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/entitypill/entitypill.module.scss
new file mode 100644
index 0000000000000000000000000000000000000000..755d2b41d564abe3f9e4eb41f56865ef7d2432f4
--- /dev/null
+++ b/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/entitypill/entitypill.module.scss
@@ -0,0 +1,49 @@
+.entity {
+  display: flex;
+  font-family: monospace;
+  font-weight: bold;
+  font-size: 10px;
+  padding: 4px 2ch;
+  border-radius: 3px;
+}
+
+.highlighted {
+  box-shadow: black 0 0 2px;
+}
+
+.handleLeft {
+  border: 0px;
+  border-radius: 0px;
+  left: 12px;
+  width: 7px;
+  height: 7px;
+  margin-bottom: 11px;
+  background: rgba(255, 255, 255, 0.6);
+  box-shadow: 0 0 0 1px rgba(0, 0, 0, 0.3);
+  transform-origin: center;
+}
+
+// .handleBottom {
+//   border: 0px;
+//   border-radius: 0px;
+//   width: 7px;
+//   height: 7px;
+//   left: 27.5px;
+//   margin-bottom: 11px;
+//   background: rgba(255, 255, 255, 0.6);
+//   box-shadow: 0 0 0 1px rgba(0, 0, 0, 0.3);
+//   transform: rotate(-45deg);
+//   transform-origin: center;
+// }
+
+.contentWrapper {
+  margin-left: 3ch;
+
+  span {
+    max-width: 20ch;
+    text-overflow: ellipsis;
+    overflow: hidden;
+    white-space: nowrap;
+    display: block;
+  }
+}
diff --git a/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/entitypill/entitypill.tsx b/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/entitypill/entitypill.tsx
new file mode 100644
index 0000000000000000000000000000000000000000..1e5f6f7ba5d6a97bd5b47daea744a3a033cd09e7
--- /dev/null
+++ b/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/entitypill/entitypill.tsx
@@ -0,0 +1,63 @@
+import { handles } from '@graphpolaris/querybuilder/usecases';
+import { useTheme } from '@mui/material';
+import React, { useEffect } from 'react';
+import { FlowElement, Handle, Position } from 'react-flow-renderer';
+import styles from './entitypill.module.scss';
+import cn from 'classnames';
+
+// export const Handless = {
+//   entity: {
+//     attributes: 'attributesHandle',
+//     relations: 'relationsHandle',
+//   },
+// };
+
+// /** Links need handles to what they are connected to (and which side) */
+// export enum Handles {
+//   RelationLeft = 'leftEntityHandle', //target
+//   RelationRight = 'rightEntityHandle', //target
+//   ToAttributeHandle = 'attributesHandle', //target
+//   ToRelation = 'relationsHandle', //source
+//   Attribute = 'AttributeHandle', //source
+//   ReceiveFunction = 'receiveFunctionHandle', //target
+//   FunctionBase = 'functionHandle_', // + name from FunctionTypes args //source
+// }
+
+/**
+ * Component to render an entity flow element
+ * @param {FlowElement<EntityData>)} param0 The data of an entity flow element.
+ */
+export const EntityRFPill = React.memo(({ data }: { data: any }) => {
+  const theme = useTheme();
+
+  return (
+    <div
+      className={cn(styles.entity, {
+        [styles.highlighted]: data.suggestedForConnection,
+      })}
+      style={{
+        background: theme.palette.queryBuilder.entity.background,
+        color: theme.palette.queryBuilder.text,
+      }}
+    >
+      <Handle
+        id={handles.entity.relation}
+        type="source"
+        position={Position.Bottom}
+        className={styles.handleLeft}
+        style={data?.isConnected ? { backgroundColor: '#2e2e2e' } : {}}
+      />
+      {/* <Handle
+        id={Handles.ToAttributeHandle}
+        type="target"
+        position={Position.Bottom}
+        className={styles.handleBottom}
+      /> */}
+      <div className={styles.contentWrapper}>
+        <span title={data.name}>{data.name}</span>
+      </div>
+    </div>
+  );
+});
+
+export default EntityRFPill;
diff --git a/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/relationpill/relationpill.module.scss b/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/relationpill/relationpill.module.scss
new file mode 100644
index 0000000000000000000000000000000000000000..aff84b2e5d8b9caba6dd9fdd34a99b54614946d4
--- /dev/null
+++ b/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/relationpill/relationpill.module.scss
@@ -0,0 +1,110 @@
+.relation {
+  display: flex;
+  text-align: center;
+  font-family: monospace;
+  font-weight: bold;
+  font-size: 10px;
+  background-color: transparent;
+}
+
+.highlighted {
+  box-shadow: black 0 0 2px;
+}
+
+.contentWrapper {
+  display: flex;
+  align-items: center;
+
+  .handleLeft {
+    position: relative;
+    z-index: 3;
+
+    top: 25%;
+    border: 0px;
+    border-radius: 0px;
+
+    background: transparent;
+    transform-origin: center;
+
+    width: 0;
+    height: 0;
+    border-top: 5px solid transparent;
+    border-bottom: 5px solid transparent;
+    border-right: rgba(255, 255, 255, 0.7) 6px solid;
+
+    &::after {
+      content: '';
+      display: block;
+      position: absolute;
+      width: 0;
+      height: 0;
+      border-top: 7px solid transparent;
+      border-bottom: 7px solid transparent;
+      border-right: rgba(0, 0, 0, 0.1) 8px solid;
+      top: -7px;
+      right: -7px;
+    }
+  }
+  .highlighted {
+    z-index: -1;
+    box-shadow: 0 0 2px 1px gray;
+  }
+
+  .content {
+    margin: 0 2ch;
+    padding: 3px 0;
+    max-width: 20ch;
+    text-overflow: ellipsis;
+    overflow: hidden;
+    white-space: nowrap;
+  }
+
+  .handleRight {
+    position: relative;
+    top: 25%;
+    border: 0px;
+    border-radius: 0px;
+
+    background: transparent;
+    transform-origin: center;
+
+    width: 0;
+    height: 0;
+    border-top: 5px solid transparent;
+    border-bottom: 5px solid transparent;
+    border-left: rgba(255, 255, 255, 0.7) 6px solid;
+
+    &::after {
+      content: '';
+      display: block;
+      position: absolute;
+      width: 0;
+      height: 0;
+      border-top: 7px solid transparent;
+      border-bottom: 7px solid transparent;
+      border-left: rgba(0, 0, 0, 0.1) 8px solid;
+      top: -7px;
+      left: -7px;
+    }
+  }
+}
+
+$height: 10px;
+.arrowLeft {
+  z-index: 2;
+  width: 0;
+  height: 0;
+  border-top: $height solid transparent;
+  border-bottom: $height solid transparent;
+
+  border-right: $height solid;
+}
+
+.arrowRight {
+  width: 0;
+  height: 0;
+  border-top: $height solid transparent;
+  border-bottom: $height solid transparent;
+
+  border-left: $height solid;
+}
diff --git a/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/relationpill/relationpill.tsx b/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/relationpill/relationpill.tsx
new file mode 100644
index 0000000000000000000000000000000000000000..f1d4d8592cf367e719ac3db963948b5c06145ef1
--- /dev/null
+++ b/apps/web-graphpolaris/src/components/querybuilder/customFlowPills/relationpill/relationpill.tsx
@@ -0,0 +1,96 @@
+import { handles } from '@graphpolaris/querybuilder/usecases';
+import { useTheme } from '@mui/material';
+import { Handle, Position } from 'react-flow-renderer';
+import cn from 'classnames';
+
+import styles from './relationpill.module.scss';
+
+/**
+ * Component to render a relation flow element
+ * @param { FlowElement<RelationData>} param0 The data of a relation flow element.
+ */
+export default function RelationRFPill({ data }: { data: any }) {
+  const theme = useTheme();
+
+  // const minRef = useRef<HTMLInputElement>(null);
+  // const maxRef = useRef<HTMLInputElement>(null);
+
+  // const [readOnlyMin, setReadOnlyMin] = useState(true);
+  // const [readOnlyMax, setReadOnlyMax] = useState(true);
+
+  // const onDepthChanged = (depth: string) => {
+  //   // Don't allow depth above 99
+  //   const limit = 99;
+  //   if (data != undefined) {
+  //     data.depth.min = data.depth.min >= limit ? limit : data.depth.min;
+  //     data.depth.max = data.depth.max >= limit ? limit : data.depth.max;
+
+  //     // Check for for valid depth: min <= max
+  //     if (depth == 'min') {
+  //       if (data.depth.min > data.depth.max) data.depth.max = data.depth.min;
+  //       setReadOnlyMin(true);
+  //     } else if (depth == 'max') {
+  //       if (data.depth.max < data.depth.min) data.depth.min = data.depth.max;
+  //       setReadOnlyMax(true);
+  //     }
+
+  //     // Set to the correct width
+  //     if (maxRef.current)
+  //       maxRef.current.style.maxWidth = calcWidth(data.depth.max);
+  //     if (minRef.current)
+  //       minRef.current.style.maxWidth = calcWidth(data.depth.min);
+  //   }
+  // };
+
+  // const calcWidth = (data: number) => {
+  //   return data.toString().length + 0.5 + 'ch';
+  // };
+
+  return (
+    <div className={styles.relation}>
+      <div
+        className={styles.arrowLeft}
+        style={{
+          borderRightColor: theme.palette.queryBuilder.relation.background,
+        }}
+      />
+      <div
+        className={cn(styles.contentWrapper, {
+          [styles.highlighted]: data.suggestedForConnection,
+        })}
+        style={{
+          color: theme.palette.queryBuilder.text,
+          background: theme.palette.queryBuilder.relation.background,
+        }}
+      >
+        <Handle
+          id={handles.relation.fromEntity}
+          type="target"
+          position={Position.Left}
+          className={styles.handleLeft}
+          style={
+            data?.isFromEntityConnected ? { borderRightColor: '#2e2e2e' } : {} // TODO: this should be color from theme
+          }
+        />
+        <span className={styles.content} title={data.name}>
+          {data.name}
+        </span>
+        <Handle
+          id={handles.relation.toEntity}
+          type="source"
+          position={Position.Right}
+          className={styles.handleRight}
+          style={
+            data?.isToEntityConnected ? { borderLeftColor: '#2e2e2e' } : {}
+          }
+        />
+      </div>
+      <div
+        className={styles.arrowRight}
+        style={{
+          borderLeftColor: theme.palette.queryBuilder.relation.background,
+        }}
+      />
+    </div>
+  );
+}
diff --git a/apps/web-graphpolaris/src/components/querybuilder/querybuilder.module.scss b/apps/web-graphpolaris/src/components/querybuilder/querybuilder.module.scss
new file mode 100644
index 0000000000000000000000000000000000000000..db83f46964a46bd398e50188c6d83bd06769b52c
--- /dev/null
+++ b/apps/web-graphpolaris/src/components/querybuilder/querybuilder.module.scss
@@ -0,0 +1,8 @@
+.reactflow {
+  width: 100%;
+  height: 500px;
+
+  // :global(.react-flow__edges) {
+  //   z-index: 4 !important;
+  // }
+}
diff --git a/apps/web-graphpolaris/src/components/querybuilder/querybuilder.stories.tsx b/apps/web-graphpolaris/src/components/querybuilder/querybuilder.stories.tsx
new file mode 100644
index 0000000000000000000000000000000000000000..b406d60b5879ace0bfe90fa43b1a151fd83198f8
--- /dev/null
+++ b/apps/web-graphpolaris/src/components/querybuilder/querybuilder.stories.tsx
@@ -0,0 +1,97 @@
+import React from 'react';
+import {
+  colorPaletteConfigSlice,
+  querybuilderSlice,
+  setQuerybuilderNodes,
+} from '@graphpolaris/shared/data-access/store';
+import { GraphPolarisThemeProvider } from '@graphpolaris/shared/data-access/theme';
+import { configureStore } from '@reduxjs/toolkit';
+import { ComponentMeta, ComponentStory } from '@storybook/react';
+import { Provider } from 'react-redux';
+import QueryBuilder from './querybuilder';
+import { MultiGraph } from 'graphology';
+import { addPill, handles } from '@graphpolaris/querybuilder/usecases';
+
+export default {
+  component: QueryBuilder,
+} as ComponentMeta<typeof QueryBuilder>;
+
+const Template: ComponentStory<typeof QueryBuilder> = (args) => (
+  <QueryBuilder {...args} />
+);
+
+// Mock palette store
+const mockStore = configureStore({
+  reducer: {
+    colorPaletteConfig: colorPaletteConfigSlice.reducer,
+    querybuilder: querybuilderSlice.reducer,
+  },
+});
+const graph = new MultiGraph();
+addPill('0', { type: 'entity', x: 100, y: 100, name: 'Entity Pill' }, graph);
+// graph.addNode('0', { type: 'entity', x: 100, y: 100, name: 'Entity Pill' });
+addPill(
+  '1',
+  { type: 'relation', x: 140, y: 140, name: 'Relation Pill' },
+  graph
+);
+addPill(
+  '2',
+  {
+    type: 'attribute',
+    x: 170,
+    y: 160,
+    name: 'Attr string',
+    datatype: 'string',
+    operator: 'EQ',
+    value: 'mark',
+  },
+  graph
+);
+addPill(
+  '3',
+  {
+    type: 'attribute',
+    x: 170,
+    y: 170,
+    name: 'Attr number',
+    datatype: 'float',
+    operator: 'EQ',
+  },
+  graph
+);
+addPill(
+  '4',
+  {
+    type: 'attribute',
+    x: 130,
+    y: 120,
+    name: 'Attr bool',
+    datatype: 'bool',
+    operator: 'EQ',
+    value: 'true',
+  },
+  graph
+);
+console.log(graph.getNodeAttributes('2'));
+graph.addEdge('2', '1', { type: 'attribute_connection' });
+graph.addEdge('3', '1', { type: 'attribute_connection' });
+graph.addEdge('4', '0', { type: 'attribute_connection' });
+graph.addEdge('0', '1', {
+  type: 'entity_relation',
+  targetHandle: handles.relation.fromEntity,
+});
+// graph.addEdge('1', '0', {
+//   type: 'entity_relation',
+//   sourceHandle: handles.relation.entity,
+// });
+mockStore.dispatch(setQuerybuilderNodes(graph.export()));
+
+export const Simple = Template.bind({});
+Simple.decorators = [
+  (story) => (
+    <Provider store={mockStore}>
+      <GraphPolarisThemeProvider>{story()}</GraphPolarisThemeProvider>
+    </Provider>
+  ),
+];
diff --git a/apps/web-graphpolaris/src/components/querybuilder/querybuilder.tsx b/apps/web-graphpolaris/src/components/querybuilder/querybuilder.tsx
new file mode 100644
index 0000000000000000000000000000000000000000..3870f76de645a2d47757c5e0a40e2c8bea89503b
--- /dev/null
+++ b/apps/web-graphpolaris/src/components/querybuilder/querybuilder.tsx
@@ -0,0 +1,104 @@
+import {
+  createReactFlowElements,
+  dragPill,
+  dragPillStarted,
+  dragPillStopped,
+} from '@graphpolaris/querybuilder/usecases';
+import {
+  setQuerybuilderNodes,
+  useAppDispatch,
+  useQuerybuilderNodes,
+} from '@graphpolaris/shared/data-access/store';
+import { useMemo, useRef } from 'react';
+import ReactFlow, {
+  ReactFlowProvider,
+  Background,
+  Node,
+  isNode,
+} from 'react-flow-renderer';
+import styles from './querybuilder.module.scss';
+import ConnectionLine from './customFlowLines/connection';
+import ConnectionDragLine from './customFlowLines/connectionDrag';
+import AttributeRFPill from './customFlowPills/attributepill/attributepill';
+import EntityRFPill from './customFlowPills/entitypill/entitypill';
+import RelationRFPill from './customFlowPills/relationpill/relationpill';
+
+const nodeTypes = {
+  entity: EntityRFPill,
+  relation: RelationRFPill,
+  attribute: AttributeRFPill,
+};
+const edgeTypes = {
+  connection: ConnectionLine,
+};
+
+const onLoad = (reactFlowInstance: any) => {
+  setTimeout(() => reactFlowInstance.fitView(), 0);
+};
+
+const QueryBuilder = (props: {}) => {
+  const nodes = useQuerybuilderNodes();
+  const dispatch = useAppDispatch();
+  const isDraggingPill = useRef(false);
+
+  const elements = useMemo(() => createReactFlowElements(nodes), [nodes]);
+
+  const onNodeDrag = (
+    event: React.MouseEvent<Element, MouseEvent>,
+    node: Node<any>
+  ) => {
+    // Get the node in the elements list to get the previous location
+    const pNode = elements.find((e) => e.id == node.id);
+    if (!(pNode && isNode(pNode))) return;
+    // This is then used to calculate the delta position
+    const dx = node.position.x - pNode.position.x;
+    const dy = node.position.y - pNode.position.y;
+
+    // Check if we started dragging, if so, call the drag started usecase
+    if (!isDraggingPill.current) {
+      dragPillStarted(node.id, nodes);
+      isDraggingPill.current = true;
+    }
+
+    // Call the drag usecase
+    dragPill(node.id, nodes, dx, dy, node.position);
+
+    // Dispatch the new graphology object, so reactflow will get rerendered
+    dispatch(setQuerybuilderNodes(nodes.export()));
+  };
+  const onNodeDragStop = (
+    event: React.MouseEvent<Element, MouseEvent>,
+    node: Node<any>
+  ) => {
+    isDraggingPill.current = false;
+
+    // Call the drag pill stopped usecase
+    dragPillStopped(node.id, nodes);
+
+    // Dispatch the new graphology object, so reactflow will get rerendered
+    dispatch(setQuerybuilderNodes(nodes.export()));
+  };
+
+  return (
+    <div>
+      <ReactFlowProvider>
+        <ReactFlow
+          elements={elements}
+          snapGrid={[10, 10]}
+          // snapToGrid
+          nodeTypes={nodeTypes}
+          edgeTypes={edgeTypes}
+          connectionLineComponent={ConnectionDragLine}
+          onLoad={onLoad}
+          onNodeDrag={onNodeDrag}
+          onNodeDragStop={onNodeDragStop}
+          className={styles.reactflow}
+        >
+          <Background gap={10} size={0.7} />
+        </ReactFlow>
+      </ReactFlowProvider>
+    </div>
+  );
+};
+
+export default QueryBuilder;
diff --git a/deployment.yml b/deployment.yml
new file mode 100644
index 0000000000000000000000000000000000000000..fc945c18719db08baaf0e539fd9a16bcecb04609
--- /dev/null
+++ b/deployment.yml
@@ -0,0 +1,35 @@
+apiVersion: apps/v1
+kind: Deployment
+metadata:
+  name: frontend-webserver-deployment
+  labels:
+    app: frontend-webserver
+spec:
+  replicas: 1
+  selector:
+    matchLabels:
+      app: frontend-webserver
+  template:
+    metadata:
+      labels:
+        app: frontend-webserver
+    spec:
+      containers:
+        - name: container
+          image: datastropheregistry.azurecr.io/frontend:latest
+          ports:
+            - containerPort: 3000
+      imagePullSecrets:
+        - name: docker-regcred
+
+---
+kind: Service
+apiVersion: v1
+metadata:
+  name: frontend-webserver
+spec:
+  selector:
+    app: frontend-webserver
+  ports:
+    - port: 3000
+      targetPort: 80
diff --git a/libs/querybuilder/usecases/.babelrc b/libs/querybuilder/usecases/.babelrc
new file mode 100644
index 0000000000000000000000000000000000000000..cf7ddd99c615a064ac18eb3109eee4f394ab1faf
--- /dev/null
+++ b/libs/querybuilder/usecases/.babelrc
@@ -0,0 +1,3 @@
+{
+  "presets": [["@nrwl/web/babel", { "useBuiltIns": "usage" }]]
+}
diff --git a/libs/querybuilder/usecases/.eslintrc.json b/libs/querybuilder/usecases/.eslintrc.json
new file mode 100644
index 0000000000000000000000000000000000000000..3456be9b9036a42c593c82b050281230e4ca0ae4
--- /dev/null
+++ b/libs/querybuilder/usecases/.eslintrc.json
@@ -0,0 +1,18 @@
+{
+  "extends": ["../../../.eslintrc.json"],
+  "ignorePatterns": ["!**/*"],
+  "overrides": [
+    {
+      "files": ["*.ts", "*.tsx", "*.js", "*.jsx"],
+      "rules": {}
+    },
+    {
+      "files": ["*.ts", "*.tsx"],
+      "rules": {}
+    },
+    {
+      "files": ["*.js", "*.jsx"],
+      "rules": {}
+    }
+  ]
+}
diff --git a/libs/querybuilder/usecases/README.md b/libs/querybuilder/usecases/README.md
new file mode 100644
index 0000000000000000000000000000000000000000..f919d5ec5c4710fe9263506b367251e7e6a92baa
--- /dev/null
+++ b/libs/querybuilder/usecases/README.md
@@ -0,0 +1,7 @@
+# querybuilder-usecases
+
+This library was generated with [Nx](https://nx.dev).
+
+## Running unit tests
+
+Run `nx test querybuilder-usecases` to execute the unit tests via [Jest](https://jestjs.io).
diff --git a/libs/querybuilder/usecases/jest.config.js b/libs/querybuilder/usecases/jest.config.js
new file mode 100644
index 0000000000000000000000000000000000000000..941f4fbc2eed65bcc6b5a914c29364dcd1fff9bb
--- /dev/null
+++ b/libs/querybuilder/usecases/jest.config.js
@@ -0,0 +1,14 @@
+module.exports = {
+  displayName: 'querybuilder-usecases',
+  preset: '../../../jest.preset.js',
+  globals: {
+    'ts-jest': {
+      tsconfig: '<rootDir>/tsconfig.spec.json',
+    },
+  },
+  transform: {
+    '^.+\\.[tj]sx?$': 'ts-jest',
+  },
+  moduleFileExtensions: ['ts', 'tsx', 'js', 'jsx'],
+  coverageDirectory: '../../../coverage/libs/querybuilder/usecases',
+};
diff --git a/libs/querybuilder/usecases/project.json b/libs/querybuilder/usecases/project.json
new file mode 100644
index 0000000000000000000000000000000000000000..271a62582eacd5887d6adbf6a5027c3b12618eaf
--- /dev/null
+++ b/libs/querybuilder/usecases/project.json
@@ -0,0 +1,23 @@
+{
+  "root": "libs/querybuilder/usecases",
+  "sourceRoot": "libs/querybuilder/usecases/src",
+  "projectType": "library",
+  "targets": {
+    "lint": {
+      "executor": "@nrwl/linter:eslint",
+      "outputs": ["{options.outputFile}"],
+      "options": {
+        "lintFilePatterns": ["libs/querybuilder/usecases/**/*.ts"]
+      }
+    },
+    "test": {
+      "executor": "@nrwl/jest:jest",
+      "outputs": ["coverage/libs/querybuilder/usecases"],
+      "options": {
+        "jestConfig": "libs/querybuilder/usecases/jest.config.js",
+        "passWithNoTests": true
+      }
+    }
+  },
+  "tags": []
+}
diff --git a/libs/querybuilder/usecases/src/index.ts b/libs/querybuilder/usecases/src/index.ts
new file mode 100644
index 0000000000000000000000000000000000000000..32e0c81f2ecfe335cb61852728753c148b9c6c8b
--- /dev/null
+++ b/libs/querybuilder/usecases/src/index.ts
@@ -0,0 +1,6 @@
+export * from './lib/attribute/getAttributeBoolOperators';
+export * from './lib/attribute/checkInput';
+export * from './lib/createReactFlowElements';
+export * from './lib/pillHandles';
+export * from './lib/dragging/dragPill';
+export * from './lib/addPill';
diff --git a/libs/querybuilder/usecases/src/lib/addPill.ts b/libs/querybuilder/usecases/src/lib/addPill.ts
new file mode 100644
index 0000000000000000000000000000000000000000..e7b359d8f0fc77f1281fa57a7c200f13acfb820e
--- /dev/null
+++ b/libs/querybuilder/usecases/src/lib/addPill.ts
@@ -0,0 +1,94 @@
+import {
+  setQuerybuilderNodes,
+  store,
+} from '@graphpolaris/shared/data-access/store';
+import Graph from 'graphology';
+import { Attributes } from 'graphology-types';
+
+/** monospace fontsize table */
+const widthPerFontsize = {
+  6: 3.6167,
+  7: 4.2167,
+  10: 6.0167,
+};
+
+/** Adds a query builder pill to the graphology nodes object. */
+export function addPill(
+  id: string,
+  attributes: Attributes,
+  nodes: Graph
+): boolean {
+  const { type, name } = attributes;
+  if (!type || !name) return false;
+  let { x, y } = attributes;
+
+  // Check if x and y are present, otherwise set them to 0
+  if (!x) x = 0;
+  if (!y) y = 0;
+
+  // Get the width and height of a node
+  const { w, h } = calcWidthHeightOfPill(attributes);
+
+  // Add a node to the graphology object
+  nodes.addNode(id, { ...attributes, x, y, w, h });
+
+  // Set the new nodes in the query builder slice
+  store.dispatch(setQuerybuilderNodes(nodes.export()));
+
+  return true;
+}
+
+/** Calculates the width and height of a query builder pill.
+ * DEPENDS ON STYLING, if styling changed, change this.
+ */
+function calcWidthHeightOfPill(attributes: Attributes): {
+  w: number;
+  h: number;
+} {
+  const { type, name } = attributes;
+
+  let w = 0;
+  let h = 0;
+  switch (type) {
+    case 'entity': {
+      // calculate width and height of entity pill
+      w = Math.min(name.length, 20) * widthPerFontsize[10]; // for fontsize 10px
+
+      const widthOfPillWithoutText = 42.1164; // WARNING: depends on styling
+      w += widthOfPillWithoutText;
+      h = 20;
+      break;
+    }
+    case 'relation': {
+      // calculate width and height of relation pill
+      w = Math.min(name.length, 20) * widthPerFontsize[10]; // for fontsize 10px
+
+      const widthOfPillWithoutText = 56.0666; // WARNING: depends on styling
+      w += widthOfPillWithoutText;
+      h = 20;
+      break;
+    }
+    case 'attribute': {
+      // calculate width and height of relation pill
+      const pixelsPerChar = widthPerFontsize[6]; // for fontsize 10px
+      w = name.length * pixelsPerChar;
+
+      const { datatype, operator } = attributes;
+      let value = attributes['value'];
+      if (!datatype || !operator) return { w: 0, h: 0 };
+      if (!value) value = '?';
+
+      // Add width of operator
+      w += operator.length * widthPerFontsize[7];
+      // use a max of 10, because max-width is set to 10ch;
+      w += Math.min(value.length, 10) * widthPerFontsize[6];
+
+      const widthOfPillWithoutText = 25.6666; // WARNING: depends on styling
+      w += widthOfPillWithoutText;
+      h = 12;
+      break;
+    }
+  }
+
+  return { w, h };
+}
diff --git a/libs/querybuilder/usecases/src/lib/attribute/checkInput.ts b/libs/querybuilder/usecases/src/lib/attribute/checkInput.ts
new file mode 100644
index 0000000000000000000000000000000000000000..a1c29ab1f05d62105afaf80faaa405c30ce60535
--- /dev/null
+++ b/libs/querybuilder/usecases/src/lib/attribute/checkInput.ts
@@ -0,0 +1,34 @@
+/** Checks if the string input is a number. */
+function isNumber(x: string): boolean {
+  if (typeof x != 'string') return false;
+  return !Number.isNaN(x) && !Number.isNaN(parseFloat(x));
+}
+function isBoolean(s: string): boolean {
+  return s == 'true' || s == 'false' || s == '0' || s == '1';
+}
+function toBoolean(s: string): string {
+  if (s == '1' || s == 'true') return 'true';
+  return 'false';
+}
+
+/** Checks if the provided value has the same as the datatype of the attribute. */
+export function CheckDatatypeConstraint(type: string, str: string): string {
+  let res = '';
+  switch (type) {
+    case 'string':
+      res = str;
+      break;
+    case 'bool':
+      isBoolean(str) ? (res = toBoolean(str)) : (res = '');
+      break;
+    case 'int':
+    case 'float':
+    case 'number':
+      isNumber(str) ? (res = '' + parseFloat(str)) : (res = '');
+      break;
+    default:
+      res = str;
+      break;
+  }
+  return res;
+}
diff --git a/libs/querybuilder/usecases/src/lib/attribute/getAttributeBoolOperators.ts b/libs/querybuilder/usecases/src/lib/attribute/getAttributeBoolOperators.ts
new file mode 100644
index 0000000000000000000000000000000000000000..22809a7cde969f5be4da3978e650231a38ca1c4f
--- /dev/null
+++ b/libs/querybuilder/usecases/src/lib/attribute/getAttributeBoolOperators.ts
@@ -0,0 +1,63 @@
+/** Determines the available boolean operators for a certain datatype. */
+export function GetAttributeBoolOperators(
+  datatype: string
+): { label: string; value: string }[] {
+  switch (datatype) {
+    case 'text':
+    case 'string':
+      return [
+        {
+          label: '=',
+          value: 'EQ',
+        },
+        {
+          label: '≠',
+          value: 'NEQ',
+        },
+        {
+          label: 'inc',
+          value: 'includes',
+        },
+        {
+          label: 'exc',
+          value: 'excludes',
+        },
+      ];
+    case 'int':
+    case 'float':
+      return [
+        {
+          label: '=',
+          value: 'EQ',
+        },
+        {
+          label: '≠',
+          value: 'NEQ',
+        },
+        {
+          label: '>',
+          value: 'GT',
+        },
+        {
+          label: '≥',
+          value: 'GTE',
+        },
+        {
+          label: '<',
+          value: 'LT',
+        },
+        {
+          label: '≤',
+          value: 'LTE',
+        },
+      ];
+    case 'bool':
+    default:
+      return [
+        {
+          label: '=',
+          value: 'EQ',
+        },
+      ];
+  }
+}
diff --git a/libs/querybuilder/usecases/src/lib/createReactFlowElements.ts b/libs/querybuilder/usecases/src/lib/createReactFlowElements.ts
new file mode 100644
index 0000000000000000000000000000000000000000..eba3bf06477f9bd53ac9e1c87ea13126ecd58e02
--- /dev/null
+++ b/libs/querybuilder/usecases/src/lib/createReactFlowElements.ts
@@ -0,0 +1,130 @@
+import Graph, { MultiGraph } from 'graphology';
+import { Attributes } from 'graphology-types';
+import { Elements, Node, Edge, XYPosition } from 'react-flow-renderer';
+
+// Takes the querybuilder graph as an input and creates react flow elements for them.
+export function createReactFlowElements(graph: Graph): Elements<Node | Edge> {
+  const elements: Elements<Node | Edge> = [];
+
+  graph.forEachNode((node: string, attributes: Attributes): void => {
+    let data;
+    let position = { x: attributes?.x || 0, y: attributes?.y || 0 };
+
+    switch (attributes.type) {
+      case 'entity':
+        data = {
+          isConnected: graph
+            .neighbors(node)
+            .some((nb) => graph.getNodeAttribute(nb, 'type') == 'relation'),
+        };
+        break;
+      case 'relation':
+        data = {
+          isFromEntityConnected: graph
+            .inNeighbors(node)
+            .some((nb) => graph.getNodeAttribute(nb, 'type') == 'entity'),
+          isToEntityConnected: graph
+            .outNeighbors(node)
+            .some((nb) => graph.getNodeAttribute(nb, 'type') == 'entity'),
+        };
+        break;
+      case 'attribute': {
+        const ERNeighbors = graph.outNeighbors(node).filter((nb) => {
+          const type = graph.getNodeAttribute(nb, 'type');
+          return type == 'entity' || type == 'relation';
+        });
+        let attributeOfA = '';
+        if (ERNeighbors.length > 0)
+          attributeOfA = graph.getNodeAttribute(ERNeighbors[0], 'type');
+        data = {
+          datatype: attributes.datatype,
+          operator: attributes.operator,
+          value: attributes.value,
+          attributeOfA: attributeOfA,
+        };
+        // Get the position of the attribute, based on the connection to entity or relation
+        const p = getAttributePosition(node, graph);
+        if (p) position = p;
+        break;
+      }
+    }
+    // Each pill should have a name and type
+    data = {
+      ...data,
+      name: attributes.name,
+      suggestedForConnection: attributes.suggestedForConnection, // Highlights the pill, with shadow or something
+    };
+
+    const RFNode: Node = {
+      id: node,
+      type: attributes.type,
+      position: position,
+      data: data,
+    };
+    elements.push(RFNode);
+  });
+
+  // Add the reactflow edges
+  graph.forEachEdge((edge, attributes, source, target): void => {
+    // connection from attributes don't have visible connection lines
+    if (attributes.type == 'attribute_connection') return;
+
+    const RFEdge: Edge = {
+      id: edge,
+      source: source,
+      target: target,
+      type: 'connection',
+      sourceHandle: attributes.sourceHandle,
+      targetHandle: attributes.targetHandle,
+    };
+    elements.push(RFEdge);
+  });
+
+  return elements;
+}
+
+/** Gets the position of an attribute based on the connection to an entity or relation.
+ * It uses the position of the parent pill and what the index is of this attribute in all
+ * the connected attributes to the parent.
+ */
+function getAttributePosition(
+  id: string,
+  nodes: MultiGraph
+): XYPosition | undefined {
+  const nbs = nodes.filterOutNeighbors(id, (_, { type }) =>
+    ['entity', 'relation'].includes(type)
+  );
+
+  if (nbs.length > 1)
+    console.log(
+      'WARNING: attribute connected to  more than one entity or relation'
+    );
+  else if (nbs.length == 1) {
+    const nb = nbs[0];
+    const connectedAttributes = nodes.filterInNeighbors(
+      nb,
+      (_, { type }) => type == 'attribute'
+    );
+
+    // An entity can have more attributes, what is the attributes index in the attributes array of that entity?
+    let nthAttibute = -1;
+    for (let i = 0; i < connectedAttributes.length; i++) {
+      if (connectedAttributes[i] == id) {
+        nthAttibute = i;
+        break;
+      }
+    }
+
+    const nbAttr = nodes.getNodeAttributes(nb);
+
+    const pos = { x: nbAttr.x + 30, y: nbAttr.y + nbAttr.h };
+    // ASSUMES THAT EACH ATTRIBUTE HAS THE SAME HEIGHT
+    const heightOfAttributes = nodes.getNodeAttribute(id, 'h') - 1;
+    pos.y += nthAttibute * heightOfAttributes;
+
+    return pos;
+  }
+
+  // If the attribute has no (attribute_)connection, don't position it.
+  return undefined;
+}
diff --git a/libs/querybuilder/usecases/src/lib/dragging/dragAttribute.ts b/libs/querybuilder/usecases/src/lib/dragging/dragAttribute.ts
new file mode 100644
index 0000000000000000000000000000000000000000..df89ab57dcc526b42c151e868a50add049efbe7d
--- /dev/null
+++ b/libs/querybuilder/usecases/src/lib/dragging/dragAttribute.ts
@@ -0,0 +1,30 @@
+import { MultiGraph } from 'graphology';
+import { GetClosestPill } from './getClosestPill';
+
+export function DragAttributePillStarted(id: string, nodes: MultiGraph) {
+  // if the attribute is still connected to an entity or relation pill, disconnect
+  const es = nodes.outEdges(id);
+  es.forEach((e) => nodes.dropEdge(e));
+}
+
+export function DragAttributePill(
+  id: string,
+  nodes: MultiGraph,
+  dx: number,
+  dy: number
+) {
+  // Get the closes entity or relation node
+  const closestNode = GetClosestPill(id, nodes, ['entity', 'relation']);
+  // If we found one, highlight it by adding an attribute
+  if (closestNode)
+    nodes.setNodeAttribute(closestNode, 'suggestedForConnection', true);
+}
+
+export function DragAttibutePillStopped(id: string, nodes: MultiGraph) {
+  // If there is currently a node with the suggestedForConnection attribute
+  // connect this attribute to it
+  nodes.forEachNode((node, { suggestedForConnection }) => {
+    if (suggestedForConnection)
+      nodes.addEdge(id, node, { type: 'attribute_connection' });
+  });
+}
diff --git a/libs/querybuilder/usecases/src/lib/dragging/dragAttributesAlong.ts b/libs/querybuilder/usecases/src/lib/dragging/dragAttributesAlong.ts
new file mode 100644
index 0000000000000000000000000000000000000000..56697ab3f9cd125629c309c9df699419365a4a49
--- /dev/null
+++ b/libs/querybuilder/usecases/src/lib/dragging/dragAttributesAlong.ts
@@ -0,0 +1,27 @@
+import Graph from 'graphology';
+
+/**
+ * Changes the position of connected attributes.
+ * @param id The id of the node which could have attributes connected to it (entity or relation)
+ * @param nodes The graphology query builder object
+ * @param dx The change in x
+ * @param dy The change in y
+ * @returns True if any attribute positions were changed
+ */
+export function DragAttributesAlong(
+  id: string,
+  nodes: Graph,
+  dx: number,
+  dy: number
+): boolean {
+  let didChangeAttributes = false;
+  nodes.forEachInNeighbor(id, (nb) => {
+    if (nodes.getNodeAttribute(nb, 'type') == 'attribute') {
+      nodes.updateNodeAttribute(nb, 'x', (x) => x + dx);
+      nodes.updateNodeAttribute(nb, 'y', (y) => y + dy);
+      didChangeAttributes = true;
+    }
+  });
+
+  return didChangeAttributes;
+}
diff --git a/libs/querybuilder/usecases/src/lib/dragging/dragEntity.ts b/libs/querybuilder/usecases/src/lib/dragging/dragEntity.ts
new file mode 100644
index 0000000000000000000000000000000000000000..171124ac9330a007b531311fec85dceeb4ccf945
--- /dev/null
+++ b/libs/querybuilder/usecases/src/lib/dragging/dragEntity.ts
@@ -0,0 +1,18 @@
+import { MultiGraph } from 'graphology';
+
+export function DragEntityPillStarted(id: string, nodes: MultiGraph) {
+  // Started dragging entity usecase
+}
+
+export function DragEntityPill(
+  id: string,
+  nodes: MultiGraph,
+  dx: number,
+  dy: number
+) {
+  // Code for dragging an entity pill should go here
+}
+
+export function DragEntityPillStopped(id: string, nodes: MultiGraph) {
+  // Stopped dragging entity pill
+}
diff --git a/libs/querybuilder/usecases/src/lib/dragging/dragPill.ts b/libs/querybuilder/usecases/src/lib/dragging/dragPill.ts
new file mode 100644
index 0000000000000000000000000000000000000000..3832c103d94fbb2d759edc082cd643051580712a
--- /dev/null
+++ b/libs/querybuilder/usecases/src/lib/dragging/dragPill.ts
@@ -0,0 +1,90 @@
+import { MultiGraph } from 'graphology';
+import { XYPosition } from 'react-flow-renderer';
+import {
+  DragAttibutePillStopped,
+  DragAttributePill,
+  DragAttributePillStarted,
+} from './dragAttribute';
+import { DragAttributesAlong } from './dragAttributesAlong';
+import {
+  DragEntityPill,
+  DragEntityPillStarted,
+  DragEntityPillStopped,
+} from './dragEntity';
+import {
+  DragRelationPill,
+  DragRelationPillStarted,
+  DragRelationPillStopped,
+} from './dragRelation';
+
+export function dragPillStarted(id: string, nodes: MultiGraph) {
+  switch (nodes.getNodeAttribute(id, 'type')) {
+    case 'attribute':
+      DragAttributePillStarted(id, nodes);
+      break;
+    case 'entity':
+      DragEntityPillStarted(id, nodes);
+      break;
+    case 'relation':
+      DragRelationPillStarted(id, nodes);
+      break;
+  }
+}
+
+/**
+ * A general drag usecase for any pill, it will select the correct usecase for each pill
+ * @param id
+ * @param nodes The graphology query builder nodes object
+ * @param dx Delta x
+ * @param dy Delta y
+ * @param position The already updated positiong (dx dy are already applied)
+ */
+export function dragPill(
+  id: string,
+  nodes: MultiGraph,
+  dx: number,
+  dy: number,
+  position: XYPosition
+) {
+  // Update the position of the node in the graphology object
+  nodes.setNodeAttribute(id, 'x', position.x);
+  nodes.setNodeAttribute(id, 'y', position.y);
+
+  // Remove the highlighted attribute from each node
+  nodes.forEachNode((node) =>
+    nodes.removeNodeAttribute(node, 'suggestedForConnection')
+  );
+
+  switch (nodes.getNodeAttribute(id, 'type')) {
+    case 'attribute':
+      DragAttributePill(id, nodes, dx, dy);
+      break;
+    case 'entity':
+      DragAttributesAlong(id, nodes, dx, dy);
+      DragEntityPill(id, nodes, dx, dy);
+      break;
+    case 'relation':
+      DragAttributesAlong(id, nodes, dx, dy);
+      DragRelationPill(id, nodes, dx, dy);
+      break;
+  }
+}
+
+export function dragPillStopped(id: string, nodes: MultiGraph) {
+  switch (nodes.getNodeAttribute(id, 'type')) {
+    case 'attribute':
+      DragAttibutePillStopped(id, nodes);
+      break;
+    case 'entity':
+      DragEntityPillStopped(id, nodes);
+      break;
+    case 'relation':
+      DragRelationPillStopped(id, nodes);
+      break;
+  }
+
+  // Remove all suggestedForConnection attributes
+  nodes.forEachNode((node) =>
+    nodes.removeNodeAttribute(node, 'suggestedForConnection')
+  );
+}
diff --git a/libs/querybuilder/usecases/src/lib/dragging/dragRelation.ts b/libs/querybuilder/usecases/src/lib/dragging/dragRelation.ts
new file mode 100644
index 0000000000000000000000000000000000000000..12bf51961ae8cff27113c73a14c81d1e6c346b91
--- /dev/null
+++ b/libs/querybuilder/usecases/src/lib/dragging/dragRelation.ts
@@ -0,0 +1,18 @@
+import { MultiGraph } from 'graphology';
+
+export function DragRelationPillStarted(id: string, nodes: MultiGraph) {
+  // Started dragging relation usecase
+}
+
+export function DragRelationPill(
+  id: string,
+  nodes: MultiGraph,
+  dx: number,
+  dy: number
+) {
+  // Code for dragging an relation pill should go here
+}
+
+export function DragRelationPillStopped(id: string, nodes: MultiGraph) {
+  // Stopped dragging relation pill
+}
diff --git a/libs/querybuilder/usecases/src/lib/dragging/getClosestPill.ts b/libs/querybuilder/usecases/src/lib/dragging/getClosestPill.ts
new file mode 100644
index 0000000000000000000000000000000000000000..0a3fd44874fa8de1f5b3d1508d456dfae0848faa
--- /dev/null
+++ b/libs/querybuilder/usecases/src/lib/dragging/getClosestPill.ts
@@ -0,0 +1,40 @@
+import { MultiGraph } from 'graphology';
+
+/**
+ * Gets the closest node to id
+ * @param id
+ * @param nodes Graphology querybuilder MultiGraph object
+ * @param allowedNodeTypes An array of the node types which are included in the search
+ * @param maxDistance The maximum distance
+ * @returns the closest node if within range
+ */
+export function GetClosestPill(
+  id: string,
+  nodes: MultiGraph,
+  allowedNodeTypes: string[],
+  maxDistance = 150
+): string | undefined {
+  const { x, y, w, h } = nodes.getNodeAttributes(id);
+  const center: { x: number; y: number } = { x: x + w / 2, y: y + h / 2 };
+
+  let minDist = maxDistance * maxDistance;
+  let closestNode: string | undefined = undefined;
+  nodes.forEachNode((node, { x, y, w, h, type }) => {
+    if (allowedNodeTypes.includes(type)) {
+      const nodeCenter: { x: number; y: number } = {
+        x: x + w / 2,
+        y: y + h / 2,
+      };
+
+      const dx = center.x - nodeCenter.x;
+      const dy = center.y - nodeCenter.y;
+      const dist = dx * dx + dy * dy;
+      if (dist < minDist) {
+        minDist = dist;
+        closestNode = node;
+      }
+    }
+  });
+
+  return closestNode;
+}
diff --git a/libs/querybuilder/usecases/src/lib/pillHandles.ts b/libs/querybuilder/usecases/src/lib/pillHandles.ts
new file mode 100644
index 0000000000000000000000000000000000000000..355bdd8e5ce2cd821e49d3bef74bb9d925637b50
--- /dev/null
+++ b/libs/querybuilder/usecases/src/lib/pillHandles.ts
@@ -0,0 +1,14 @@
+// This file describes the connection points (handles) on a query builder pill
+// For example the connection from entity to left relation handle
+
+export const handles = {
+  entity: {
+    /** The handle for a connection from an entity to a relation pill */
+    relation: 'entity:to_relation',
+  },
+  relation: {
+    /** The handle for a connection from a relation to an entity pill */
+    toEntity: 'relation:to_entity',
+    fromEntity: 'relation:from_entity',
+  },
+};
diff --git a/libs/querybuilder/usecases/src/lib/querybuilder-usecases.spec.ts b/libs/querybuilder/usecases/src/lib/querybuilder-usecases.spec.ts
new file mode 100644
index 0000000000000000000000000000000000000000..684984e68f06f7cff30cc3f8c00749fd61c1aa9b
--- /dev/null
+++ b/libs/querybuilder/usecases/src/lib/querybuilder-usecases.spec.ts
@@ -0,0 +1,7 @@
+import { querybuilderUsecases } from './querybuilder-usecases';
+
+describe('querybuilderUsecases', () => {
+  it('should work', () => {
+    expect(querybuilderUsecases()).toEqual('querybuilder-usecases');
+  });
+});
diff --git a/libs/querybuilder/usecases/src/lib/querybuilder-usecases.ts b/libs/querybuilder/usecases/src/lib/querybuilder-usecases.ts
new file mode 100644
index 0000000000000000000000000000000000000000..06d687eb90fbdd6d6e752c90417a12a854aff52b
--- /dev/null
+++ b/libs/querybuilder/usecases/src/lib/querybuilder-usecases.ts
@@ -0,0 +1,3 @@
+export function querybuilderUsecases(): string {
+  return 'querybuilder-usecases';
+}
diff --git a/libs/querybuilder/usecases/tsconfig.json b/libs/querybuilder/usecases/tsconfig.json
new file mode 100644
index 0000000000000000000000000000000000000000..6ebadfb9de07f71cd20ee1102f3512550505ad2a
--- /dev/null
+++ b/libs/querybuilder/usecases/tsconfig.json
@@ -0,0 +1,20 @@
+{
+  "extends": "../../../tsconfig.base.json",
+  "files": [],
+  "include": [],
+  "references": [
+    {
+      "path": "./tsconfig.lib.json"
+    },
+    {
+      "path": "./tsconfig.spec.json"
+    }
+  ],
+  "compilerOptions": {
+    "forceConsistentCasingInFileNames": true,
+    "strict": true,
+    "noImplicitReturns": true,
+    "noFallthroughCasesInSwitch": true,
+    "resolveJsonModule": true
+  }
+}
diff --git a/libs/querybuilder/usecases/tsconfig.lib.json b/libs/querybuilder/usecases/tsconfig.lib.json
new file mode 100644
index 0000000000000000000000000000000000000000..efdd77fbf5b34f06e8efa8ad8bc87e11a3c1e9af
--- /dev/null
+++ b/libs/querybuilder/usecases/tsconfig.lib.json
@@ -0,0 +1,10 @@
+{
+  "extends": "./tsconfig.json",
+  "compilerOptions": {
+    "outDir": "../../../dist/out-tsc",
+    "declaration": true,
+    "types": []
+  },
+  "include": ["**/*.ts"],
+  "exclude": ["**/*.spec.ts"]
+}
diff --git a/libs/querybuilder/usecases/tsconfig.spec.json b/libs/querybuilder/usecases/tsconfig.spec.json
new file mode 100644
index 0000000000000000000000000000000000000000..d8716fecfa3b7929f162b71e7a966c579a63c071
--- /dev/null
+++ b/libs/querybuilder/usecases/tsconfig.spec.json
@@ -0,0 +1,19 @@
+{
+  "extends": "./tsconfig.json",
+  "compilerOptions": {
+    "outDir": "../../../dist/out-tsc",
+    "module": "commonjs",
+    "types": ["jest", "node"]
+  },
+  "include": [
+    "**/*.test.ts",
+    "**/*.spec.ts",
+    "**/*.test.tsx",
+    "**/*.spec.tsx",
+    "**/*.test.js",
+    "**/*.spec.js",
+    "**/*.test.jsx",
+    "**/*.spec.jsx",
+    "**/*.d.ts"
+  ]
+}
diff --git a/libs/shared/data-access/api/src/lib/database.ts b/libs/shared/data-access/api/src/lib/database.ts
index 8d6700de6f55dde670227d1cedf1080c44637e28..da706c42a71c1cedc8a777afd5870480b3ec17eb 100644
--- a/libs/shared/data-access/api/src/lib/database.ts
+++ b/libs/shared/data-access/api/src/lib/database.ts
@@ -14,7 +14,7 @@ export type AddDatabaseRequest = {
 
 export function AddDatabase(request: AddDatabaseRequest): Promise<void> {
   return new Promise((resolve, reject) => {
-    fetch('https://datastrophe.science.uu.nl/user/database', {
+    fetch('https://api.graphpolaris.com/user/database', {
       method: 'POST',
       credentials: 'same-origin',
       headers: new Headers({
@@ -34,7 +34,7 @@ export function AddDatabase(request: AddDatabaseRequest): Promise<void> {
 
 export function GetAllDatabases(): Promise<Array<string>> {
   return new Promise<Array<string>>((resolve, reject) => {
-    fetch('https://datastrophe.science.uu.nl/user/database', {
+    fetch('https://api.graphpolaris.com/user/database', {
       method: 'GET',
       credentials: 'same-origin',
       headers: new Headers({
@@ -57,7 +57,7 @@ export function GetAllDatabases(): Promise<Array<string>> {
 
 export function DeleteDatabase(name: string): Promise<void> {
   return new Promise((resolve, reject) => {
-    fetch('https://datastrophe.science.uu.nl/user/database/' + name, {
+    fetch('https://api.graphpolaris.com/user/database/' + name, {
       method: 'DELETE',
       credentials: 'same-origin',
       headers: new Headers({
diff --git a/libs/shared/data-access/api/src/lib/user.ts b/libs/shared/data-access/api/src/lib/user.ts
index 457ce20f6af6c8be31727f2b9757134827b4fc70..b3d7b3fc9e6b5411369a4a5b11e3ed8e7e009065 100644
--- a/libs/shared/data-access/api/src/lib/user.ts
+++ b/libs/shared/data-access/api/src/lib/user.ts
@@ -10,7 +10,7 @@ export type User = {
 
 export function GetUserInfo(): Promise<User> {
   return new Promise<User>((resolve, reject) => {
-    fetch('https://datastrophe.science.uu.nl/user/', {
+    fetch('https://api.graphpolaris.com/user/', {
       method: 'GET',
       credentials: 'same-origin',
       headers: new Headers({
diff --git a/libs/shared/data-access/authorization/src/lib/authorizationHandler.ts b/libs/shared/data-access/authorization/src/lib/authorizationHandler.ts
index fc92922419ad609e85767e55dd35b06b2a581012..bc6001fd2aed8d24d60abf15411167af64629e81 100644
--- a/libs/shared/data-access/authorization/src/lib/authorizationHandler.ts
+++ b/libs/shared/data-access/authorization/src/lib/authorizationHandler.ts
@@ -1,5 +1,3 @@
-import { Cookies } from 'react-cookie';
-
 export class AuthorizationHandler {
   private static _instance: AuthorizationHandler;
   private accessToken = '';
@@ -52,7 +50,7 @@ export class AuthorizationHandler {
    */
   private async getNewAccessToken(): Promise<authResponse> {
     // If we have an access token already, append it to the url as a query param to keep sessionID the same
-    let url = 'https://datastrophe.science.uu.nl/auth/refresh';
+    let url = 'https://api.graphpolaris.com/auth/refresh';
     if (this.accessToken != '') {
       url += '?access_token=' + this.accessToken;
     }
@@ -104,7 +102,7 @@ export class AuthorizationHandler {
    * initialiseRefreshToken attempts to initialise a refresh token
    */
   private async initialiseRefreshToken() {
-    fetch('https://datastrophe.science.uu.nl/auth/refresh', {
+    fetch('https://api.graphpolaris.com/auth/refresh', {
       method: 'POST',
       credentials: 'include',
     })
diff --git a/libs/shared/data-access/store/src/index.ts b/libs/shared/data-access/store/src/index.ts
index c855ded7f0ce61ed13dc286d637aff7a2c3446ef..3e2aec2be9ea77de09c1c598a15a1ef30fe68713 100644
--- a/libs/shared/data-access/store/src/index.ts
+++ b/libs/shared/data-access/store/src/index.ts
@@ -7,6 +7,12 @@ export {
   schemaSlice,
   selectSchemaLayout
 } from './lib/schemaSlice';
+export {
+  querybuilderSlice,
+  setQuerybuilderNodes,
+  updateQBAttributeOperator,
+  updateQBAttributeValue,
+} from './lib/querybuilderSlice';
 export {
   selectGraphQueryResult,
   selectGraphQueryResultLinks,
diff --git a/libs/shared/data-access/store/src/lib/colorPaletteConfigSlice.ts b/libs/shared/data-access/store/src/lib/colorPaletteConfigSlice.ts
index bb41dc4a815f5dccb73c3ac51de8ed57728483e0..64d340262ec623a4f427a7c5e149c350beaff8d5 100644
--- a/libs/shared/data-access/store/src/lib/colorPaletteConfigSlice.ts
+++ b/libs/shared/data-access/store/src/lib/colorPaletteConfigSlice.ts
@@ -1,4 +1,3 @@
-import { palette } from '@mui/system';
 import { createSlice, PayloadAction } from '@reduxjs/toolkit';
 import type { RootState } from './store';
 
@@ -6,6 +5,22 @@ import type { RootState } from './store';
 export interface ExtraColorsForMui5 {
   /** Colors that can be used for data visualisation, e.g. nodes, edges */
   dataPointColors: string[];
+
+  queryBuilder: {
+    text: string;
+    entity: {
+      background: string;
+      lighterbg?: string;
+    };
+    relation: {
+      background: string;
+      lighterbg?: string;
+    };
+    attribute: {
+      background: string;
+      lighterbg?: string;
+    };
+  };
 }
 
 /** Our custom color palette config. With the palette options from MUI that we are going to use. */
@@ -45,7 +60,21 @@ export interface ColorPaletteConfig {
 // But we don't reference that type directly here to stay decoupled
 export const initialState: ColorPaletteConfig = {
   lightPalette: {
-    custom: { dataPointColors: ['#ff0000', '#00ff00', '#0000ff'] },
+    custom: {
+      dataPointColors: ['#ff0000', '#00ff00', '#0000ff'],
+      queryBuilder: {
+        text: 'black',
+        entity: {
+          background: '#FC4F4F',
+        },
+        relation: {
+          background: '#FF9F45',
+        },
+        attribute: {
+          background: '#C7C7C7',
+        },
+      },
+    },
     // If light and dark are not set, these will be calculated using main.
     // light/dark have nothing with darkmode, these are just a light and dark variation
     primary: {
@@ -60,7 +89,21 @@ export const initialState: ColorPaletteConfig = {
     },
   },
   darkPalette: {
-    custom: { dataPointColors: ['#ff0000', '#00ff00', '#0000ff'] },
+    custom: {
+      dataPointColors: ['#ff0000', '#00ff00', '#0000ff'],
+      queryBuilder: {
+        text: 'black',
+        entity: {
+          background: '#FC4F4F',
+        },
+        relation: {
+          background: '#FF9F45',
+        },
+        attribute: {
+          background: '#C7C7C7',
+        },
+      },
+    },
     primary: {
       main: '#e3f3fd',
     },
diff --git a/libs/shared/data-access/store/src/lib/hooks.ts b/libs/shared/data-access/store/src/lib/hooks.ts
index 801f34a8077abdcdd9efdf2bfaa4161f00c10e26..6b67cbead6799c62e78c19c23f58830593e2d195 100644
--- a/libs/shared/data-access/store/src/lib/hooks.ts
+++ b/libs/shared/data-access/store/src/lib/hooks.ts
@@ -1,18 +1,20 @@
 import { TypedUseSelectorHook, useDispatch, useSelector } from 'react-redux';
 import { selectGraphQueryResult } from './graphQueryResultSlice';
 import { selectSchema, selectSchemaLayout } from './schemaSlice';
+import { selectQuerybuilderNodes } from './querybuilderSlice';
 import type { RootState, AppDispatch } from './store';
 
 // Use throughout your app instead of plain `useDispatch` and `useSelector`
 export const useAppDispatch = () => useDispatch<AppDispatch>();
 export const useAppSelector: TypedUseSelectorHook<RootState> = useSelector;
 
-// Gives the graphQueryResult from the store
+/** Gives the graphQueryResult from the store */
 export const useGraphQueryResult = () => useAppSelector(selectGraphQueryResult);
 
 // Gives the schema form the store (as a graphology object)
 export const useSchema = () => useAppSelector(selectSchema);
 
-
 // Gives the schema form the store (as a graphology object)
-export const useSchemaLayout = () => useAppSelector(selectSchemaLayout);
\ No newline at end of file
+export const useSchemaLayout = () => useAppSelector(selectSchemaLayout);
+export const useQuerybuilderNodes = () =>
+  useAppSelector(selectQuerybuilderNodes);
diff --git a/libs/shared/data-access/store/src/lib/querybuilderSlice.ts b/libs/shared/data-access/store/src/lib/querybuilderSlice.ts
new file mode 100644
index 0000000000000000000000000000000000000000..877444a06b9ebe70c06a866e9ecc6d28efc001b5
--- /dev/null
+++ b/libs/shared/data-access/store/src/lib/querybuilderSlice.ts
@@ -0,0 +1,79 @@
+import { createSlice, PayloadAction } from '@reduxjs/toolkit';
+import type { RootState } from './store';
+import { MultiGraph } from 'graphology';
+import { Attributes, SerializedGraph } from 'graphology-types';
+
+// Define the initial state using that type
+export const initialState = {
+  graphologySerialized: new MultiGraph().export(),
+  // schemaLayout: 'Graphology_noverlap',
+};
+
+export const querybuilderSlice = createSlice({
+  name: 'querybuilder',
+  // `createSlice` will infer the state type from the `initialState` argument
+  initialState,
+  reducers: {
+    setQuerybuilderNodes: (
+      state,
+      action: PayloadAction<SerializedGraph<Attributes, Attributes, Attributes>>
+    ) => {
+      state.graphologySerialized = action.payload;
+    },
+    updateQBAttributeOperator: (
+      state,
+      action: PayloadAction<{ id: string; operator: string }>
+    ) => {
+      const graph = MultiGraph.from(state.graphologySerialized);
+      graph.setNodeAttribute(
+        action.payload.id,
+        'operator',
+        action.payload.operator
+      );
+      state.graphologySerialized = graph.export();
+    },
+    updateQBAttributeValue: (
+      state,
+      action: PayloadAction<{ id: string; value: string }>
+    ) => {
+      const graph = MultiGraph.from(state.graphologySerialized);
+      graph.setNodeAttribute(action.payload.id, 'value', action.payload.value);
+      state.graphologySerialized = graph.export();
+    },
+    // addQuerybuilderNode: (
+    //   state,
+    //   action: PayloadAction<{ id: string; attributes: Attributes }>
+    // ) => {
+    //   const graph = MultiGraph.from(state.graphologySerialized);
+    //   graph.addNode(action.payload.id, action.payload.attributes);
+    //   state.graphologySerialized = graph.export();
+    // },
+    // setGraphLayout: (state, action: PayloadAction<AllLayoutAlgorithms>) => {
+    //   state.schemaLayout = action.payload;
+    // },
+  },
+});
+
+export const {
+  setQuerybuilderNodes,
+  updateQBAttributeOperator,
+  updateQBAttributeValue,
+} = querybuilderSlice.actions;
+
+/** Select the querybuilder nodes and convert it to a graphology object */
+export const selectQuerybuilderNodes = (state: RootState): MultiGraph => {
+  // This is really weird but for some reason all the attributes appeared as read-only otherwise
+  return MultiGraph.from(
+    MultiGraph.from(state.querybuilder.graphologySerialized).export()
+  );
+};
+
+// /**
+//  * selects the SchemaLayout enum
+//  * @param {GraphLayout} state
+//  * @returns {GraphLayout} enum of type GraphLayout
+//  */
+// export const selectSchemaLayout = (state: RootState) =>
+//   state.schema.schemaLayout;
+
+export default querybuilderSlice.reducer;
diff --git a/libs/shared/data-access/store/src/lib/schemaSlice.ts b/libs/shared/data-access/store/src/lib/schemaSlice.ts
index 4d0afa25e6ee20c813fb242805f3fbd0bcf3032b..c7d1febd677be2187fb20b67074f1839e87c9770 100644
--- a/libs/shared/data-access/store/src/lib/schemaSlice.ts
+++ b/libs/shared/data-access/store/src/lib/schemaSlice.ts
@@ -79,8 +79,10 @@ export const { readInSchemaFromBackend, setSchema } = schemaSlice.actions;
  * Select the schema and convert it to a graphology object
  * */
 export const selectSchema = (state: RootState) => {
-  // console.log(state);
-  return MultiGraph.from(state.schema.graphologySerialized);
+  // This is really weird but for some reason all the attributes appeared as read-only otherwise
+  return MultiGraph.from(
+    MultiGraph.from(state.schema.graphologySerialized).export()
+  );
 };
 
 // /**
diff --git a/libs/shared/data-access/store/src/lib/store.ts b/libs/shared/data-access/store/src/lib/store.ts
index 7198337bde3e8b98b11d6336552743c71d8e413a..30a41ddb43d55a86517bb818054405da4e45daf6 100644
--- a/libs/shared/data-access/store/src/lib/store.ts
+++ b/libs/shared/data-access/store/src/lib/store.ts
@@ -1,6 +1,7 @@
 import { configureStore } from '@reduxjs/toolkit';
 import colorPaletteConfigSlice from './colorPaletteConfigSlice';
 import graphQueryResultSlice from './graphQueryResultSlice';
+import querybuilderSlice from './querybuilderSlice';
 import schemaSlice from './schemaSlice';
 
 export const store = configureStore({
@@ -8,6 +9,7 @@ export const store = configureStore({
     graphQueryResult: graphQueryResultSlice,
     schema: schemaSlice,
     colorPaletteConfig: colorPaletteConfigSlice,
+    querybuilder: querybuilderSlice,
   },
 });
 
diff --git a/libs/shared/data-access/theme/src/lib/mapColorsConfigToMuiTheme.ts b/libs/shared/data-access/theme/src/lib/mapColorsConfigToMuiTheme.ts
index 43235684422e51bacf514cf32e21f1bf69c7305b..aa9a6de51d703d3d79d1c03ab6f573e24d5e5cf4 100644
--- a/libs/shared/data-access/theme/src/lib/mapColorsConfigToMuiTheme.ts
+++ b/libs/shared/data-access/theme/src/lib/mapColorsConfigToMuiTheme.ts
@@ -1,5 +1,6 @@
 import { ColorPaletteConfig } from '@graphpolaris/shared/data-access/store';
 import { ThemeOptions } from '@mui/material/styles';
+import Color from 'color';
 
 /** Maps our color palette config (stored in redux) to the mui 5 theme format */
 export default function MapColorsConfigToMuiTheme(
@@ -14,11 +15,70 @@ export default function MapColorsConfigToMuiTheme(
             primary: colorsConfig.darkPalette.primary,
             secondary: colorsConfig.darkPalette.secondary,
             dataPointColors: colorsConfig.darkPalette.custom.dataPointColors,
+            queryBuilder: {
+              text: colorsConfig.darkPalette.custom.queryBuilder.text,
+              entity: {
+                background:
+                  colorsConfig.darkPalette.custom.queryBuilder.entity
+                    .background,
+                lighterbg: Color(
+                  colorsConfig.darkPalette.custom.queryBuilder.entity.background
+                )
+                  .lighten(0.2)
+                  .toString(),
+              },
+              relation: {
+                background:
+                  colorsConfig.darkPalette.custom.queryBuilder.relation
+                    .background,
+                lighterbg: Color(
+                  colorsConfig.darkPalette.custom.queryBuilder.relation
+                    .background
+                )
+                  .lighten(0.2)
+                  .toString(),
+              },
+              attribute: {
+                background:
+                  colorsConfig.darkPalette.custom.queryBuilder.attribute
+                    .background,
+              },
+            },
           }
         : {
             primary: colorsConfig.lightPalette.primary,
             secondary: colorsConfig.lightPalette.secondary,
             dataPointColors: colorsConfig.lightPalette.custom.dataPointColors,
+            queryBuilder: {
+              text: colorsConfig.lightPalette.custom.queryBuilder.text,
+              entity: {
+                background:
+                  colorsConfig.lightPalette.custom.queryBuilder.entity
+                    .background,
+                lighterbg: Color(
+                  colorsConfig.lightPalette.custom.queryBuilder.entity
+                    .background
+                )
+                  .lighten(0.2)
+                  .toString(),
+              },
+              relation: {
+                background:
+                  colorsConfig.lightPalette.custom.queryBuilder.relation
+                    .background,
+                lighterbg: Color(
+                  colorsConfig.lightPalette.custom.queryBuilder.relation
+                    .background
+                )
+                  .lighten(0.2)
+                  .toString(),
+              },
+              attribute: {
+                background:
+                  colorsConfig.lightPalette.custom.queryBuilder.attribute
+                    .background,
+              },
+            },
           }),
     },
   };
diff --git a/libs/shared/models/yarn.lock b/libs/shared/models/yarn.lock
new file mode 100644
index 0000000000000000000000000000000000000000..fb57ccd13afbd082ad82051c2ffebef4840661ec
--- /dev/null
+++ b/libs/shared/models/yarn.lock
@@ -0,0 +1,4 @@
+# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY.
+# yarn lockfile v1
+
+
diff --git a/libs/shared/schema-utils/yarn.lock b/libs/shared/schema-utils/yarn.lock
new file mode 100644
index 0000000000000000000000000000000000000000..fb57ccd13afbd082ad82051c2ffebef4840661ec
--- /dev/null
+++ b/libs/shared/schema-utils/yarn.lock
@@ -0,0 +1,4 @@
+# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY.
+# yarn lockfile v1
+
+
diff --git a/nginx/nginx.conf b/nginx/nginx.conf
new file mode 100644
index 0000000000000000000000000000000000000000..ef2e21a890fc402b46472b5c18fd5b95c3a53a2e
--- /dev/null
+++ b/nginx/nginx.conf
@@ -0,0 +1,16 @@
+server {
+  listen 80;
+
+  location / {
+    root   /usr/share/nginx/html;
+    index  index.html index.htm;
+    try_files $uri $uri/ /index.html;
+  }
+
+  error_page 500 502 503 504 /50x.html;
+
+  location = /50x.html {
+      root /usr/share/nginx/html;
+  }
+
+}
diff --git a/package.json b/package.json
index 910a0f4083e87cea15b91f10f3475dcdcde43e50..9d07a7788766fc39c5af14ad5142bf853f2776fe 100644
--- a/package.json
+++ b/package.json
@@ -19,6 +19,8 @@
     "@types/cytoscape": "^3.19.4",
     "@types/react-grid-layout": "^1.3.0",
     "@types/styled-components": "^5.1.21",
+    "classnames": "^2.3.1",
+    "color": "^4.2.1",
     "core-js": "^3.6.5",
     "cytoscape": "^3.21.0",
     "graphology": "^0.24.0",
@@ -57,6 +59,7 @@
     "@svgr/webpack": "^5.4.0",
     "@testing-library/react": "12.1.2",
     "@testing-library/react-hooks": "7.0.2",
+    "@types/color": "^3.0.3",
     "@types/jest": "27.0.2",
     "@types/node": "16.11.7",
     "@types/react": "17.0.30",
diff --git a/tsconfig.base.json b/tsconfig.base.json
index 7a399ebf1769e00e2136b3160fbb57ad1ccd3f1a..a5bafb69c61a39d4c988430267936172aeb36df7 100644
--- a/tsconfig.base.json
+++ b/tsconfig.base.json
@@ -19,6 +19,9 @@
       "@graphpolaris/models": ["libs/shared/models/src/index.ts"],
       "@graphpolaris/schema-utils": ["libs/shared/schema-utils/src/index.ts"],
       "@graphpolaris/schema/usecases": ["libs/schema/usecases/src/index.ts"],
+      "@graphpolaris/querybuilder/usecases": [
+        "libs/querybuilder/usecases/src/index.ts"
+      ],
       "@graphpolaris/shared/data-access/api": [
         "libs/shared/data-access/api/src/index.ts"
       ],
diff --git a/workspace.json b/workspace.json
index 4b974196b33da4ad2e075a2a3bc4317db624ffa1..d829d3ca0515a309a5a268383dcf3764953f2251 100644
--- a/workspace.json
+++ b/workspace.json
@@ -1,6 +1,7 @@
 {
   "version": 2,
   "projects": {
+    "querybuilder-usecases": "libs/querybuilder/usecases",
     "schema-usecases": "libs/schema/usecases",
     "shared-data-access-api": "libs/shared/data-access/api",
     "shared-data-access-authorization": "libs/shared/data-access/authorization",
diff --git a/yarn.lock b/yarn.lock
index 3d6d637e556c329bdf8245c184f4c09526f253cf..ad3e6305a7719eb8dc8d0e2a9ad48f4ff0bc32e8 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -4072,7 +4072,7 @@
   dependencies:
     "@types/node" "*"
 
-"@types/color-convert@^2.0.0":
+"@types/color-convert@*", "@types/color-convert@^2.0.0":
   version "2.0.0"
   resolved "https://registry.yarnpkg.com/@types/color-convert/-/color-convert-2.0.0.tgz#8f5ee6b9e863dcbee5703f5a517ffb13d3ea4e22"
   integrity sha512-m7GG7IKKGuJUXvkZ1qqG3ChccdIM/qBBo913z+Xft0nKCX4hAU/IxKwZBU4cpRZ7GS5kV4vOblUkILtSShCPXQ==
@@ -4084,6 +4084,13 @@
   resolved "https://registry.yarnpkg.com/@types/color-name/-/color-name-1.1.1.tgz#1c1261bbeaa10a8055bbc5d8ab84b7b2afc846a0"
   integrity sha512-rr+OQyAjxze7GgWrSaJwydHStIhHq2lvY3BOC2Mj7KnzI7XK0Uw1TOOdI9lDoajEbSWLiYgoo4f1R51erQfhPQ==
 
+"@types/color@^3.0.3":
+  version "3.0.3"
+  resolved "https://registry.yarnpkg.com/@types/color/-/color-3.0.3.tgz#e6d8d72b7aaef4bb9fe80847c26c7c786191016d"
+  integrity sha512-X//qzJ3d3Zj82J9sC/C18ZY5f43utPbAJ6PhYt/M7uG6etcF6MRpKdN880KBy43B0BMzSfeT96MzrsNjFI3GbA==
+  dependencies:
+    "@types/color-convert" "*"
+
 "@types/connect-history-api-fallback@^1.3.5":
   version "1.3.5"
   resolved "https://registry.yarnpkg.com/@types/connect-history-api-fallback/-/connect-history-api-fallback-1.3.5.tgz#d1f7a8a09d0ed5a57aee5ae9c18ab9b803205dae"
@@ -6420,6 +6427,11 @@ classcat@^5.0.3:
   resolved "https://registry.yarnpkg.com/classcat/-/classcat-5.0.3.tgz#38eaa0ec6eb1b10faf101bbcef2afb319c23c17b"
   integrity sha512-6dK2ke4VEJZOFx2ZfdDAl5OhEL8lvkl6EHF92IfRePfHxQTqir5NlcNVUv+2idjDqCX2NDc8m8YSAI5NI975ZQ==
 
+classnames@^2.3.1:
+  version "2.3.1"
+  resolved "https://registry.yarnpkg.com/classnames/-/classnames-2.3.1.tgz#dfcfa3891e306ec1dad105d0e88f4417b8535e8e"
+  integrity sha512-OlQdbZ7gLfGarSqxesMesDa5uz7KFbID8Kpq/SxIoNGDqY8lSYs0D+hhtBXhcdB3rcbXArFr7vlHheLk1voeNA==
+
 clean-css@^4.2.3:
   version "4.2.4"
   resolved "https://registry.yarnpkg.com/clean-css/-/clean-css-4.2.4.tgz#733bf46eba4e607c6891ea57c24a989356831178"
@@ -6571,16 +6583,32 @@ color-name@1.1.3:
   resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.3.tgz#a7d0558bd89c42f795dd42328f740831ca53bc25"
   integrity sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=
 
-color-name@~1.1.4:
+color-name@^1.0.0, color-name@~1.1.4:
   version "1.1.4"
   resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2"
   integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==
 
+color-string@^1.9.0:
+  version "1.9.0"
+  resolved "https://registry.yarnpkg.com/color-string/-/color-string-1.9.0.tgz#63b6ebd1bec11999d1df3a79a7569451ac2be8aa"
+  integrity sha512-9Mrz2AQLefkH1UvASKj6v6hj/7eWgjnT/cVsR8CumieLoT+g900exWeNogqtweI8dxloXN9BDQTYro1oWu/5CQ==
+  dependencies:
+    color-name "^1.0.0"
+    simple-swizzle "^0.2.2"
+
 color-support@^1.1.2:
   version "1.1.3"
   resolved "https://registry.yarnpkg.com/color-support/-/color-support-1.1.3.tgz#93834379a1cc9a0c61f82f52f0d04322251bd5a2"
   integrity sha512-qiBjkpbMLO/HL68y+lh4q0/O1MZFj2RX6X/KmMa3+gJD3z+WwI1ZzDHysvqHGS3mP6mznPckpXmw1nI9cJjyRg==
 
+color@^4.2.1:
+  version "4.2.1"
+  resolved "https://registry.yarnpkg.com/color/-/color-4.2.1.tgz#498aee5fce7fc982606c8875cab080ac0547c884"
+  integrity sha512-MFJr0uY4RvTQUKvPq7dh9grVOTYSFeXja2mBXioCGjnjJoXrAp9jJ1NQTDR73c9nwBSAQiNKloKl5zq9WB9UPw==
+  dependencies:
+    color-convert "^2.0.1"
+    color-string "^1.9.0"
+
 colord@^2.9.1:
   version "2.9.2"
   resolved "https://registry.yarnpkg.com/colord/-/colord-2.9.2.tgz#25e2bacbbaa65991422c07ea209e2089428effb1"
@@ -10305,6 +10333,11 @@ is-arrayish@^0.2.1:
   resolved "https://registry.yarnpkg.com/is-arrayish/-/is-arrayish-0.2.1.tgz#77c99840527aa8ecb1a8ba697b80645a7a926a9d"
   integrity sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0=
 
+is-arrayish@^0.3.1:
+  version "0.3.2"
+  resolved "https://registry.yarnpkg.com/is-arrayish/-/is-arrayish-0.3.2.tgz#4574a2ae56f7ab206896fb431eaeed066fdf8f03"
+  integrity sha512-eVRqCvVlZbuw3GrM63ovNSNAeA1K16kaR/LRY/92w0zxQ5/1YzwblUX652i4Xs9RwAGjW9d9y6X88t8OaAJfWQ==
+
 is-bigint@^1.0.1:
   version "1.0.4"
   resolved "https://registry.yarnpkg.com/is-bigint/-/is-bigint-1.0.4.tgz#08147a1875bc2b32005d41ccd8291dffc6691df3"
@@ -15059,6 +15092,13 @@ signal-exit@^3.0.0, signal-exit@^3.0.2, signal-exit@^3.0.3:
   resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-3.0.6.tgz#24e630c4b0f03fea446a2bd299e62b4a6ca8d0af"
   integrity sha512-sDl4qMFpijcGw22U5w63KmD3cZJfBuFlVNbVMKje2keoKML7X2UzWbc4XrmEbDwg0NXJc3yv4/ox7b+JWb57kQ==
 
+simple-swizzle@^0.2.2:
+  version "0.2.2"
+  resolved "https://registry.yarnpkg.com/simple-swizzle/-/simple-swizzle-0.2.2.tgz#a4da6b635ffcccca33f70d17cb92592de95e557a"
+  integrity sha1-pNprY1/8zMoz9w0Xy5JZLeleVXo=
+  dependencies:
+    is-arrayish "^0.3.1"
+
 sisteransi@^1.0.5:
   version "1.0.5"
   resolved "https://registry.yarnpkg.com/sisteransi/-/sisteransi-1.0.5.tgz#134d681297756437cc05ca01370d3a7a571075ed"