Files
arkts-skills/harmonyos-build-deploy/SKILL.md

17 KiB

name, description
name description
harmonyos-build-deploy HarmonyOS application build, clean, package and device installation. Use when building HarmonyOS projects with hvigorw, cleaning build artifacts, managing ohpm dependencies, packaging HAP/HSP/APP bundles, installing to devices via hdc, or troubleshooting installation errors like "version code not same".

HarmonyOS Build & Deploy

Complete workflow for building, cleaning, packaging, and installing HarmonyOS applications.

First Step: Confirm Operation with User

IMPORTANT: Before executing any build or deploy operation, confirm which specific operation(s) the user wants to perform. Ask the user to choose from:

Operation Description
Clean build artifacts Remove previous build cache and outputs
Install dependencies Use ohpm to install project dependencies
Build project Use hvigorw to build HAP/APP packages
Install to device Use hdc to install the app on a device
Full pipeline Clean → install deps → build → deploy to device

Why confirm first:

  • Different scenarios require different operations (e.g., incremental build vs clean build)
  • Avoid unnecessary time-consuming operations
  • Give user control over the workflow
  • Prevent accidental device installation

After user responds:

  • Execute only the selected operations
  • Report progress and results clearly

Quick Reference

# Build complete app (incremental)
hvigorw assembleApp --mode project -p product=default -p buildMode=release --no-daemon

# Install to device (check actual output path in your project)
hdc -t <UDID> shell "rm -rf /data/local/tmp/install && mkdir -p /data/local/tmp/install"
hdc -t <UDID> file send <output_path>/signed /data/local/tmp/install
hdc -t <UDID> shell "bm install -p /data/local/tmp/install/signed"

Note: Build output path varies by project. Common paths:

  • outputs/default/signed/
  • outputs/project/bundles/signed/

Check your project's actual output after build.

Workflows

IMPORTANT: Build, clean, and deploy operations are long-running (build ~30s, file transfer ~20s). Always delegate these workflows to a subagent to avoid timeout. This also provides better error handling and clearer progress reporting to the user.

Clean Build & Deploy

Delegate to subagent with the following steps:

  1. Clean: hvigorw clean --no-daemon
  2. Install dependencies: ohpm install --all
  3. Build: hvigorw assembleApp --mode project -p product=default -p buildMode=release --no-daemon
  4. Find build output (check outputs/default/signed/ or outputs/project/bundles/signed/)
  5. Deploy to device:
    hdc -t <UDID> shell "rm -rf /data/local/tmp/install && mkdir -p /data/local/tmp/install"
    hdc -t <UDID> file send <output_path>/signed /data/local/tmp/install
    hdc -t <UDID> shell "bm install -p /data/local/tmp/install/signed"
    
  6. Launch: hdc -t <UDID> shell "aa start -a EntryAbility -b <bundleName>"
  7. Report success/failure with details

Deploy Only (No Rebuild)

Delegate to subagent with the following steps:

  1. Read AppScope/app.json5 to get bundleName
  2. Push existing build output to device:
    hdc -t <UDID> shell "rm -rf /data/local/tmp/install && mkdir -p /data/local/tmp/install"
    hdc -t <UDID> file send <output_path>/signed /data/local/tmp/install
    hdc -t <UDID> shell "bm install -p /data/local/tmp/install/signed"
    
  3. Launch: hdc -t <UDID> shell "aa start -a EntryAbility -b <bundleName>"
  4. Report success/failure with details

Restart App

Delegate to subagent:

  1. Force stop: hdc -t <UDID> shell "aa force-stop <bundleName>"
  2. Launch: hdc -t <UDID> shell "aa start -a EntryAbility -b <bundleName>"
  3. Report success/failure

Clean App Cache/Data

Delegate to subagent:

  1. Clean cache: hdc -t <UDID> shell "bm clean -n <bundleName> -c"
  2. Clean data: hdc -t <UDID> shell "bm clean -n <bundleName> -d"
  3. Report success/failure

