uster</h1><p class="page-description"></p><table class="properties"><tbody><tr class="property-row property-row-created_by"><th><span class="icon property-icon"><div data-testid="/icons/user-circle_gray.svg" style="width:14px;height:14px;flex-shrink:0;transform:scale(1.2);mask:url(/icons/user-circle_gray.svg?mode=light) no-repeat center;-webkit-mask:url(/icons/user-circle_gray.svg?mode=light) no-repeat center;background-color:rgba(71, 70, 68, 0.6);fill:rgba(71, 70, 68, 0.6)"></div></span>Created by</th><td><span class="user"><img src="How%20to%20Ping%20Services%20in%20K8s%20Cluster%202676cd51990d80a58774e1e1d58dbb0c/Screenshot_2025-08-02_at_10.38.11_AM.png" class="icon user-icon"/>JiaLin Huang</span></td></tr><tr class="property-row property-row-last_edited_time"><th><span class="icon property-icon"><div data-testid="/icons/clock_gray.svg" style="width:14px;height:14px;flex-shrink:0;transform:scale(1.2);mask:url(/icons/clock_gray.svg?mode=light) no-repeat center;-webkit-mask:url(/icons/clock_gray.svg?mode=light) no-repeat center;background-color:rgba(71, 70, 68, 0.6);fill:rgba(71, 70, 68, 0.6)"></div></span>Last edited</th><td><time>@2025年9月7日 20:21</time></td></tr><tr class="property-row property-row-multi_select"><th><span class="icon property-icon"><div data-testid="/icons/list_gray.svg" style="width:14px;height:14px;flex-shrink:0;transform:scale(1.2);mask:url(/icons/list_gray.svg?mode=light) no-repeat center;-webkit-mask:url(/icons/list_gray.svg?mode=light) no-repeat center;background-color:rgba(71, 70, 68, 0.6);fill:rgba(71, 70, 68, 0.6)"></div></span>Tags</th><td><span class="selected-value select-value-color-default">Kubernetes</span><span class="selected-value select-value-color-green">Post</span></td></tr></tbody></table></header><div class="page-body"><h1 class="">TL;DR</h1><p class="">5 ways to ping services in K8s (kind)</p><ol type="1" class="numbered-list" start="1"><li>Docker exec (Kind only): localhost:NodePort</li></ol><ol type="1" class="numbered-list" start="2"><li>Service DNS: service-name:port (works 99% of the time)</li></ol><ol type="1" class="numbered-list" start="3"><li>ClusterIP: virtual-ip:port</li></ol><ol type="1" class="numbered-list" start="4"><li>NodeIP + NodePort: node-ip:random-port</li></ol><ol type="1" class="numbered-list" start="5"><li>Direct Pod IP: pod-ip:container-port</li></ol><p class="">
</p><p class="">NodePort services get all 5 methods. ClusterIP services only get methods 2, 3, and 5.</p><p class="">
</p><h1 class="">Testing Context</h1><h3 class="">Kind, hosting cluster in docker containers</h3><p class="">I&#x27;m using Kind for this demo. Kind runs K8s nodes as Docker containers, isolating from your host machine. Alternatives: k3s/k0s (install directly on host), Minikube (similar Docker isolation).</p><h3 class="">Two simple Nodejs servers</h3><p class="">My setup: two Node.js APIs on Alpine images in noah namespace</p><ul class="bulleted-list"><li style="list-style-type:disc">joke-service: ClusterIP</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">number-service: NodePort</li></ul><p class="">Alpine services only have <mark class="highlight-red">wget</mark>. Kind&#x27;s Debian control-plane has <mark class="highlight-red">curl</mark>. That&#x27;s why examples vary. (below you will see)</p><script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js" integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg==" crossorigin="anonymous" referrerPolicy="no-referrer"></script><link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css" integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ==" crossorigin="anonymous" referrerPolicy="no-referrer"/><pre class="code code-wrap"><code class="language-Docker" style="white-space:pre-wrap;word-break:break-all">FROM node:18-alpine
WORKDIR /app
COPY package.json .
RUN npm install --production
COPY server.js .
EXPOSE 3000
CMD [&quot;npm&quot;, &quot;start&quot;]</code></pre><h3 class="">YAMLs</h3><p class="">This follows the standard Kubernetes pattern:</p><blockquote class="">Deployment + Service + Pods</blockquote><p class="">The YAML configs are basically identical except for that final <mark class="highlight-red">type </mark>field. That one line changes everything about how you can access them.</p><script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js" integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg==" crossorigin="anonymous" referrerPolicy="no-referrer"></script><link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css" integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ==" crossorigin="anonymous" referrerPolicy="no-referrer"/><pre class="code code-wrap"><code class="language-Bash" style="white-space:pre-wrap;word-break:break-all">  # Same for both services, except the type
  apiVersion: v1
  kind: Service
  metadata:
    name: joke-service  # or number-service
    namespace: noah
  spec:
    selector:
      app: joke-service  # or number-service
    ports:
    - port: 80
      targetPort: 3000
      protocol: TCP
    type: ClusterIP  # or NodePort for number-service</code></pre><script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js" integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg==" crossorigin="anonymous" referrerPolicy="no-referrer"></script><link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css" integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ==" crossorigin="anonymous" referrerPolicy="no-referrer"/><pre class="code code-wrap"><code class="language-Bash" style="white-space:pre-wrap;word-break:break-all">kind: Cluster
