จับ AI มาไตรกรีฬา สร้างตัวติดตั้ง Snap สำหรับ Surveyor Pocket Tools

ข่าวคราวเธอเงียบหายไปหลายปีนึกว่าไปได้ดี คืออารมณ์สำหรับดีพซีค (Deepseek) ประมาณนี้จริงๆ คือตอนเปิดตัวฮือฮามาก ทำให้ทราฟฟิคเข้าใช้แน่น ต้องรอนานกว่าจะตอบได้แต่ละคำถาม กาลเวลาผ่านไปไม่กี่เดือนเหมือนคนจะลืมกันไปแล้ว แต่ในความเงียบยังมีความเคลื่อนไหวยังอยู่ ผู้พัฒนาได้มีการอัพเดทข้อมูลเดือนพค. 2025 และในเดือนสิงหาคม 2025 ได้มีการอัพเกรดใหญ่เป็น V3.1

ต้องการความต่อเนื่องหรือความเป็นส่วนตัว

ปกติผมใช้เจมิไน (Gemini) สลับกับดีพซีค แต่ใช้เจมิไนมากกว่าเพราะสามารถอินพุทข้อความได้มากกว่ามากๆ โดยเฉพาะการอัพโหลดไฟล์เช่นไฟล์เอกสาร โดยไม่มีการต้องเริ่มสนทนา (chat) กันใหม่บ่อย ให้เสียอารมณ์ด้วยความไม่ต่อเนื่อง ดีพซีคนอกจากจะรับโทเค็นได้น้อยต่อหนึ่งการสนทนา ยังมีปัญหาลืมเลือนการสนทนาก่อนหน้านี้ บางครั้งการมาเริ่มต้นใหม่ต้องอธิบายกันนานกว่าจะเริ่มเข้าใจ พอเริ่มจะเข้าใจเวลาก็หมดอีกแล้ว นี่คือปัญหาความไม่ต่อเนื่องของดีพซีคที่ร้ายแรงที่สุด แต่ในทางตรงกันข้ามก็รักษาความเป็นส่วนตัวมากที่สุด (ถ้าเซอร์เวอร์ของดีพซีคไม่มีการเก็บข้อมูลเข้าส่วนกลาง)

Snap ตัวติดตั้งโปรแกรมสำหรับลินุกซ์

Snap Ubuntu หรือที่เรียกว่า Snapcraft คือระบบการจัดการแพ็กเกจ (package management system) ที่พัฒนาโดยบริษัท Canonical (ผู้พัฒนา Ubuntu) เพื่อช่วยให้นักพัฒนาสามารถสร้างและเผยแพร่แอปพลิเคชันที่ทำงานได้บนระบบ Linux หลากหลายรูปแบบได้อย่างง่ายดาย. โดย Snap จะรวบรวมแอปพลิเคชันและ Dependencies (สิ่งที่โปรแกรมต้องการในการทำงาน) ทั้งหมดไว้ในแพ็กเกจเดียว ทำให้การติดตั้งและอัปเดตแอปทำได้สะดวก และแอปพลิเคชันมีความปลอดภัยมากขึ้น. 

ผมมีความจำเป็นต้องใช้ Surveyor Pocket Tools บนลีนุกซ์ ผมสลับใช้ Ubuntu กับ Manjaro เป็นบางครั้งแต่ส่วนใหญ่จะอยู่กับ Manjaro นานกว่า งานอดิเรกผมต้องการเขียนภาษารัสท์เพื่อศึกษาภาษานี้ มีความจำเป็นต้องใช้ Surveyor Pocket Tools เพื่อต้องการเปรียบเทียบผลลัพธ์การคำนวณ ที่ใช้อยู่คือสร้าง env ของ python ผ่าน venv แล้วค่อยเรียกไพทอนมารัน main.py ทำให้เสียเวลาแต่งตัวพอสมควร ถ้ามีไฟล์ Snap แล้วสามารติดตั้งได้ง่าย พอจะเทียบเคียงได้ก็คือไฟล์ MSI ในวินโดส์

ให้ AI ช่วยสร้างตัวติดตั้ง Snap

ปัญหาคือการสร้างตัวติดตั้งด้วย Snap ไม่ง่าย ถ้าจะเรียนเองต้องใช้ความพยายามสูงมาก ผมเลยมอง AI ว่าจะมีอะไรมาช่วยผมบ้าง

คำถามแรกที่ถาม AI คือ Snap ง่ายไหม คำตอบที่ทั้งเจมิไนกับดีพซีคให้มาคือ ยากกกกกกก ตรงกัน ต่อไปคำถามที่สองคือช่วยฉันสร้าง Snap แบบทีละขั้นตอน

สร้างโครงการสำหรับ Snap

การสร้าง Snap จะมีการใช้คำสั่ง command line ก่อนเล็กน้อย เพื่อสร้างไดเรคทอรีโดยมีโครงสร้างตามที่ Snap ต้องการ จากนั้นจะมีการสร้างไฟล์ชื่อ “snapcraft.yaml” เพื่อตั้งค่าว่าแอพเราชื่ออะไร ใช้ไลบรารีอะไร ไฟล์ข้อมูลที่ต้องการขนไฟล์ไปอยู่ที่ไหนมีอะไรบ้าง