Build Commands (hvigorw)

Incremental Build (Default)

Use incremental build for normal development - only changed modules are rebuilt:

# Build complete app (incremental)
hvigorw assembleApp --mode project -p product=default -p buildMode=release --no-daemon

Single Module Build

Build only a specific module for faster iteration:

# Build single HAP module
hvigorw assembleHap -p module=entry@default --mode module -p buildMode=release --no-daemon

# Build single HSP module
hvigorw assembleHsp -p module=feature_module@default --mode module -p buildMode=release --no-daemon

# Build single HAR module
hvigorw assembleHar -p module=library@default --mode module -p buildMode=release --no-daemon

# Build multiple modules at once
hvigorw assembleHsp -p module=module1@default,module2@default --mode module -p buildMode=release --no-daemon

Module name format: {moduleName}@{targetName}

  • moduleName: Directory name of the module (e.g., entry, feature_home)
  • targetName: Target defined in module's build-profile.json5 (usually default)

When to use single module build:

  • Developing/debugging a specific module
  • Faster build times during iteration
  • Testing changes in isolated module

Note: After single module build, you still need to run assembleApp to package the complete application for installation.

Clean Build (When Needed)

Only perform clean build when:

  • First time building the project
  • Encountering unexplained build errors
  • After modifying build-profile.json5 or SDK version
  • Dependency resolution issues
# Clean build artifacts
hvigorw clean --no-daemon

# Deep clean (for dependency issues)
ohpm clean && ohpm cache clean
ohpm install --all
hvigorw --sync -p product=default -p buildMode=release --no-daemon
hvigorw assembleApp --mode project -p product=default -p buildMode=release --no-daemon

Install Dependencies (ohpm)

# Install all dependencies (after clean or first clone)
ohpm install --all

# With custom registry
ohpm install --all --registry "https://repo.harmonyos.com/ohpm/"

Sync Project

Only needed after modifying build-profile.json5 or oh-package.json5:

hvigorw --sync -p product=default -p buildMode=release --no-daemon

Build Types

hvigorw assembleHap    # Build HAP (Harmony Ability Package)
hvigorw assembleHsp    # Build HSP (Harmony Shared Package)
hvigorw assembleHar    # Build HAR (Harmony Archive)
hvigorw assembleApp    # Build complete APP bundle

Build Parameters

Parameter Description
-p product={name} Target product defined in build-profile.json5
-p buildMode={debug|release} Build mode
-p module={name}@{target} Target module with --mode module
--mode project Build all modules in project
--mode module Build specific module only
--no-daemon Disable daemon (recommended for CI)
--analyze=advanced Enable build analysis

Build Outputs

Build output path varies by project configuration. Common patterns:

outputs/
├── default/signed/                      # Pattern 1
│   ├── entry-default-signed.hap
│   └── *.hsp
└── project/bundles/signed/              # Pattern 2
    ├── entry-default-signed.hap
    └── *.hsp

Tip: After build, check the actual output directory in your project.

Module Types

Type Extension Description
HAP .hap Harmony Ability Package - Application entry module
HSP .hsp Harmony Shared Package - Dynamic shared library
HAR .har Harmony Archive - Static library (compiled into HAP)
APP .app Complete application bundle (all HAP + HSP)

Finding Modules

All modules are defined in build-profile.json5 at the project root, in the modules array.

Module Definition Structure

{
  "modules": [
    {
      "name": "entry",              // Module name (used in build commands)
      "srcPath": "./entry",         // Module source path (relative to project root)
      "targets": [                  // Build target config (optional)
        {
          "name": "default",
          "applyToProducts": ["default", "app_store"]
        }
      ]
    },
    {
      "name": "support_http",
      "srcPath": "./support/support_http",
      "targets": [...]
    }
  ]
}

Key Fields

Field Description
name Module name, used in build commands (e.g., -p module=entry@default)
srcPath Module source path relative to project root
targets Build target config, specifies which products this module applies to