apiVersion: kind.x-k8s.io/v1alpha4
name: kubecmds-test
nodes:
- role: control-plane
  kubeadmConfigPatches:
  - |
    kind: InitConfiguration
    nodeRegistration:
      kubeletExtraArgs:
        node-labels: &quot;ingress-ready=true&quot;
  extraPortMappings:
  - containerPort: 80
    hostPort: 8080
    protocol: TCP
  - containerPort: 443
    hostPort: 8443
    protocol: TCP
- role: worker
- role: worker</code></pre><h3 class="">Quick Reference</h3><script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js" integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg==" crossorigin="anonymous" referrerPolicy="no-referrer"></script><link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css" integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ==" crossorigin="anonymous" referrerPolicy="no-referrer"/><pre class="code code-wrap"><code class="language-Bash" style="white-space:pre-wrap;word-break:break-all">~ kubectl get svc -n noah
NAME             TYPE        CLUSTER-IP      EXTERNAL-IP   PORT(S)        AGE
joke-service     ClusterIP   10.96.236.182   &lt;none&gt;        80/TCP         9h
number-service   NodePort    10.96.183.131   &lt;none&gt;        80:30405/TCP   9h

~ kubectl get deployment -n noah
NAME             READY   UP-TO-DATE   AVAILABLE   AGE
joke-service     2/2     2            2           9h
number-service   2/2     2            2           9h


~ kubectl get pods -n noah
NAME                             READY   STATUS    RESTARTS   AGE
joke-service-5f4f7b5f44-rt2qd    1/1     Running   0          4h31m
joke-service-5f4f7b5f44-rvdct    1/1     Running   0          4h31m
number-service-b944ff5bc-kj4xj   1/1     Running   0          4h3m
number-service-b944ff5bc-vk47l   1/1     Running   0          4h3m