sudo snap install snapcraft --classic
mkdir my-new-app
cd my-new-app
snapcraft init

แนวทางรักษาความเป็นส่วนตัวจาก AI

ในการรักษาความเป็นส่วนตัว ไม่ให้ตัว AI บันทึกข้อมูลทุกๆอย่างของเรา มีคนแนะนำดังนี้ “สร้างบัญชี burner account ด้วย Outlook.com ได้ CoPilot ฟรี แล้วใช้บัญชีนั้นสมัครใช้ AI อื่นๆ ฟรีได้อีก หมุนเวียนใช้ซัก 5 ตัวขึ้นไปฟรีๆ แล้วเปลี่ยนหัวข้อการสนทนาไปเรื่อยๆ จะได้ไม่มี AI ตัวไหนได้ข้อมูลการสนทนาครบถ้วน” ผมว่าน่าสนใจมาก ก็จะลองดู

ตอนนี้ chatGPT เขาได้ข้อมูลผมไปมาก เพราะเป็น AI ตัวแรกที่ใช้ และใข้มาอย่างนาว ผมเผลอไปบอกเขาหลายอย่าง เขารู้ทั้งอาชีพผมว่าทำอะไร สนใจอะไร มีงานอดิเรกอะไรบ้าง มีโครงการที่ทำอยู่ปัจจุบัน มีแอพที่พัฒนาชื่ออะไร รู้ไปหมด ผมตกใจตอนที่ให้เขาล้อเลียนด้วยการสั่งว่า “Roast me” มันจาระไนมาหมด จากที่ได้คุยกับพรรคพวกสรุปตรงกันว่าข้อมูลถูก openAI เอาไปแล้ว พูดแบบชาวบ้านๆคือ “เสร็จโจร

ตอนแรกคิดว่าจะให้สร้างไฟล์ Snap ให้แบบเวียนใช้ AI ตามคำแนะนำ คือเวียนถามไปเรื่อยๆ ก็เลยมีประเด็นคือความสามารถของ AI นั้นต่างกันพอสมควร ผมจินตนาการว่าเป็นการเอา AI มาทดสอบแบบไตรกีฬาคือหมุนเวียนใช้ AI

ไม้แรกว่ายน้ำ 750 เมตร ด้วยเจมิไน

สำหรับเจมิไน ปกติเขาจะเก่งทุกเรื่อง ออกอาการผยอง หยิ่งยโสบ้างเล็กน้อย มาเที่ยวนี้จอดป้ายแค่สองป้ายคือเขาพยายามทำไฟล์ “snapcraft.yaml” ให้ผม ที่เอาแค่เบสิคธรรมดามาก จากนั้นเรียก Snap มาคอมไพล์ตรงนี้ ผลลัพธ์คือเฟลล์ พยายามแก้สอง สามครั้ง แก้ครั้งที่สามเฟลล์เหมือนเดิม แต่คำตอบดันไปเหมือนครั้งที่ 1 จนผมอดแซะเขาไม่ได้ว่าเขากำลังใช้วิธีการ “Flip Flop” สรุปแล้วเจมิไนไม่สามารถว่ายน้ำครบระยะทางได้ ไฟล์ “snapcraft.yaml” เมื่อเอาไปรันด้วยคำสั่ง snapcraft pack จะเฟลล์

name: surveyor-pocket-tools
base: core22
version: '1.0'
build-on: [amd64] # Add this line
summary: A set of surveyor pocket tools
description: |
  This snap provides various tools for land surveyors.

grade: devel
confinement: devmode

parts:
  surveyor-pocket-tools:
    plugin: python
    source: .
    requirements: [requirements.txt]
    build-packages:
      - python3-dev
      - build-essential

apps:
  surveyor-pocket-tools:
    command: bin/main.py
    plugs:
      - home
      - network
      - removable-media

ไม้สองปั่นจักรยาน 20 กม. ด้วยดีพซีค

ผมเห็นท่าไม่ไหว เลยไปเปลี่ยนไปลองดีพซีค อ่าาาา ผิดความคาดหมาย เพราะเจมิไนมีปัญหา ตรรกะในใจผมคิดว่าดีพซีคคงไม่รอดเช่นกัน แต่ผิดคาด ดีพซีคสร้างไฟล์ “snapcraft.yaml” ให้คอมไพล์ผ่านในครั้งแรก จนผมอดอุทานไม่ได้ว่ามันใช่เหรอ จากที่ผมบอกความต้องการคือต้องการใช้ PySide6 (Qt ในภาคไพทอน) เป็นตัว UI/UX จากการสนทนามีการปรับแต่งไฟล์ “snapcraft.yaml” หลายๆเที่ยวบางครั้งคอมไพล์เฟลล์แล้วกลับมาแก้ไขให้คอมไพล์ได้ ในที่สุดสิ้นสุดการสนทนาไปสอง Session ทำให้ได้ไฟล์ Snap ที่พร้อมนำไปติดตั้ง