Module Type Identification

The module type is defined in each module's module.json5 file:

// {module}/src/main/module.json5
{
  "module": {
    "name": "entry",
    "type": "entry",    // "entry" = HAP, "shared" = HSP, "har" = HAR
    ...
  }
}
type field value Module Type Description
"entry" HAP Application entry point
"feature" HAP Feature module (also a HAP)
"shared" HSP Dynamic shared package
"har" HAR Static library (compiled into other modules)

To identify all module types in a project:

# Read type from each module's module.json5
# Check {srcPath}/src/main/module.json5 for each module in build-profile.json5

Finding Module Build Outputs

Module build outputs are located at:

{srcPath}/build/default/outputs/default/

Note: Debug and Release builds output to the same directory. The difference is in the signing configuration used (defined in build-profile.json5signingConfigs).

Output Files

File Description
{name}-default-signed.hsp Signed HSP (ready for installation)
{name}-default-unsigned.hsp Unsigned HSP
{name}.har HAR static library
app/{name}-default.hsp Intermediate artifact
mapping/sourceMaps.map Source maps for debugging

Example

For module support_http with srcPath: "./support/support_http":

support/support_http/build/default/outputs/default/
├── support_http-default-signed.hsp    ← Signed, ready to install
├── support_http-default-unsigned.hsp
├── support_http.har
├── app/
│   └── support_http-default.hsp
├── mapping/
│   └── sourceMaps.map
└── pack.info

Search Commands

# Find all signed HSP/HAP outputs
dir /s /b "*-signed.hsp" "*-signed.hap" 2>nul           # Windows
find . -name "*-signed.hsp" -o -name "*-signed.hap"     # Linux/macOS

# Find specific module's output
dir /s /b "{srcPath}\build\default\outputs\default\*"   # Windows
ls -la {srcPath}/build/default/outputs/default/         # Linux/macOS

Notes

  1. Build required: If build/ directory doesn't exist, run build first
  2. Project-level outputs: Complete app bundle is in project root outputs/ after assembleApp
  3. oh_modules outputs: Dependency modules may have outputs in oh_modules/@xxx/build/... (these are resolved dependencies)

Unwanted Modules in Output Directory

Sometimes HSP files appear in the output directory that are not listed in build-profile.json5. These modules should not be deployed.

Cause: Build scripts or dependency configurations may copy precompiled HSP files to the output directory, even though they are not part of the current build.

How to identify:

  1. Check build-profile.json5modules array
  2. If an HSP file in output is not in the modules list, it should be removed before installation

Solution: Remove these HSP files before installation:

# Example: Remove modules not in build-profile.json5
rm <output_path>/signed/unwanted-module-default-signed.hsp

Note: Installation will fail with "version code not same" error if these unwanted modules have a different versionCode than the main app. The root cause is that these modules shouldn't be deployed at all.

Device Installation (hdc)

hdc (HarmonyOS Device Connector) is the CLI tool for device communication, similar to Android's adb.

Check Device

hdc list targets              # List connected devices (returns UDID)
hdc -t <UDID> shell "whoami"  # Test connection

Push and Install

# Clear device directory
hdc -t <UDID> shell "rm -rf /data/local/tmp/install && mkdir -p /data/local/tmp/install"

# Push signed bundles
hdc -t <UDID> file send path/to/signed /data/local/tmp/install

# Install all HAP/HSP in directory
hdc -t <UDID> shell "bm install -p /data/local/tmp/install/signed"

Verify and Launch

# Check installation
hdc -t <UDID> shell "bm dump -n <bundleName>"

# Launch app (default ability)
hdc -t <UDID> shell "aa start -a EntryAbility -b <bundleName>"

Uninstall

hdc -t <UDID> shell "bm uninstall -n <bundleName>"

hdc Command Reference

