Intern 支持在 BrowserStack、SauceLabs 和 TestingBot 等服務上遠程運行測試。您可以選用其中一個服務,注冊一個帳號并將憑據(jù)提供給 cli-test-intern。默認情況下,所有測試服務會在 IE11、Firefox 和 Chrome 等瀏覽器上運行測試。
成都創(chuàng)新互聯(lián)于2013年開始,是專業(yè)互聯(lián)網(wǎng)技術服務公司,擁有項目成都網(wǎng)站建設、網(wǎng)站設計網(wǎng)站策劃,項目實施與項目整合能力。我們以讓每一個夢想脫穎而出為使命,1280元蘭山做網(wǎng)站,已為上家服務,為蘭山各地企業(yè)和個人服務,聯(lián)系電話:028-86922220
使用 BrowserStack 服務,需提供 access key 和用戶名。Access key 和用戶名可在命令行中指定或設置為環(huán)境變量,詳見 Intern 文檔。
dojo test -a -c browserstack -k --userName
或使用環(huán)境變量
BROWSERSTACK_USERNAME= BROWSERSTACK_ACCESS_KEY= dojo test -a -c browserstack
使用 SauceLabs 服務,需提供 access key 和用戶名。Access key 和用戶名可在命令行中指定或設置為環(huán)境變量,詳見 Intern 文檔。
dojo test -a -c saucelabs -k --userName
或使用環(huán)境變量
SAUCE_USERNAME= SAUCE_ACCESS_KEY= dojo test -a -c saucelabs
使用 TestingBot 服務,需提供 key 和 secret。Key 和 secret 可在命令行中指定或設置為環(huán)境變量,詳見 Intern 文檔。
dojo test -a -c testingbot -k -s
或使用環(huán)境變量
TESTINGBOT_SECRET= TESTINGBOT_KEY= dojo test -a -c saucelabs
當使用 @dojo/framework/testing
時,harness()
是最重要的 API,主要用于設置每一個測試并提供一個執(zhí)行虛擬 DOM 斷言和交互的上下文。目的在于當更新 properties
或 children
,以及部件失效時,鏡像部件的核心行為,并且不需要任何特殊或自定義邏輯。
interface HarnessOptions {
customComparators?: CustomComparator[];
middleware?: [MiddlewareResultFactory, MiddlewareResultFactory][];
}
harness(renderFunction: () => WNode, customComparators?: CustomComparator[]): Harness;
harness(renderFunction: () => WNode, options?: HarnessOptions): Harness;
renderFunction
: 返回被測部件 WNode 的函數(shù)customComparators
: 一組自定義的比較器描述符。每個描述符提供一個比較器函數(shù),用于比較通過 selector
和 property
定位到的 properties
options
: harness 的擴展選項,包括 customComparators
和一組 middleware/mocks 元組。harness 函數(shù)返回一個 Harness
對象,該對象提供了幾個與被測部件交互的 API:
Harness
expect
: 對被測部件完整的渲染結果執(zhí)行斷言expectPartial
: 對被測部件部分渲染結果執(zhí)行斷言trigger
: 用于在被測部件的節(jié)點上觸發(fā)函數(shù)getRender
: 根據(jù)提供的索引,從 harness 中返回對應的渲染器使用 @dojo/framework/core
中的 w()
函數(shù)生成一個用于測試的部件是非常簡單的:
const { describe, it } = intern.getInterface('bdd');
import WidgetBase from '@dojo/framework/widget-core/WidgetBase';
import harness from '@dojo/framework/testing/harness';
import { w, v } from '@dojo/framework/widget-core/d';
class MyWidget extends WidgetBase<{ foo: string }> {
protected render() {
const { foo } = this.properties;
return v('div', { foo }, this.children);
}
}
const h = harness(() => w(MyWidget, { foo: 'bar' }, ['child']));
如下所示,harness 函數(shù)也支持 tsx
。README 文檔中其余示例均使用編程式的 w()
API,在 單元測試 中可查看更多 tsx
示例。
const h = harness(() => child );
renderFunction
是延遲執(zhí)行的,所以可在斷言之間包含額外的邏輯來操作部件的 properties
和 children
。
describe('MyWidget', () => {
it('renders with foo correctly', () => {
let foo = 'bar';
const h = harness(() => {
return w(MyWidget, { foo }, [ 'child' ]));
};
h.expect(/** assertion that includes bar **/);
// update the property that is passed to the widget
foo = 'foo';
h.expect(/** assertion that includes foo **/)
});
});
當初始化 harness 時,可將 mock 中間件指定為 HarnessOptions
值的一部分。Mock 中間件被定義為由原始的中間件和 mock 中間件的實現(xiàn)組成的元組。Mock 中間件的創(chuàng)建方式與其他中間件相同。
import myMiddleware from './myMiddleware';
import myMockMiddleware from './myMockMiddleware';
import harness from '@dojo/framework/testing/harness';
import MyWidget from './MyWidget';
describe('MyWidget', () => {
it('renders', () => {
const h = harness(() => , { middleware: [[myMiddleware, myMockMiddleware]] });
h.expect(/** assertion that executes the mock middleware instead of the normal middleware **/);
});
});
Harness 會自動 mock 很多核心中間件,并注入到任何需要他們的中間件中:
當測試使用了 Dojo 中間件的部件時,有很多 mock 中間件可以使用。Mock 會導出一個 factory,該 factory 會創(chuàng)建一個受限作用域的 mock 中間件,會在每個測試中使用。
node
中間件使用 @dojo/framework/testing/mocks/middleware/node
中的 createNodeMock
可 mock 一個 node 中間件。要設置從 node mock 中返回的期望值,需要調用創(chuàng)建的 mock node 中間件,并傳入 key
和期望的 DOM node。
import createNodeMock from '@dojo/framework/testing/mocks/middleware/node';
// create the mock node middleware
const mockNode = createNodeMock();
// create a mock DOM node
const domNode = {};
// call the mock middleware with a key and the DOM
// to return.
mockNode('key', domNode);
intersection
中間件使用 @dojo/framework/testing/mocks/middleware/intersection
中的 createIntersectionMock
可 mock 一個 intersection 中間件。要設置從 intersection mock 中返回的期望值,需要調用創(chuàng)建的 mock intersection 中間件,并傳入 key
和期望的 intersection 詳情。
考慮以下部件:
import { create, tsx } from '@dojo/framework/core/vdom';
import intersection from '@dojo/framework/core/middleware/intersection';
const factory = create({ intersection });
const App = factory(({ middleware: { intersection } }) => {
const details = intersection.get('root');
return {JSON.stringify(details)};
});
使用 mock intersection 中間件:
import { tsx } from '@dojo/framework/core/vdom';
import createIntersectionMock from '@dojo/framework/testing/mocks/middleware/intersection';
import intersection from '@dojo/framework/core/middleware/intersection';
import harness from '@dojo/framework/testing/harness';
import MyWidget from './MyWidget';
describe('MyWidget', () => {
it('test', () => {
// create the intersection mock
const intersectionMock = createIntersectionMock();
// pass the intersection mock to the harness so it knows to
// replace the original middleware
const h = harness(() => , { middleware: [[intersection, intersectionMock]] });
// call harness.expect as usual, asserting the default response
h.expect(() => {`{"intersectionRatio":0,"isIntersecting":false}`});
// use the intersection mock to set the expected return
// of the intersection middleware by key
intersectionMock('root', { isIntersecting: true });
// assert again with the updated expectation
h.expect(() => {`{"isIntersecting": true }`});
});
});
resize
中間件使用 @dojo/framework/testing/mocks/middleware/resize
中的 createResizeMock
可 mock 一個 resize 中間件。要設置從 resize mock 中返回的期望值,需要調用創(chuàng)建的 mock resize 中間件,并傳入 key
和期望的容納內容的矩形區(qū)域。
const mockResize = createResizeMock();
mockResize('key', { width: 100 });
考慮以下部件:
import { create, tsx } from '@dojo/framework/core/vdom'
import resize from '@dojo/framework/core/middleware/resize'
const factory = create({ resize });
export const MyWidget = factory(function MyWidget({ middleware }) => {
const { resize } = middleware;
const contentRects = resize.get('root');
return {JSON.stringify(contentRects)};
});
使用 mock resize 中間件:
import { tsx } from '@dojo/framework/core/vdom';
import createResizeMock from '@dojo/framework/testing/mocks/middleware/resize';
import resize from '@dojo/framework/core/middleware/resize';
import harness from '@dojo/framework/testing/harness';
import MyWidget from './MyWidget';
describe('MyWidget', () => {
it('test', () => {
// create the resize mock
const resizeMock = createResizeMock();
// pass the resize mock to the harness so it knows to replace the original
// middleware
const h = harness(() => , { middleware: [[resize, resizeMock]] });
// call harness.expect as usual
h.expect(() => null);
// use the resize mock to set the expected return of the resize middleware
// by key
resizeMock('root', { width: 100 });
// assert again with the updated expectation
h.expect(() => {`{"width":100}`});
});
});
Store
中間件使用 @dojo/framework/testing/mocks/middleware/store
中的 createMockStoreMiddleware
可 mock 一個強類型的 store 中間件,也支持 mock process。為了 mock 一個 store 的 process,可傳入一個由原始 store process 和 stub process 組成的元組。中間件會改為調用 stub,而不是調用原始的 process。如果沒有傳入 stub,中間件將停止調用所有的 process。
要修改 mock store 中的值,需要調用 mockStore
,并傳入一個返回一組 store 操作的函數(shù)。這將注入 store 的 path
函數(shù),以創(chuàng)建指向需要修改的狀態(tài)的指針。
mockStore((path) => [replace(path('details', { id: 'id' })]);
考慮以下部件:
src/MyWidget.tsx
import { create, tsx } from '@dojo/framework/core/vdom'
import { myProcess } from './processes';
import MyState from './interfaces';
// application store middleware typed with the state interface
// Example: `const store = createStoreMiddleware();`
import store from './store';
const factory = create({ store }).properties<{ id: string }>();
export default factory(function MyWidget({ properties, middleware: store }) {
const { id } = properties();
const { path, get, executor } = store;
const details = get(path('details');
let isLoading = get(path('isLoading'));
if ((!details || details.id !== id) && !isLoading) {
executor(myProcess)({ id });
isLoading = true;
}
if (isLoading) {
return ;
}
return ;
});
使用 mock store 中間件:
tests/unit/MyWidget.tsx
import { tsx } from '@dojo/framework/core/vdom'
import createMockStoreMiddleware from '@dojo/framework/testing/mocks/middleware/store';
import harness from '@dojo/framework/testing/harness';
import { myProcess } from './processes';
import MyWidget from './MyWidget';
import MyState from './interfaces';
import store from './store';
// import a stub/mock lib, doesn't have to be sinon
import { stub } from 'sinon';
describe('MyWidget', () => {
it('test', () => {
const properties = {
id: 'id'
};
const myProcessStub = stub();
// type safe mock store middleware
// pass through an array of tuples `[originalProcess, stub]` for mocked processes
// calls to processes not stubbed/mocked get ignored
const mockStore = createMockStoreMiddleware([[myProcess, myProcessStub]]);
const h = harness(() => , {
middleware: [store, mockStore]
});
h.expect(/* assertion template for `Loading`*/);
// assert again the stubbed process
expect(myProcessStub.calledWith({ id: 'id' })).toBeTruthy();
mockStore((path) => [replace(path('isLoading', true)]);
h.expect(/* assertion template for `Loading`*/);
expect(myProcessStub.calledOnce()).toBeTruthy();
// use the mock store to apply operations to the store
mockStore((path) => [replace(path('details', { id: 'id' })]);
mockStore((path) => [replace(path('isLoading', true)]);
h.expect(/* assertion template for `ShowDetails`*/);
properties.id = 'other';
h.expect(/* assertion template for `Loading`*/);
expect(myProcessStub.calledTwice()).toBeTruthy();
expect(myProcessStub.secondCall.calledWith({ id: 'other' })).toBeTruthy();
mockStore((path) => [replace(path('details', { id: 'other' })]);
h.expect(/* assertion template for `ShowDetails`*/);
});
});
在某些情況下,我們在測試期間無法得知屬性的確切值,所以需要使用自定義比較描述符(custom compare descriptor)。
描述符中有一個用于定位要檢查的虛擬節(jié)點的 selector
,一個應用自定義比較的屬性名和一個接收實際值并返回一個 boolean 類型斷言結果的比較器函數(shù)。
const compareId = {
selector: '*', // all nodes
property: 'id',
comparator: (value: any) => typeof value === 'string' // checks the property value is a string
};
const h = harness(() => w(MyWidget, {}), [compareId]);
對于所有的斷言,返回的 harness
API 將只對 id
屬性使用 comparator
進行測試,而不是標準的相等測試。
harness
API 支持 CSS style 選擇器概念,來定位要斷言和操作的虛擬 DOM 中的節(jié)點。查看支持的選擇器的完整列表以了解更多信息。
除了標準 API 之外還提供:
key
屬性簡寫為 @
符號.
來定位樣式類時,使用 classes
屬性而不是 class
屬性harness.expect
測試中最常見的需求是斷言部件的 render
函數(shù)的輸出結構。expect
接收一個返回被測部件期望的渲染結果的函數(shù)作為參數(shù)。
API
expect(expectedRenderFunction: () => DNode | DNode[], actualRenderFunction?: () => DNode | DNode[]);
expectedRenderFunction
: 返回查詢節(jié)點期望的 DNode
結構的函數(shù)actualRenderFunction
: 一個可選函數(shù),返回被斷言的實際 DNode
結構h.expect(() =>
v('div', { key: 'foo' }, [w(Widget, { key: 'child-widget' }), 'text node', v('span', { classes: ['class'] })])
);
expect
也可以接收第二個可選參數(shù),返回要斷言的渲染結果的函數(shù)。
h.expect(() => v('div', { key: 'foo' }), () => v('div', { key: 'foo' }));
如果實際的渲染輸出和期望的渲染輸出不同,就會拋出一個異常,并使用結構化的可視方法,用 (A)
(實際值)和 (E)
(期望值)指出所有不同點。
Example assertion failure output:
v('div', {
'classes': [
'root',
(A) 'other'
(E) 'another'
],
'onclick': 'function'
}, [
v('span', {
'classes': 'span',
'id': 'random-id',
'key': 'label',
'onclick': 'function',
'style': 'width: 100px'
}, [
'hello 0'
])
w(ChildWidget, {
'id': 'random-id',
'key': 'widget'
})
w('registry-item', {
'id': true,
'key': 'registry'
})
])
harness.trigger
harness.trigger()
在 selector
定位的節(jié)點上調用 name
指定的函數(shù)。
interface FunctionalSelector {
(node: VNode | WNode): undefined | Function;
}
trigger(selector: string, functionSelector: string | FunctionalSelector, ...args: any[]): any;
selector
: 用于查找目標節(jié)點的選擇器functionSelector
: 要么是從節(jié)點的屬性中找到的被調用的函數(shù)名,或者是從節(jié)點的屬性中返回一個函數(shù)的函數(shù)選擇器args
: 為定位到的函數(shù)傳入的參數(shù)如果有返回結果,則返回的是被觸發(fā)函數(shù)的結果。
用法示例:
// calls the `onclick` function on the first node with a key of `foo`
h.trigger('@foo', 'onclick');
// calls the `customFunction` function on the first node with a key of `bar` with an argument of `100`
// and receives the result of the triggered function
const result = h.trigger('@bar', 'customFunction', 100);
functionalSelector
返回部件屬性中的函數(shù)。函數(shù)也會被觸發(fā),與使用普通字符串 functionSelector
的方式相同。
用法示例:
假定有如下 VDOM 樹結構:
v(Toolbar, {
key: 'toolbar',
buttons: [
{
icon: 'save',
onClick: () => this._onSave()
},
{
icon: 'cancel',
onClick: () => this._onCancel()
}
]
});
并且你想觸發(fā) save 按鈕的 onClick
函數(shù)。
h.trigger('@buttons', (renderResult: DNode) => {
return renderResult.properties.buttons[0].onClick;
});
注意:如果沒能找到指定的選擇器,則 trigger
會拋出一個錯誤。
harness.getRender
harness.getRender()
返回索引指定的渲染器,如果沒有提供索引則返回最后一個渲染器。
getRender(index?: number);
index
: 要返回的渲染器的索引用法示例:
// Returns the result of the last render
const render = h.getRender();
// Returns the result of the render for the index provided
h.getRender(1);
斷言模板(assertion template)允許你構建期望的渲染函數(shù),用于傳入 h.expect()
中。斷言模板背后的思想來自經(jīng)常要斷言整個渲染輸出,并需要修改斷言的某些部分。
要使用斷言模板,需要先導入模塊:
import assertionTemplate from '@dojo/framework/testing/assertionTemplate';
然后,在你的測試中,你可以編寫一個基本斷言,它是部件的默認渲染狀態(tài):
假定有以下部件:
src/widgets/Profile.ts
import WidgetBase from '@dojo/framework/widget-core/WidgetBase';
import { v } from '@dojo/framework/widget-core/d';
import * as css from './styles/Profile.m.css';
export interface ProfileProperties {
username?: string;
}
export default class Profile extends WidgetBase {
protected render() {
const { username } = this.properties;
return v('h2', { classes: [css.root] }, [`Welcome ${username || 'Stranger'}!`]);
}
}
基本斷言如下所示:
tests/unit/widgets/Profile.ts
const { describe, it } = intern.getInterface('bdd');
import harness from '@dojo/framework/testing/harness';
import assertionTemplate from '@dojo/framework/testing/assertionTemplate';
import { w, v } from '@dojo/framework/widget-core/d';
import Profile from '../../../src/widgets/Profile';
import * as css from '../../../src/widgets/styles/Profile.m.css';
const profileAssertion = assertionTemplate(() =>
v('h2', { classes: [css.root], '~key': 'welcome' }, ['Welcome Stranger!'])
);
在測試中這樣寫:
tests/unit/widgets/Profile.ts
const profileAssertion = assertionTemplate(() =>
v('h2', { classes: [css.root], '~key': 'welcome' }, ['Welcome Stranger!'])
);
describe('Profile', () => {
it('default renders correctly', () => {
const h = harness(() => w(Profile, {}));
h.expect(profileAssertion);
});
});
it('default renders correctly', () => {
const h = harness(() => w(Profile, {}));
h.expect(profileAssertion);
});
現(xiàn)在我們看看,為 Profile
部件傳入 username
屬性后,如何測試輸出結果:
tests/unit/widgets/Profile.ts
describe('Profile', () => {
...
it('renders given username correctly', () => {
// update the expected result with a given username
const namedAssertion = profileAssertion.setChildren('~welcome', [
'Welcome Kel Varnsen!'
]);
const h = harness(() => w(Profile, { username: 'Kel Varnsen' }));
h.expect(namedAssertion);
});
});
這里,我們使用 baseAssertion 的 setChildren()
api,然后我們使用特殊的 ~
選擇器來定位 key 值為 ~message
的節(jié)點。~key
屬性(使用 tsx 的模板中是 assertion-key
)是斷言模板的一個特殊屬性,在斷言時會被刪除,因此在匹配渲染結構時不會顯示出來。此功能允許你修飾斷言模板,以便能簡單的選擇節(jié)點,而不需要擴展實際的部件渲染函數(shù)。一旦我們獲取到 message
節(jié)點,我們就可以將其子節(jié)點設置為期望的 the number 5
,然后在 h.expect
中使用生成的模板。需要注意的是,斷言模板在設置值時總是返回一個新的斷言模板,這可以確保你不會意外修改現(xiàn)有模板(可能導致其他測試失?。⒃试S你基于新模板,增量逐層構建出新的模板。
斷言模板具有以下 API:
insertBefore(selector: string, children: () => DNode[]): AssertionTemplateResult;
insertAfter(selector: string, children: () => DNode[]): AssertionTemplateResult;
insertSiblings(selector: string, children: () => DNode[], type?: 'before' | 'after'): AssertionTemplateResult;
append(selector: string, children: () => DNode[]): AssertionTemplateResult;
prepend(selector: string, children: () => DNode[]): AssertionTemplateResult;
replaceChildren(selector: string, children: () => DNode[]): AssertionTemplateResult;
setChildren(selector: string, children: () => DNode[], type?: 'prepend' | 'replace' | 'append'): AssertionTemplateResult;
setProperty(selector: string, property: string, value: any): AssertionTemplateResult;
setProperties(selector: string, value: any | PropertiesComparatorFunction): AssertionTemplateResult;
getChildren(selector: string): DNode[];
getProperty(selector: string, property: string): any;
getProperties(selector: string): any;
replace(selector: string, node: DNode): AssertionTemplateResult;
remove(selector: string): AssertionTemplateResult;
您可能已經(jīng)注意到,在測試部件時,我們主要測試對屬性進行各種修改后,用戶界面是否正確渲染。它們不包含真正的業(yè)務邏輯,但您可能想測試例如單擊按鈕后是否調用了屬性方法。這個測試不關心方法實際做了什么,只關心是否正確調用了接口。在這種情況下,您可以使用類似 Sinon 庫。
src/widgets/Action.ts
import WidgetBase from '@dojo/framework/widget-core/WidgetBase';
import { v, w } from '@dojo/framework/widget-core/d';
import Button from '@dojo/widgets/button';
import * as css from './styles/Action.m.css';
export default class Action extends WidgetBase<{ fetchItems: () => void }> {
protected render() {
return v('div', { classes: [css.root] }, [w(Button, { onClick: this.handleClick, key: 'button' }, ['Fetch'])]);
}
private handleClick() {
this.properties.fetchItems();
}
}
您可能想測試,當單擊按鈕后是否會調用 this.properties.fetchItems
方法。
tests/unit/widgets/Action.ts
const { describe, it } = intern.getInterface('bdd');
import harness from '@dojo/framework/testing/harness';
import { w, v } from '@dojo/framework/widget-core/d';
import { stub } from 'sinon';
describe('Action', () => {
const fetchItems = stub();
it('can fetch data on button click', () => {
const h = harness(() => w(Home, { fetchItems }));
h.expect(() => v('div', { classes: [css.root] }, [w(Button, { onClick: () => {}, key: 'button' }, ['Fetch'])]));
h.trigger('@button', 'onClick');
assert.isTrue(fetchItems.calledOnce);
});
});
在這種情況下,你可以 mock 一個 Action 部件的 fetchItems
方法,該方法將嘗試獲取數(shù)據(jù)項。然后就可以使用 @button
定位到按鈕,并觸發(fā)按鈕的 onClick
事件,然后校驗 fetchItems
方法是否被調用過一次。
要了解更多 mocking 信息,請閱讀 Sinon 文檔。
與單元測試加載和執(zhí)行代碼的流程不同,功能測試在瀏覽器中加載一個頁面并測試應用程序的交互功能。
如果要校驗某個路由對應的頁面內容,可以通過更新鏈接來簡化測試。
src/widgets/Menu.ts
import WidgetBase from '@dojo/framework/widget-core/WidgetBase';
import { w } from '@dojo/framework/widget-core/d';
import Link from '@dojo/framework/routing/ActiveLink';
import Toolbar from '@dojo/widgets/toolbar';
import * as css from './styles/Menu.m.css';
export default class Menu extends WidgetBase {
protected render() {
return w(Toolbar, { heading: 'My Dojo App!', collapseWidth: 600 }, [
w(
Link,
{
id: 'home', // add id attribute
to: 'home',
classes: [css.link],
activeClasses: [css.selected]
},
['Home']
),
w(
Link,
{
id: 'about', // add id attribute
to: 'about',
classes: [css.link],
activeClasses: [css.selected]
},
['About']
),
w(
Link,
{
id: 'profile', // add id attribute
to: 'profile',
classes: [css.link],
activeClasses: [css.selected]
},
['Profile']
)
]);
}
}
在使用應用程序時,您需要單擊 profile
鏈接,然后被導航到歡迎用戶頁面。你可以編寫一個功能測試來驗證此行為。
tests/functional/main.ts
const { describe, it } = intern.getInterface('bdd');
const { assert } = intern.getPlugin('chai');
describe('routing', () => {
it('profile page correctly loads', ({ remote }) => {
return (
remote
// loads the HTML file in local node server
.get('../../output/dev/index.html')
// find the id of the anchor tag
.findById('profile')
// click on the link
.click()
// end this action
.end()
// find the h2 tag
.findByTagName('h2')
// get the text in the h2 tag
.getVisibleText()
.then((text) => {
// verify the content of the h2 tag on the profile page
assert.equal(text, 'Welcome Dojo User!');
})
);
});
});
當運行功能測試時,Dojo 會提供一個與頁面交互的 remote
對象。因為加載頁面和與頁面交互是異步操作,所以必須在測試中返回 remote
對象。
可在命令行中執(zhí)行功能測試。
命令行
npm run test:functional
這會將 HTML 頁面加載到您計算機中 Chrome 的 remote 實例中,以測試交互功能。
功能測試是非常有用的,它能確保在瀏覽器中,您的程序代碼能按預期正常工作。
您可以閱讀更多關于 Intern 功能測試 的內容。