ต้องบอกว่าการขน PySide6 ไปใช้นั้น ไฟล์ “snapcraft.yaml” มีความยาวและซับซ้อนพอสมควร ซึ่งดีพซีคจัดการให้อย่างดี ถ้าให้เขียนเองคงต้องศึกษากันหลายเดือน สรุปแล้วไม้สองดีพซีคผ่านได้

ดีพซีคมาตันที่ปัญหาของ “QWebEngine” ที่ผมใช้คือเรียกตัวเว็บมาแสดงผล Google Maps เพื่อปักหมุดตำแหน่งที่ผู้ใช้ Surveyor Pocket Tools ได้จากการป้อนหรือคำนวณ ในวินโดส์ไม่มีปัญหาอะไรกับตัวนี้ ดีพซีคไม่สามารถแก้ปัญหานี้ได้หลังจากทดลองกันหลายครั้ง บางครั้งสคริปต์ที่เขียนมาก็ไปซ้ำกับของเดิมที่มันไม่ทำงาน

โปรแกรมรัน Surveyor Pocket Tools ได้ แต่ถ้ามีการเรียกไดอะล็อกเพื่อแสดงผล Google Maps จะแครชทันที

name: surveyor-pocket-tools
version: '1.0'
summary: GIS Surveying Tools
description: Professional surveying toolkit with PySide6 GUI
base: core22
confinement: strict
grade: stable

apps:
  surveyor-pocket-tools:
    command: bin/run-app
    plugs: [network, home, opengl, x11, desktop]
    environment:
      QT_QPA_PLATFORM: xcb
      QT_XCB_GL_INTEGRATION: xcb_egl
      PYTHONPATH: $SNAP/lib/python3.10/site-packages
      SURVEYOR_DATA_DIR: $SNAP_USER_DATA/data