Command Description
hdc list targets List connected devices
hdc -t <UDID> shell "<cmd>" Execute shell command on device
hdc -t <UDID> file send <local> <remote> Push file/directory to device
hdc -t <UDID> file recv <remote> <local> Pull file/directory from device
hdc tconn <IP>:<port> Connect to device over Wi-Fi
hdc kill Kill hdc server
hdc start Start hdc server
hdc version Show hdc version

Wireless Debugging (Wi-Fi)

Connect to a device over the network instead of USB:

# 1. Connect device via USB first and get its IP address
hdc -t <UDID> shell "ifconfig"    # Find wlan0 IP

# 2. Enable TCP port on device (if supported)
hdc -t <UDID> shell "param set persist.hdc.port 5555"

# 3. Connect wirelessly
hdc tconn <device_IP>:5555

# 4. Verify connection
hdc list targets
# Should show the IP-based target alongside or instead of USB target

Note: Wireless debugging may require the device and host to be on the same network. Not all devices support hdc tconn.

Bundle Manager (bm)

Run via hdc -t <UDID> shell "bm ...":

Command Description
bm install -p <path> Install from directory (all HAP/HSP)
bm install -p <file.hap> Install single HAP file
bm install -r -p <path> Reinstall (replace existing, keep data)
bm uninstall -n <bundleName> Uninstall application
bm dump -n <bundleName> Show package info
bm dump -a List all installed packages
bm clean -n <bundleName> -c Clean application cache
bm clean -n <bundleName> -d Clean application data

Ability Assistant (aa)

Run via hdc -t <UDID> shell "aa ...":

Command Description
aa start -a <ability> -b <bundle> Start specific ability
aa force-stop <bundleName> Force stop application
aa dump -a Dump all running abilities

Device Logging (hilog)

Use hilog to view device logs for debugging. Run via hdc -t <UDID> shell "hilog ...":

# Stream all logs (like adb logcat)
hdc -t <UDID> shell "hilog"

# Filter by tag
hdc -t <UDID> shell "hilog -T <tag>"

# Filter by log level (DEBUG, INFO, WARN, ERROR, FATAL)
hdc -t <UDID> shell "hilog -L ERROR"

# Combine tag and level filters
hdc -t <UDID> shell "hilog -T MyApp -L WARN"

# Clear log buffer
hdc -t <UDID> shell "hilog -r"

# Show only recent logs (last N lines)
hdc -t <UDID> shell "hilog -t 100"

Common hilog Options

Option Description
-T <tag> Filter by log tag
-L <level> Minimum log level: DEBUG, INFO, WARN, ERROR, FATAL
-D <domain> Filter by domain (hex, e.g., 0x0001)
-r Clear log buffer
-t <count> Show only last N log entries
-x Exit after printing existing logs (no streaming)

Logging in ArkTS Code

import { hilog } from '@kit.PerformanceAnalysisKit';

const DOMAIN: number = 0x0000;
const TAG: string = 'MyApp';

hilog.info(DOMAIN, TAG, 'Application started');
hilog.error(DOMAIN, TAG, 'Failed to load data: %{public}s', error.message);

Note: Use %{public}s for string parameters that should be visible in logs. Without {public}, parameters are masked as <private> in release builds.

Troubleshooting

Error Cause Solution
version code not same HSP in output not in build-profile.json5 Remove unwanted HSP files before install
install parse profile prop check error Signature/profile mismatch Check signing config in build-profile.json5
install failed due to older sdk version Device SDK < app's compatibleSdkVersion Update device or lower compatibleSdkVersion
Device not found Connection issue Check USB, enable debugging, hdc kill && hdc start
signature verification failed Invalid or expired certificate Regenerate signing certificate

Key Configuration Files

File Description
AppScope/app.json5 App metadata (bundleName, versionCode, versionName)
build-profile.json5 Modules list, products, signing configs
{module}/src/main/module.json5 Module config (abilities, permissions)
{module}/oh-package.json5 Module dependencies

Reference Files

  • arkts-development: ArkTS/ArkUI development patterns, state management, component lifecycle, and API usage. Use alongside this skill when developing HarmonyOS application code.