~ kubectl get nodes -o wide
NAME                          STATUS   ROLES           AGE   VERSION   INTERNAL-IP   EXTERNAL-IP   OS-IMAGE                         KERNEL-VERSION     CONTAINER-RUNTIME
kubecmds-test-control-plane   Ready    control-plane   9h    v1.34.0   172.18.0.4    &lt;none&gt;        Debian GNU/Linux 12 (bookworm)   5.15.49-linuxkit   containerd://2.1.3
kubecmds-test-worker          Ready    &lt;none&gt;          9h    v1.34.0   172.18.0.3    &lt;none&gt;        Debian GNU/Linux 12 (bookworm)   5.15.49-linuxkit   containerd://2.1.3
kubecmds-test-worker2         Ready    &lt;none&gt;          9h    v1.34.0   172.18.0.2    &lt;none&gt;        Debian GNU/Linux 12 (bookworm)   5.15.49-linuxkit   containerd://2.1.3</code></pre><p class="">
</p><h1 class="">How to Hit type=NodePort Services</h1><h3 class="">Docker Exec (Kind Special Trick)</h3><p class=""><code>docker exec -it kubecmds-test-control-plane curl http://localhost:30405/health</code></p><p class="">Jump into the Kind container first. Then hit localhost just like you&#x27;re using k3s on bare metal.</p><p class="">Pretty neat actually.</p><p class=""><mark class="highlight-red">30405</mark> is found by <mark class="highlight-red">kubectl get svc</mark></p><h3 class="">Service DNS (The Easy Button)</h3><p class=""><code>kubectl exec -it number-service-xxx -n noah -- wget -qO- http://number-service:80/health</code></p><p class="">Just use the service name. K8s handles DNS magic for you.</p><p class="">Caveats:</p><ul class="bulleted-list"><li style="list-style-type:disc">Cross-namespace: Need full FQDN like <mark class="highlight-red">service-name.namespace.svc.cluster.local</mark></li></ul><ul class="bulleted-list"><li style="list-style-type:disc">CoreDNS down: If DNS pods crash, you&#x27;re stuck with ClusterIP direct access</li></ul><p class="">
</p><h3 class="">ClusterIP Direct</h3><p class=""><code>kubectl exec -it number-service-b944ff5bc-kj4xj -n noah curl http://10.96.183.131:80/health</code></p><p class="">Works fine but you gotta look up the virtual IP first.</p><p class="">Run <mark class="highlight-red">kubectl get svc</mark> to find it.</p><p class="">
</p><h3 class="">NodeIP + NodePort (The Painful Way)</h3><p class=""><code>kubectl exec -it number-service-xxx -n noah -- wget -qO- http://172.18.0.3:30405/health</code></p><p class="">Most annoying method.</p><p class="">First run <mark class="highlight-red">kubectl get nodes -o wide</mark> for the node IP. </p><p class="">Then <mark class="highlight-red">kubectl get svc</mark> for the random port number.</p><p class="">Then piece it all together. Why would you do this to yourself</p><p class="">
</p><h3 class="">Direct Pod IP Access</h3><p class=""><code>kubectl exec -it number-service-xxx -n noah -- wget -qO- http://10.244.1.18:3000/health</code></p><p class="">Hit the pod directly on its container port. Run <mark class="highlight-red">kubectl get pods -o wide</mark> to find the pod IP.<br/>Or check <mark class="highlight-red">kubectl describe svc number-service -n noah</mark> and look at Endpoints.</p><script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js" integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg==" crossorigin="anonymous" referrerPolicy="no-referrer"></script><link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css" integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ==" crossorigin="anonymous" referrerPolicy="no-referrer"/><pre class="code code-wrap"><code class="language-YAML" style="white-space:pre-wrap;word-break:break-all">Endpoints:                10.244.1.18:3000,10.244.2.23:3000</code></pre><p class="">Bypasses the service layer completely. Useful for debugging individual pod issues.</p><p class="">
</p><p class="">
</p><h1 class="">type=ClusterIP Services Are Limited</h1><p class="">What doesn&#x27;t work:</p><ul class="bulleted-list"><li style="list-style-type:disc">Docker exec from outside? Nope.</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">NodePort method? Nope.</li></ul><p class="">
</p><h1 class="">Takeaways</h1><p class="">Service DNS (service-name:port) works 99% of the time. Simple, reliable, consistent across all K8s setups. IPs change when pods restart, but service names don&#x27;t.</p><p class="">Other methods are mainly for debugging networking or testing individual pods without load<br/>balancing.</p><p class="">
</p><p class="">
</p></div></article><span class="sans" style="font-size:14px;padding-top:2em"></span></body>