To activate typing (JSDoc), which will help you configure the plugin correctly, I recommend using the Visual Studio Code IDE editor.
npm install --save-dev jjplugin
npx jjPluginBuild
GitLab / GitHub topic required to make the plugin visible for JJAssisntant: jjplugin
GitLab / GitHub topic mark of the tested plugin in development available only in debug mode: dev
The latest plugins will be installed and the console log will be cleaned after restarting the application.
src/index.js (example and description):
module.exports = addPlugin(
{
// Pluginu config - the more universal you choose the key names, the less likely it is to annoy
// the user by inserting duplicate values across other plugins ("facebook", "login", "password").
// Any sensitive data (e.g. passwords) must be stored by plugins only manually through this configuration,
// and must not be sent to third parties, and if they do not offer the service themselves,
// not even to the authors of the plugin.
serviceName: {
propertyArray: [{ // user can add and delete items
name: { type: 'string' }, // required property for an array element
propertyWithoutValue: { type: 'string' }, // user will be prompted to fill in the value
propertyWithValue: { type: 'string', value: 'aaa' }, // default value
}],
stringProp: { type: 'string', value: 'test', pattern: '[a-z]+' },
optionMultyProp: { type: 'string', value: ['aaa', 'ccc'], options: ['aaa', 'bbb', 'ccc'] },
optionSingleProp: { type: 'string', value: 'aaa', options: ['aaa', 'bbb', 'ccc'] },
optionWithPreview: { type: 'string', value: 'aaa', options: {aaa: 'Pretty Aaa', bbb: 'Pretty Bbb'} },
optionVariableProp: { type: 'string', value: [], options: 'myOptions' },
myOptions: { type: 'optionsList', options: ['xxx', 'yyy'] }, // user can editing options values
numberProp: { type: 'number', value: 50, min: 0, max: 100, step: 10, desc: '[%]' },
booleanProp: { type: 'boolean', value: true },
button: { type: 'button', functionName: 'setStatus', parameters: ['aaa', 123] }, // function from scriptInitializer
link: { type: 'link', value: 'https://www.google.com' },
},
},
{ // specify the supported OS and CPU
os: { linux: true, darwin: true, win32: false, android: true },
pluginFormatVersion: 1,
},
{
// scriptInitializer() is run when the application starts before the plugin starts,
// and returns methods (implementing interfaceForAI.js types) that ChatGPT can use
scriptInitializer: async context => {
return new FacebookChat({...context, browserTab: await context.browserTabStart('https://facebook.com/messages/t')});
},
translations: /** @type { const } */ ({
// The English version is mandatory. Other languages are automatically translated from it.
hello: {'en-US': 'Hello ${name}!'} // context.translate.hello({name: 'Peter'})
bay: {'en-US': 'Bay'} // context.translate.bay
}),
},
{
// specification of programs or operations necessary for the plugin to run
moduleRequirementsPayed,
moduleRequirementsFree: [{name: 'SMS app',
linux: {
checkShell: 'npm list | grep SMS@',
installShell: 'npm install SMS'
},
android: {
packageName: 'jjplugin.obsgrass.sms',
minVersion: '1.2.0',
downloadUrl: 'https://github.com/ObscurusGrassator/jjplugin-sms/releases/download/1.2.0/JJPluginSMS_v1.2.0.apk'
}
}],
// other optional funcions
scriptAfterAwaking, scriptAfterAsleep,
scriptAfterBrowserBackButton, scriptAfterDeviceDisplayOffChange,
scriptAfterBrowserResolutionChange: async (context, displayWidth, displayHeight, displayStatusbarHeight) => {},
scriptDestructor: async context => {
await context.methodsForAI.logout();
context.methodsForAI.options.browserTab.destructor();
},
}
);
context
that enters functions:
/**
* @typedef { {
* config: Config,
* configSave: () => Promise<void>,
* methodsForAI: ReturnType<scriptInitializer>,
* translate: {[key in keyof translations]: translations[key][language]
* | (templateArgs: {[k: string]: string}) => translations[key][language]},
* getSummaryAccept: (commandForAccept: string) => Promise<boolean>,
* speech: (text: string, listenReply: boolean, options: {speakDisable?: boolean})
* => Promise<{ text: string, bool: Boolean }>,
* browserTabStart: (url: string, adaptableResolution: boolean, onlyInBackground: boolean)
* => Promise<BrowserPuppeteer>,
* mobileAppOpen: (
* packageName: string,
* serviceName: `${string}Service`,
* permissionsRequestActivity: `${string}Activity` | undefined,
* inputs: [string, string | number | boolean][],
* options: { timeOutSec: number }
* ) => Promise,
* } } Ctx
*/
BrowserPuppeteer -> processComunication.js
src/interfaceForAI.js
This is a mandatory file, containing the types and interface methods that ChatGPT can use. In order for ChatGPT to be able to use them, they must be sufficiently intuitive and documented via JSDoc.
module.exports = class InterfaceForAI {
/**
* @param { string } smsNumber
* @param { string } message
* @returns { Promise<void> }
*/
async sendMessage(smsNumber, message) {}
// this "myOptions" typedef will be replaced at runtime ( example by { 'xxx' | 'yyy' } )
// according to the definition in plugin config ( addPlugin({... myOptions: { type: 'optionsList', ...})
/** @typedef { string } myOptions */
/**
* @param { myOptions } status )
* @returns { Promise<void> }
*/
async setStatus(status) {}
...
methods implementation e.g. in the class
/** @typedef { import('./interfaceForAI.js') } InterfaceForAI */
/** @implements { InterfaceForAI } */
module.exports = class FacebookChat {
constructor(options) {
/**
* @type { { browserTab: import('jjplugin').BrowserPuppeteer }
* & import('jjplugin').Ctx<import('jjplugin').ConfigFrom<typeof import('./index')['config']>, FacebookChat>
* }
*/
this.options = options;
}
async sendMessage(smsNumber, message) {
message = message.replace(/ __? /g, ' ');
// Mandatory for all operations performing any change !!
if (await this.options.getSummaryAccept(`FacebookChat plugin: Can I send a message to a number: ${smsNumber} with text: ${message}?`)) {
... implementácia
await this.options.speech('Sent.');
} else {
await this.options.speech('The command has been cancelled.');
}
}
async setStatus(status) {
this.options.config.facebook.optionVariableProp.value = stauts;
this.options.configSave();
await this.options.speech('Status set to: ' + status);
}
...
POZOR: getSummaryAccept(summary) Do not forget to ask the user for additional consent for each command performing any modification by summarizing the individual details of his request, so that the user can make sure that the system has correctly recognized his request before editing, because some modifications can mean mental or even financial inconvenience for individual users.
POZOR: browserTabStart(url) The browser consumes too much battery, so it closes automatically on the phone if the user does not work with it for 30 seconds. I recommend minimizing the use of the browser and closing it after finishing work with it.
https://github.com/ObscurusGrassator/jjplugin-facebook-chat
https://github.com/ObscurusGrassator/jjplugin-sms
Example of JavaScript plugin communication with an installed Java background service of Android application:
context.mobileAppOpen('jjplugin.obsgrass.sms', 'JJPluginSMSService', 'MainActivity', [["paramA", paramA], ["paramB", paramB]]);
If the application requires some permissions to run, create an activity to request these permissions. Otherwise, the third parameter in context.mobileAppOpen() is optional.
If you want to read the logs of your plugin in JJAssistent in debug mode, set the sending of logs via intent.
You can send arbitrary user String extras arguments to the service via a two-dimensional array. In addition to these, the system arguments "intentFilterBroadcastString" and the unique "requestID" are also sent, thanks to which the intent response is correctly matched. The response must contain "requestID" and either "result" or "error":
import android.app.Service;
import android.content.Intent;
import android.content.ComponentName;
public class JJPluginSMSService extends Service {
private static String oldRequestID = "";
public int onStartCommand(Intent intent, int flags, int startId) {
Bundle extras = intent.getExtras();
// Sometimes the service is called multiple times at the same time with the same parameters.
if (extras.getString("requestID").equals(oldRequestID)) {
return Service.START_REDELIVER_INTENT;
} else oldRequestID = extras.getString("requestID");
// Sending plugin logs to JJAssistant
Boolean loging = true;
new Thread(() -> {
try {
Runtime.getRuntime().exec("logcat -c"); // remove history
Process process = Runtime.getRuntime().exec("logcat");
BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(process.getInputStream()));
String line;
while ((line = bufferedReader.readLine()) != null) {
if (!loging) break;
Intent intent2 = new Intent(extras.getString("intentFilterBroadcastString"));
JSONObject result = new JSONObject();
result.put("level", "");
result.put("tag", "");
result.put("text", line);
intent2.putExtra("requestID", "logCat");
intent2.putExtra("result", result.toString());
sendBroadcast(intent2);
}
}
catch (Exception e) {
Log.e("MainActivity-logcatRead", "Error: " + e.getMessage());
}
}).start();
// ... your result/error logic
// Sending result/error to JJAssistant and exit process
Intent intent = new Intent(extras.getString("intentFilterBroadcastString"));
intent.putExtra("requestID", extras.getString("requestID"));
if (error == null)
intent.putExtra("result", result);
else intent.putExtra("error", error);
sendBroadcast(intent);
loging = false;
stopSelf();
onDestroy();
}
Complete example of Android service: jjplugin-sms
WARNING: Sometimes the service is called multiple times at the same time with the same parameters.
A working background service is here, for example: https://github.com/ObscurusGrassator/jjplugin-sms/blob/main/android-apk-source/app/src/main/java/jjplugin/obsgrass/sms/JJPluginSMSService.java
You can deactivate the activation of the activity (if none exists) by editing the MODE
option value in .idea/workspace.xml
file to <option name="MODE" name="do_nothing"
.