Skip to content

Commit d7f4b91

Browse files
[BUGFIX] Improvement of adapter.findForeignObject (#3020)
* Improvement of adapter.findForeignObject * Formatting * Fixing tests * Fixing typing * Added positive tests * Added positive tests
1 parent bc4ae0b commit d7f4b91

File tree

4 files changed

+78
-44
lines changed

4 files changed

+78
-44
lines changed

packages/adapter/src/lib/adapter/adapter.ts

+40-18
Original file line numberDiff line numberDiff line change
@@ -507,12 +507,7 @@ export interface AdapterClass {
507507
): ioBroker.SetObjectPromise;
508508

509509
/**
510-
* Finds an object by its ID or name
511-
*/
512-
findForeignObjectAsync(idOrName: string, type: string): Promise<{ id: string; name: string }>;
513-
514-
/**
515-
* Creates an object with type channel. It must be located under a device
510+
* Creates an object with a type channel. It must be located under a device
516511
*
517512
* @deprecated use `extendObject` instead
518513
*/
@@ -1022,11 +1017,6 @@ export class AdapterClass extends EventEmitter {
10221017
*/
10231018
this.getForeignObjectsAsync = tools.promisify(this.getForeignObjects, this);
10241019

1025-
/**
1026-
* Promise-version of `Adapter.findForeignObject`
1027-
*/
1028-
this.findForeignObjectAsync = tools.promisify(this.findForeignObject, this, ['id', 'name']);
1029-
10301020
/**
10311021
* Promise-version of `Adapter.getForeignObject`
10321022
*/
@@ -4380,7 +4370,7 @@ export class AdapterClass extends EventEmitter {
43804370
}
43814371
}
43824372

4383-
// don't forget, that enums returns names in row[x].id and not IDs, you can find id in rows[x].value._id
4373+
// don't forget that enums returns names in row[x].id and not IDs, you can find id in rows[x].value._id
43844374
let _enums;
43854375
try {
43864376
_enums = await this.getEnumsAsync(enums);
@@ -4449,21 +4439,21 @@ export class AdapterClass extends EventEmitter {
44494439
}
44504440

44514441
// external signature
4452-
findForeignObject(idOrName: string, type: string | null, callback: ioBroker.FindObjectCallback): void;
4442+
findForeignObject(idOrName: string, type: ioBroker.CommonType | null, callback: ioBroker.FindObjectCallback): void;
44534443
findForeignObject(
44544444
idOrName: string,
4455-
type: string | null,
4456-
options: unknown,
4445+
type: ioBroker.CommonType | null,
4446+
options: { user?: `system.user.${string}`; language?: ioBroker.Languages },
44574447
callback: ioBroker.FindObjectCallback,
44584448
): void;
44594449

44604450
/**
44614451
* Find any object by name or ID.
44624452
*
4463-
* Find object by the exact name or ID.
4453+
* Find an object by the exact name or ID.
44644454
*
44654455
* @param id exactly object ID (without namespace)
4466-
* @param type optional common.type of state: 'number', 'string', 'boolean', 'file', ...
4456+
* @param type optional `common.type` of state: 'number', 'string', 'boolean', 'file', ...
44674457
* @param options optional user context
44684458
* @param callback return result
44694459
* ```js
@@ -4504,7 +4494,39 @@ export class AdapterClass extends EventEmitter {
45044494
return tools.maybeCallbackWithError(callback, e);
45054495
}
45064496

4507-
this.#objects.findObject(id, type, options || {}, callback);
4497+
this.#objects.findObject(id, type as ioBroker.CommonType | null, options || {}, callback);
4498+
}
4499+
4500+
/**
4501+
* Find an object by the exact name or ID.
4502+
*
4503+
* @param id exactly object ID (without a namespace)
4504+
* @param type optional `common.type` of the state: 'number', 'string', 'boolean', 'file', ...
4505+
* @param options optional user context with language
4506+
* @param options.user current user
4507+
* @param options.language language in which the search must be done for multi-language names
4508+
* @result if the object was found by ID it will return id and may be the multi-language name it exists. If the object was found by name it will return id and the multi-language name. If the object was not found, it will return only name that was searched for.
4509+
*/
4510+
findForeignObjectAsync(
4511+
id: string,
4512+
type: ioBroker.CommonType | null,
4513+
options?: { user?: `system.user.${string}`; language?: ioBroker.Languages },
4514+
): Promise<{ id?: string; name: ioBroker.StringOrTranslated }> {
4515+
return new Promise<{ id?: string; name: ioBroker.StringOrTranslated }>((resolve, reject) =>
4516+
this.findForeignObject(
4517+
id,
4518+
type,
4519+
options || {},
4520+
(err, id?: string, name?: ioBroker.StringOrTranslated): void => {
4521+
if (err) {
4522+
reject(err);
4523+
} else {
4524+
// if not error, name cannot be undefined
4525+
resolve(id ? { id, name: name! } : { name: name! });
4526+
}
4527+
},
4528+
),
4529+
);
45084530
}
45094531

45104532
// external signatures

packages/controller/test/lib/testObjects.ts

+13-3
Original file line numberDiff line numberDiff line change
@@ -99,11 +99,17 @@ export function register(it: Mocha.TestFunction, expect: Chai.ExpectStatic, cont
9999
expect(idOrName).to.be.equal('test3');
100100
expect(id).to.be.equal(undefined);
101101

102-
objects.findObject('test2', 'channel', (err, id, idOrName) => {
102+
objects.findObject('test2', 'boolean', (err, id, idOrName) => {
103103
expect(err).to.be.not.ok;
104104
expect(idOrName).to.be.equal('test2');
105105
expect(id).to.be.equal(undefined);
106-
done();
106+
107+
objects.findObject('test2', 'string', (err, id, idOrName) => {
108+
expect(err).to.be.not.ok;
109+
expect(idOrName).to.be.equal('test2');
110+
expect(id).to.be.equal(testId);
111+
done();
112+
});
107113
});
108114
});
109115
});
@@ -125,10 +131,14 @@ export function register(it: Mocha.TestFunction, expect: Chai.ExpectStatic, cont
125131
})
126132
.then(id => {
127133
expect(id).to.be.equal(undefined);
128-
return objects.findObject('test3', 'channel');
134+
return objects.findObject('test3', 'boolean');
129135
})
130136
.then(id => {
131137
expect(id).to.be.equal(undefined);
138+
return objects.findObject('test2', 'string');
139+
})
140+
.then(id => {
141+
expect(id).to.be.equal(testId);
132142
done();
133143
})
134144
.catch(err => {

packages/controller/test/lib/testObjectsFunctions.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -312,7 +312,7 @@ export function register(it: Mocha.TestFunction, expect: Chai.ExpectStatic, cont
312312
expect(err).to.be.null;
313313
expect(id).to.be.equal(`${context.adapterShortName}.0.${gid}`);
314314

315-
context.adapter.findForeignObject('test1', 'channel', function (err, id) {
315+
context.adapter.findForeignObject('test1', 'boolean', function (err, id) {
316316
expect(err).to.be.null;
317317

318318
expect(id).to.be.undefined;

packages/db-objects-redis/src/lib/objects/objectsInRedisClient.ts

+24-22
Original file line numberDiff line numberDiff line change
@@ -4726,19 +4726,18 @@ export class ObjectsInRedisClient {
47264726
*/
47274727
private _findObject(
47284728
idOrName: string,
4729-
type: string | null,
4730-
options: CallOptions,
4729+
type: ioBroker.CommonType | null,
4730+
options: CallOptions & { language?: ioBroker.Languages },
47314731
callback?: ioBroker.FindObjectCallback,
47324732
): void {
4733+
// Try to read by ID
47334734
this._getObject(idOrName, options, (err, obj) => {
47344735
// Assume it is ID
4735-
if (
4736-
obj &&
4737-
utils.checkObject(obj, options, CONSTS.ACCESS_READ) &&
4738-
(!type || (obj.common && obj.common.type === type))
4739-
) {
4736+
if (obj && utils.checkObject(obj, options, CONSTS.ACCESS_READ) && (!type || obj.common?.type === type)) {
47404737
return tools.maybeCallbackWithError(callback, null, idOrName, obj.common.name);
47414738
}
4739+
4740+
// Get all objects that this user may read
47424741
this._getKeys(
47434742
'*',
47444743
options,
@@ -4751,14 +4750,14 @@ export class ObjectsInRedisClient {
47514750
return tools.maybeCallbackWithError(callback, err);
47524751
}
47534752

4754-
let objs;
4753+
let objs: (string | null)[];
47554754
try {
47564755
objs = await this.client.mget(keys);
47574756
} catch (e) {
47584757
return tools.maybeCallbackWithRedisError(callback, e);
47594758
}
47604759
objs = objs || [];
4761-
// Assume it is name
4760+
// Assume it is a name
47624761
for (let i = 0; i < keys.length; i++) {
47634762
const strObj = objs[i];
47644763
let obj: ioBroker.AnyObject | null;
@@ -4768,12 +4767,15 @@ export class ObjectsInRedisClient {
47684767
this.log.error(`${this.namespace} Cannot parse JSON ${keys[i]}: ${objs[i]}`);
47694768
continue;
47704769
}
4771-
if (
4772-
obj?.common &&
4773-
obj.common.name === idOrName &&
4774-
(!type || ('type' in obj.common && obj.common.type === type))
4775-
) {
4776-
return tools.maybeCallbackWithError(callback, null, obj._id, idOrName);
4770+
4771+
if (obj?.common && (!type || ('type' in obj.common && obj.common.type === type))) {
4772+
let name = obj?.common?.name;
4773+
if (name && typeof name === 'object') {
4774+
name = name[options.language || 'en'] || name.en;
4775+
}
4776+
if (name === idOrName) {
4777+
return tools.maybeCallbackWithError(callback, null, obj._id, obj.common.name);
4778+
}
47774779
}
47784780
}
47794781
return tools.maybeCallbackWithError(callback, null, undefined, idOrName);
@@ -4786,25 +4788,25 @@ export class ObjectsInRedisClient {
47864788
// The user has provided a callback, thus we call it
47874789
findObject(
47884790
idOrName: string,
4789-
type: string | null,
4790-
options: CallOptions | null,
4791+
type: ioBroker.CommonType | null,
4792+
options: (CallOptions & { language?: ioBroker.Languages }) | null,
47914793
callback: ioBroker.FindObjectCallback,
47924794
): void;
47934795

47944796
// The user has provided a callback without options
4795-
findObject(idOrName: string, type: string | null, callback: ioBroker.FindObjectCallback): void;
4797+
findObject(idOrName: string, type: ioBroker.CommonType | null, callback: ioBroker.FindObjectCallback): void;
47964798

47974799
// No callback provided by user, we return a promise
47984800
findObject(
47994801
idOrName: string,
4800-
type?: string | null,
4801-
options?: CallOptions | null,
4802+
type?: ioBroker.CommonType | null,
4803+
options?: (CallOptions & { language?: ioBroker.Languages }) | null,
48024804
): Promise<ioBroker.CallbackReturnTypeOf<ioBroker.FindObjectCallback>>;
48034805

48044806
findObject(
48054807
idOrName: string,
4806-
type: string | null,
4807-
options: CallOptions | null,
4808+
type: ioBroker.CommonType | null,
4809+
options: (CallOptions & { language?: ioBroker.Languages }) | null,
48084810
callback?: ioBroker.FindObjectCallback,
48094811
): void | Promise<ioBroker.CallbackReturnTypeOf<ioBroker.FindObjectCallback>> {
48104812
if (typeof type === 'function') {

0 commit comments

Comments
 (0)