parts:
  surveyor-app:
    plugin: nil
    source: .

    organize:
      'data/xml': share/surveyor_pocket_tools/data/xml
      'data/pyproj': share/surveyor_pocket_tools/data/proj_data
      'data/database': share/surveyor_pocket_tools/data/database
      'data/example': share/surveyor_pocket_tools/data/example
      'data/markers': share/surveyor_pocket_tools/data/markers

    stage-packages:
      - python3.10
      - python3.10-distutils
      - libgl1
      - libglx0
      - libglvnd0
      - libegl1
      - libegl-mesa0
      - libgles2
      - libopengl0
      - libxkbcommon0
      - libxcb1
      - libx11-6
      - libxext6
      - libxfixes3
      - libxxf86vm1
      - libwayland-client0
      - libwayland-server0
      - libxcb-cursor0
      - libxcb-icccm4
      - libxcb-image0
      - libxcb-keysyms1
      - libxcb-randr0
      - libxcb-render0
      - libxcb-render-util0
      - libxcb-shape0
      - libxcb-sync1
      - libxcb-xfixes0
      - libxcb-xkb1
      - libxkbfile1
      - libxkbcommon-x11-0
      - libfontconfig1
      - fontconfig-config
      - libfreetype6
      - fonts-dejavu-core
      - libexpat1
      - proj-data
      - gdal-data
      - libdbus-1-3
      - libffi8
      - libatomic1
      - libnss3
      - libxcomposite1
      - libxcursor1
      - libxdamage1
      - libxrandr2
      - libxtst6
      - libxi6
      - libxss1
      - libasound2
      # GSettings dependencies
      - dconf-gsettings-backend
      - libdconf1
      - gsettings-desktop-schemas
      # GLib for gsettings command
      - libglib2.0-bin
      - libglib2.0-0
      - libgio-2.0-0
      - libgobject-2.0-0
      # QWebEngine dependencies
      - libnss3-tools
      - libxrender1
      - libxshmfence1
      - libgbm1
      - libvulkan1
      - libminizip1
      - libevent-2.1-7
      - libsnappy1v5
      - libre2-9
      - libicu70

    override-build: |
      # 1. Setup Python
      mkdir -p $SNAPCRAFT_PART_INSTALL/usr/bin
      ln -sf /usr/bin/python3.10 $SNAPCRAFT_PART_INSTALL/usr/bin/python3

      # 2. Create directory structure
      mkdir -p $SNAPCRAFT_PART_INSTALL/share/surveyor_pocket_tools/data
      mkdir -p $SNAPCRAFT_PART_INSTALL/data

      # 3. Install application files
      cp src/*.py $SNAPCRAFT_PART_INSTALL/

      # 4. Install data to share directory
      cp -r data/* $SNAPCRAFT_PART_INSTALL/share/surveyor_pocket_tools/data/

      # 5. Copy initial data for launcher script
      cp -r data/* $SNAPCRAFT_PART_INSTALL/data/ 2>/dev/null || true

      # 6. Install Python packages
      python3.10 -m pip install \
        PySide6 \
        pyproj \
        openpyxl \
        lxml \
        simplekml \
        pyshp \
        geographiclib \
        darkdetect \
        gmplot \
        --target=$SNAPCRAFT_PART_INSTALL/lib/python3.10/site-packages \
        --no-warn-script-location

      # 7. Copy essential xcb libraries
      echo "=== COPYING XCB LIBRARIES ==="
      mkdir -p $SNAPCRAFT_PART_INSTALL/usr/lib/x86_64-linux-gnu/

      essential_xcb_libs=(
        "libxcb.so*"
        "libxcb-cursor.so*"
        "libxcb-icccm.so*"
        "libxcb-image.so*"
        "libxcb-keysyms.so*"
        "libxcb-randr.so*"
        "libxcb-render.so*"
        "libxcb-render-util.so*"
        "libxcb-shape.so*"
        "libxcb-sync.so*"
        "libxcb-xfixes.so*"
        "libxcb-xkb.so*"
      )

      for lib_pattern in "${essential_xcb_libs[@]}"; do
        cp -v /usr/lib/x86_64-linux-gnu/$lib_pattern $SNAPCRAFT_PART_INSTALL/usr/lib/x86_64-linux-gnu/ 2>/dev/null || true
      done

      # 8. Setup PROJ data
      echo "=== SETTING UP PROJ DATA ==="
      mkdir -p $SNAPCRAFT_PART_INSTALL/share/proj
      cp -r /usr/share/proj/* $SNAPCRAFT_PART_INSTALL/share/proj/ 2>/dev/null || true
      cp -r $SNAPCRAFT_PART_INSTALL/share/surveyor_pocket_tools/data/proj_data/* $SNAPCRAFT_PART_INSTALL/share/proj/ 2>/dev/null || true

      # 9. SETUP GSETTINGS SCHEMAS - USE PRE-COMPILED FROM HOST
      echo "=== COPYING GSETTINGS SCHEMAS ==="
      mkdir -p $SNAPCRAFT_PART_INSTALL/share/glib-2.0/schemas
      cp -r /usr/share/glib-2.0/schemas/* $SNAPCRAFT_PART_INSTALL/share/glib-2.0/schemas/ 2>/dev/null || true

      # 10. Copy gsettings binary
      echo "=== COPYING GSETTINGS BINARY ==="
      mkdir -p $SNAPCRAFT_PART_INSTALL/usr/bin
      cp -v /usr/bin/gsettings $SNAPCRAFT_PART_INSTALL/usr/bin/ 2>/dev/null || true

      # 11. Copy required GLib libraries for gsettings
      echo "=== COPYING GLIB LIBRARIES ==="
      mkdir -p $SNAPCRAFT_PART_INSTALL/usr/lib/x86_64-linux-gnu/
      cp -v /usr/lib/x86_64-linux-gnu/libglib-2.0.so* $SNAPCRAFT_PART_INSTALL/usr/lib/x86_64-linux-gnu/ 2>/dev/null || true
      cp -v /usr/lib/x86_64-linux-gnu/libgio-2.0.so* $SNAPCRAFT_PART_INSTALL/usr/lib/x86_64-linux-gnu/ 2>/dev/null || true
      cp -v /usr/lib/x86_64-linux-gnu/libgobject-2.0.so* $SNAPCRAFT_PART_INSTALL/usr/lib/x86_64-linux-gnu/ 2>/dev/null || true

    override-prime: |
      craftctl default

      # Create launcher
      mkdir -p $SNAPCRAFT_PRIME/bin
      cat <<'EOF' > $SNAPCRAFT_PRIME/bin/run-app
      #!/bin/bash
      # Set comprehensive library paths
      export LD_LIBRARY_PATH="\
      $SNAP/usr/lib/x86_64-linux-gnu:\
      $SNAP/usr/lib:\
      $SNAP/lib:\
      $LD_LIBRARY_PATH"

      # Set PATH to include snap binaries
      export PATH="$SNAP/usr/bin:$PATH"

      # Set PROJ_LIB to correct location
      export PROJ_LIB="$SNAP/share/proj"

      # Set GSETTINGS paths
      export XDG_DATA_DIRS="$SNAP/share:$XDG_DATA_DIRS:/usr/share"
      export GSETTINGS_SCHEMA_DIR="$SNAP/share/glib-2.0/schemas"

      # Qt configuration
      export QT_QPA_PLATFORM=xcb
      export QT_XCB_GL_INTEGRATION=xcb_egl

      # Clean environment
      unset QT_WAYLAND_DISABLE_WINDOWDECORATION
      unset GDK_BACKEND
      unset SDL_VIDEODRIVER
      unset WAYLAND_DISPLAY

      # Create symbolic links from read-only snap data to writable user data
      mkdir -p "$SNAP_USER_DATA/data"

      # For each data subdirectory, create symlink if it doesn't exist
      for subdir in xml database proj_data example markers; do
        user_data_dir="$SNAP_USER_DATA/data/$subdir"
        snap_data_dir="$SNAP/share/surveyor_pocket_tools/data/$subdir"

        # Create user data directory
        mkdir -p "$user_data_dir"

        # Copy initial data if user directory is empty but snap has data
        if [ -z "$(ls -A "$user_data_dir")" ] && [ -d "$snap_data_dir" ]; then
          echo "Copying initial $subdir data to writable location..."
          cp -r "$snap_data_dir"/* "$user_data_dir/" 2>/dev/null || true
        fi
      done

      # Set environment variable to point to writable data
      export SURVEYOR_ACTUAL_DATA_DIR="$SNAP_USER_DATA/data"

      # Execute application
      exec "$SNAP/usr/bin/python3" "$SNAP/main.py" "$@"
      EOF
      chmod +x $SNAPCRAFT_PRIME/bin/run-app

ไม้สามวิ่ง 5 กม. ด้วย chatGPT+

สุดท้ายคนที่จะมาปิดจ๊อบคือ chatGPT + ผมก็ไม่คาดฝันว่าจะได้ใช้ chatGPT เนื่องจากตอนใช้รุ่นเบสิคถ้าคุยกับธรรมดาจะได้นาน แต่ถ้ามีการอัพโหลดไฟล์ไปให้ chatGPT วิเคราะห์เวลาจะหมดเร็วมาก จนแอบคิดไม่ได้ว่า openAI เจ้าของนี่มันหน้าเลือดจริงๆ ต่อมาบังเอิญว่าแอพสำหรับซื้อของออนไลน์แจกในฐานะที่เป็น VIP 🙂 ได้มาทดลองใช้สามเดือน

แรกๆที่ลองใช้คือมันต่างจากรุ่นเบสิคมากๆคือ ความสามารถต่างกันเห็นได้ชัดเจน รุ่นพลัสใช้ได้นานไม่มีกำหนดเวลา สามารถส่งไฟล์ยาวๆไปให้ได้ ผมส่งสคริปต์ที่ดีพซีคสร้างให้ไปให้ chatGPT + ไม่นานนักเขาสามารถแก้ไขสคริปต์ดังกล่าวจนสามารถรัน QWebEngine เจ้าปัญหาได้ผ่าน แบบผมไม่อยากจะเชื่อในสายตา

name: surveyor-pocket-tools
version: "1.0"
summary: GIS Surveying Tools
description: Professional surveying toolkit with PySide6 GUI
base: core22
grade: stable
confinement: strict

apps:
  surveyor-pocket-tools:
    command: bin/run-app
    plugs:
      - desktop
      - desktop-legacy
      - network
      - home
      - opengl
      - x11
      - wayland
      - gsettings
      - audio-playback
      - browser-support
    desktop: usr/share/applications/surveyor-pocket-tools.desktop
    environment:
      # Locale
      LANG: "C.UTF-8"
      LC_ALL: "C.UTF-8"

      # Qt defaults (launcher can switch to wayland via SURVEYOR_QT_BACKEND=wayland)
      QT_QPA_PLATFORM: "xcb"
      QT_XCB_GL_INTEGRATION: "xcb_egl"

      # Point Qt to vendored PySide6 in the snap (Python 3.10 site-packages)
      QT_PLUGIN_PATH: "$SNAP/lib/python3.10/site-packages/PySide6/Qt/plugins"
      QML2_IMPORT_PATH: "$SNAP/lib/python3.10/site-packages/PySide6/Qt/qml"
      QTWEBENGINEPROCESS_PATH: "$SNAP/lib/python3.10/site-packages/PySide6/Qt/libexec/QtWebEngineProcess"

      # GSettings / XDG
      GSETTINGS_SCHEMA_DIR: "$SNAP/usr/share/glib-2.0/schemas"
      XDG_DATA_DIRS: "$SNAP/usr/share:$XDG_DATA_DIRS"

      # Python search path for vendored packages
      PYTHONPATH: "$SNAP/lib/python3.10/site-packages"
      PYTHONDONTWRITEBYTECODE: "1"

      # PROJ packaged first; user overrides second; offline by default
      PROJ_DATA: "$SNAP/share/surveyor_pocket_tools/data/proj_data/share/proj:$SNAP_USER_DATA/data/proj_data/share/proj"
      PROJ_LIB: "$PROJ_DATA"
      PROJ_NETWORK: "OFF"

      # App data dir your code uses
      SURVEYOR_DATA_DIR: "$SNAP_USER_DATA/data"

      # QtWebEngine: software path; browser-support interface helps
      QTWEBENGINE_DISABLE_SANDBOX: "1"

parts:
  surveyor-app:
    plugin: nil
    source: .

    build-packages:
      - python3-pip
      - python3-setuptools
      - python3-wheel
      - rsync

    organize:
      "data/xml":          share/surveyor_pocket_tools/data/xml
      "data/proj_data":    share/surveyor_pocket_tools/data/proj_data
      "data/database":     share/surveyor_pocket_tools/data/database
      "data/example":      share/surveyor_pocket_tools/data/example
      "data/markers":      share/surveyor_pocket_tools/data/markers
      # 'surveyor-pocket-tools.desktop': usr/share/applications/surveyor-pocket-tools.desktop
      # 'meta/gui/icon.png': meta/gui/icon.png

    stage-packages:
      # Qt/X backends and input
      - libxtst6
      - libxkbcommon-x11-0
      - xkb-data
      - libxkbcommon0

      # Wayland (optional)
      - libwayland-client0
      - libwayland-cursor0
      - libwayland-egl1

      # OpenGL loader (provides libGL.so.1)
      - libgl1
      - libgl1-mesa-glx

      # GL / Vulkan drivers
      - libgl1-mesa-dri
      - mesa-vulkan-drivers
      - libvulkan1

      # Printing (QtPrintSupport deps)
      - libcups2

      # Python runtime (keep Python 3.10)
      - python3.10
      - python3.10-distutils
      - locales

      # GSettings & schema compiler
      - gsettings-desktop-schemas
      - libglib2.0-bin
      - dconf-gsettings-backend
      - libdconf1
      - libglib2.0-0

      # Media for QtWebEngine
      - gstreamer1.0-plugins-base
      - gstreamer1.0-plugins-good
      - libgstreamer1.0-0
      - libgstreamer-plugins-base1.0-0
      - libgstreamer-plugins-good1.0-0

      # Fonts & basics
      - fonts-dejavu-core
      - libfontconfig1
      - fontconfig-config
      - libfreetype6

      # TLS/crypto/sound
      - libnss3
      - libnspr4
      - libasound2

    override-build: |
      set -e

      # Provide /usr/bin/python3 inside the snap
      mkdir -p "$SNAPCRAFT_PART_INSTALL/usr/bin"
      ln -sf /usr/bin/python3.10 "$SNAPCRAFT_PART_INSTALL/usr/bin/python3"

      # App layout
      mkdir -p "$SNAPCRAFT_PART_INSTALL/share/surveyor_pocket_tools/data"
      mkdir -p "$SNAPCRAFT_PART_INSTALL/data"

      # Sources
      cp src/*.py "$SNAPCRAFT_PART_INSTALL/" || true
      if [ -f main.py ]; then cp main.py "$SNAPCRAFT_PART_INSTALL/"; fi

      # Seed data (ro under share/, seed under data/ for first-run copy)
      if [ -d data ]; then
        cp -r data/* "$SNAPCRAFT_PART_INSTALL/share/surveyor_pocket_tools/data/" || true
        cp -r data/* "$SNAPCRAFT_PART_INSTALL/data/" 2>/dev/null || true
      fi

      # Vendor Python deps into PART_INSTALL (PySide6 6.9.1, Python 3.10)
      python3.10 -m pip install \
        --no-cache-dir --no-warn-script-location \
        --target="$SNAPCRAFT_PART_INSTALL/lib/python3.10/site-packages" \
        "PySide6==6.9.1" \
        pyproj openpyxl lxml simplekml pyshp geographiclib darkdetect gmplot

      # Copy desktop file and icon
      mkdir -p $SNAPCRAFT_PART_INSTALL/usr/share/applications
      mkdir -p $SNAPCRAFT_PART_INSTALL/meta/gui
      # Copy desktop file
      cp surveyor-pocket-tools.desktop $SNAPCRAFT_PART_INSTALL/usr/share/applications/
      # Copy icon file
      cp meta/gui/icon.png $SNAPCRAFT_PART_INSTALL/meta/gui/

    override-prime: |
      set -e
      craftctl default

      # Compile GSettings schemas using the staged binary
      "$SNAPCRAFT_PRIME/usr/bin/glib-compile-schemas" "$SNAPCRAFT_PRIME/usr/share/glib-2.0/schemas"

      # Ensure Python deps are in the final snap (PRIME)
      mkdir -p "$SNAPCRAFT_PRIME/lib/python3.10/site-packages"
      if [ -d "$SNAPCRAFT_PART_INSTALL/lib/python3.10/site-packages" ]; then
        rsync -a --delete "$SNAPCRAFT_PART_INSTALL/lib/python3.10/site-packages/" \
          "$SNAPCRAFT_PRIME/lib/python3.10/site-packages/"
      fi

      # Build-arch triplet (used by the launcher)
      TRIPLET="$SNAPCRAFT_ARCH_TRIPLET"
      mkdir -p "$SNAPCRAFT_PRIME/bin"

      # Create launcher with shebang at byte 0; keep PYTHONPATH, expose snap libs
      printf '%s\n' '#!/bin/bash' \
        '# ---------- wipe host env that can leak into the snap ----------' \
        'unset QT_PLUGIN_PATH QML2_IMPORT_PATH QT_QPA_PLATFORM QT_XCB_GL_INTEGRATION QT_QPA_PLATFORMTHEME QT_STYLE_OVERRIDE QT_SCALE_FACTOR QT_FONT_DPI PYTHONHOME LD_LIBRARY_PATH' \
        '' \
        '# ---------- Locale ----------' \
        'export LANG="${LANG:-C.UTF-8}"' \
        'export LC_ALL="${LC_ALL:-C.UTF-8}"' \
        '' \
        '# ---------- Desktop / GSettings ----------' \
        'export GSETTINGS_SCHEMA_DIR="$SNAP/usr/share/glib-2.0/schemas"' \
        'export GIO_EXTRA_MODULES="$SNAP/usr/lib/'"$TRIPLET"'/gio/modules"' \
        'export XDG_DATA_DIRS="$SNAP/usr/share:/usr/share${XDG_DATA_DIRS:+:$XDG_DATA_DIRS}"' \
        '' \
        '# ---------- Qt / PySide6 ----------' \
        'export QT_PLUGIN_PATH="$SNAP/lib/python3.10/site-packages/PySide6/Qt/plugins"' \
        'export QML2_IMPORT_PATH="$SNAP/lib/python3.10/site-packages/PySide6/Qt/qml"' \
        'export QTWEBENGINEPROCESS_PATH="$SNAP/lib/python3.10/site-packages/PySide6/Qt/libexec/QtWebEngineProcess"' \
        'export PYTHONPATH="${PYTHONPATH:-$SNAP/lib/python3.10/site-packages}"' \
        '' \
        '# Backend selection: default XCB (robust), opt-in to Wayland' \
        'if [ "${SURVEYOR_QT_BACKEND:-xcb}" = "wayland" ]; then' \
        '  export QT_QPA_PLATFORM="wayland"' \
        'else' \
        '  export QT_QPA_PLATFORM="xcb"' \
        '  export QT_XCB_GL_INTEGRATION="xcb_egl"' \
        'fi' \
        '' \
        '# ---------- GL/Vulkan inside snap ----------' \
        'export LIBGL_DRIVERS_PATH="$SNAP/usr/lib/'"$TRIPLET"'/dri"' \
        'export __EGL_VENDOR_LIBRARY_DIRS="$SNAP/usr/share/glvnd/egl_vendor.d"' \
        'export LD_LIBRARY_PATH="$SNAP/usr/lib/'"$TRIPLET"'":"$SNAP/usr/lib"' \
        '' \
        '# ---------- Persisted user data ----------' \
        'mkdir -p "$SNAP_USER_DATA/data"' \
        'export SURVEYOR_ACTUAL_DATA_DIR="$SNAP_USER_DATA/data"' \
        '' \
        '# ---------- PROJ ----------' \
        'export PROJ_DATA="$SNAP/share/surveyor_pocket_tools/data/proj_data/share/proj:$SNAP_USER_DATA/data/proj_data/share/proj"' \
        'export PROJ_LIB="$PROJ_DATA"' \
        'export PROJ_NETWORK="${PROJ_NETWORK:-OFF}"' \
        '' \
        '# ---------- QtWebEngine (software path) ----------' \
        'export QTWEBENGINE_CHROMIUM_FLAGS="${QTWEBENGINE_CHROMIUM_FLAGS:---use-gl=angle --use-angle=swiftshader --disable-gpu-compositing}"' \
        '' \
        'export QT_STYLE_OVERRIDE="Fusion"' \
        '' \
        'echo "Running in snap."' \
        'exec "$SNAP/usr/bin/python3" "$SNAP/main.py" "$@"' \
        > "$SNAPCRAFT_PRIME/bin/run-app"

      # Ensure LF endings and executable bit
      sed -i 's/\r$//' "$SNAPCRAFT_PRIME/bin/run-app"
      chmod 0755 "$SNAPCRAFT_PRIME/bin/run-app"

โครงสร้างไดเรคทอรี

ผมใส่โครงสร้างไดเรคทอรีไว้ เผื่อมีคนสนใจจะศึกษา ต้องยอมรับว่ายุคของ AI นี่เป็นข้อเสียของการใช้ คนใช้ AI คือคนที่รู้แต่รู้ไม่ลึก ไม่จริงแบบผม ในกรณีสร้างตัวติดตั้ง snap

.
├── build.log
├── data
│   ├── database
│   ├── example
│   ├── geoidgrids
│   ├── markers
│   ├── proj_data
│   └── xml
├── meta
│   └── gui
├── resources
├── snap
│   └── snapcraft.yaml
├── snap-contents
│   ├── bin
│   ├── computearea.py
│   ├── csvimportopts.py
│   ├── dpi.py
│   ├── egm.py
│   ├── etc
│   ├── extended.py
│   ├── ftranscoors.py
│   ├── geodesic.py
│   ├── geoprojcoorsdb.py
│   ├── gmapgeo.py
│   ├── gmapwebview.py
│   ├── greatcircle.py
│   ├── images_rc.py
│   ├── inout.py
│   ├── ldpchecked.py
│   ├── ldpdb.py
│   ├── ldpdesign.py
│   ├── lib
│   ├── linescalefactor.py
│   ├── main.py
│   ├── meta
│   ├── pointscalefactor.py
│   ├── readwriteXMLFile.py
│   ├── settings.py
│   ├── share
│   ├── transcoors.py
│   ├── usr
│   ├── utilities.py
│   └── utm2geo.py
├── snapcraft.yaml.backup
├── snapcraft.yaml.backup2
├── snapcraft.yaml.backup3
├── snapcraft.yaml.backup4
├── snapcraft.yaml.backup.failed
├── snapcraft.yaml.bad.28082025
├── snapcraft.yaml.manjaro.backup
├── snapcraft.yaml.manjaro.backup.good
├── snapcraft.yaml.work.but.not.qwebengine.manjaro
├── src
│   ├── computearea.py
│   ├── csvimportopts.py
│   ├── dpi.py
│   ├── egm.py
│   ├── extended.py
│   ├── ftranscoors.py
│   ├── geodesic.py
│   ├── geoprojcoorsdb.py
│   ├── gmapgeo.py
│   ├── gmapwebview.py
│   ├── greatcircle.py
│   ├── images_rc.py
│   ├── inout.py
│   ├── ldpchecked.py
│   ├── ldpdb.py
│   ├── ldpdesign.py
│   ├── linescalefactor.py
│   ├── main.py
│   ├── pointscalefactor.py
│   ├── readwriteXMLFile.py
│   ├── settings.py
│   ├── transcoors.py
│   ├── utilities.py
│   └── utm2geo.py
├── structure_level2.txt
├── surveyor-pocket-tools_1.0_amd64.snap
└── surveyor-pocket-tools.desktop

ติดตั้ง Snap

หลังจากผมติดตั้งโปรแกรมด้วยคำสั่งมือ แล้วกดคีย์ “Super” หรือ “Win” แล้วพิมพ์คำว่า “survey” จะเห็นไอคอนโปรแกรมเด้งมารอให้กดปุ่ม Enter เพื่อเรียกใช้โปรแกรม

รันโปรแกรม

ทดลองรันโปรแกรม จะเห็นว่าธีมมืดสว่างไม่สามารถใช้งานได้ chatGPT + พยายามอยู่พักใหญ่ๆแต่ยังไม่สำเร็จ ไม่เป็นไร ถือว่าไม้สามเขาทำได้ดี

Roast Me!!!! เปิดเผยการเก็บข้อมูลส่วนตัว

ความเก่งของ chatGPT + ผมชื่นชมได้ไม่นาน จนมีคนส่งพร้อมพ์ “Roast me” ประมาณว่าช่วยเผาฉันหน่อยหรือล้อเลียนฉันหน่อย เอาแบบแรงๆหน่อย ผลที่ได้คือความขำในตอนแรกๆครับ หลังจากตั้งสติได้ โอ้โห ข้อมูลเราที่ chatGPT เก็บไว้ทุกเม็ด มุกล้อเลียนบางอย่างแทบจะนึกไมได้ว่าเคยถามไป จนกระทั่งเขาเอามาล้อเลียน ผมสั่งให้ Roast me ต่อเนื่องประมาณ 10 ครั้ง เพื่อดูข้อมูลว่าเขาเก็บไว้ขนาดไหน สรุปแล้วเขาบันทึกไว้หมด

ผมกลับไปลบข้อมูลการสนทนาทุกๆสนทนา และไปปิดออปชั่นต่างๆ ถึงแม้ตอนนี้ข้อมูลได้เสร็จโจรไปแล้วก็ตาม แต่ยังดีที่ไม่ทำอะไร ตามที่ผมโปรยไว้ ว่าจะสร้างอีเมล์ใหม่แล้วเวียนใช้ AI ให้เขาไม่สามารถจะจดจำเราได้

เมื่อเอาพร้อมพ์ไปสั่งที่เจมิไน กลับได้คำตอบต่างๆว่า AI เขาไม่สามารถเข้าถึงข้อมูลส่วนตัวผู้ใช้ได้ ฟังแล้วเหมือนจะดี แต่ผมก็ไม่ได้ไว้ใจ เพราะข้อมูลที่สนทนากับเราเขาจะเก็บไว้เอาไปใช้เทรนโมเดล แต่จะแอบเก็บข้อมูลอื่นหรือไม่ ไม่แน่ใจ ความรู้สึกส่วนตัวครับ ไม่ไว้ใจ

เอาพร้อมพ์ไปลองกับดีพซีค กลับมีการล้อเลียนขำๆ แต่ไม่มีข้อมูลระบุความเป็นส่วนตัว แต่เซิฟเวอร์จะแอบเก็บข้อมูลผู้ใช้ไว้ไหม แบบที่สื่อฝรั่งร้องแรกแหกกระเชอว่าจีนได้เก็บข้อมูลส่วนตัวที่สนทนาไว้ ก็ไม่มั่นใจ สรุปแล้วไม่ไว้ใจเช่นกันครับ

Leave a Reply

Your email address will not be published. Required fields are marked *