1 /* 2 * Copyright (C) 2016 The Android Open Source Project 3 * Copyright (C) 2016 Mopria Alliance, Inc. 4 * 5 * Licensed under the Apache License, Version 2.0 (the "License"); 6 * you may not use this file except in compliance with the License. 7 * You may obtain a copy of the License at 8 * 9 * http://www.apache.org/licenses/LICENSE-2.0 10 * 11 * Unless required by applicable law or agreed to in writing, software 12 * distributed under the License is distributed on an "AS IS" BASIS, 13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 14 * See the License for the specific language governing permissions and 15 * limitations under the License. 16 */ 17 18 package com.android.bips.ipp; 19 20 import android.content.Context; 21 import android.content.pm.PackageInfo; 22 import android.content.pm.PackageManager; 23 import android.net.Uri; 24 import android.os.AsyncTask; 25 import android.os.Build; 26 import android.os.Handler; 27 import android.printservice.PrintJob; 28 import android.text.TextUtils; 29 import android.util.Log; 30 31 import com.android.bips.R; 32 import com.android.bips.jni.BackendConstants; 33 import com.android.bips.jni.JobCallback; 34 import com.android.bips.jni.JobCallbackParams; 35 import com.android.bips.jni.LocalJobParams; 36 import com.android.bips.jni.LocalPrinterCapabilities; 37 import com.android.bips.jni.PdfRender; 38 import com.android.bips.util.FileUtils; 39 40 import java.io.File; 41 import java.util.Locale; 42 import java.util.function.Consumer; 43 44 public class Backend implements JobCallback { 45 private static final String TAG = Backend.class.getSimpleName(); 46 private static final boolean DEBUG = false; 47 48 static final String TEMP_JOB_FOLDER = "jobs"; 49 50 // Error codes strictly to be in negative number 51 static final int ERROR_FILE = -1; 52 static final int ERROR_CANCEL = -2; 53 static final int ERROR_UNKNOWN = -3; 54 55 private static final String VERSION_UNKNOWN = "(unknown)"; 56 57 private final Handler mMainHandler; 58 private final Context mContext; 59 private JobStatus mCurrentJobStatus; 60 private Consumer<JobStatus> mJobStatusListener; 61 private AsyncTask<Void, Void, Integer> mStartTask; 62 Backend(Context context)63 public Backend(Context context) { 64 if (DEBUG) Log.d(TAG, "Backend()"); 65 66 mContext = context; 67 mMainHandler = new Handler(context.getMainLooper()); 68 PdfRender.getInstance(mContext); 69 70 // Load required JNI libraries 71 System.loadLibrary(BackendConstants.WPRINT_LIBRARY_PREFIX); 72 73 // Create and initialize JNI layer 74 nativeInit(this, context.getApplicationInfo().dataDir, Build.VERSION.SDK_INT); 75 nativeSetSourceInfo(context.getString(R.string.app_name).toLowerCase(Locale.US), 76 getApplicationVersion(context).toLowerCase(Locale.US), 77 BackendConstants.WPRINT_APPLICATION_ID.toLowerCase(Locale.US)); 78 } 79 80 /** Return the current application version or VERSION_UNKNOWN */ getApplicationVersion(Context context)81 private String getApplicationVersion(Context context) { 82 try { 83 PackageInfo packageInfo = context.getPackageManager() 84 .getPackageInfo(context.getPackageName(), 0); 85 return packageInfo.versionName; 86 } catch (PackageManager.NameNotFoundException e) { 87 return VERSION_UNKNOWN; 88 } 89 } 90 91 /** Asynchronously get printer capabilities, returning results or null to a callback */ getCapabilities(Uri uri, long timeout, boolean highPriority, final Consumer<LocalPrinterCapabilities> capabilitiesConsumer)92 public GetCapabilitiesTask getCapabilities(Uri uri, long timeout, boolean highPriority, 93 final Consumer<LocalPrinterCapabilities> capabilitiesConsumer) { 94 if (DEBUG) Log.d(TAG, "getCapabilities()"); 95 96 GetCapabilitiesTask task = new GetCapabilitiesTask(this, uri, timeout, highPriority) { 97 @Override 98 protected void onPostExecute(LocalPrinterCapabilities result) { 99 capabilitiesConsumer.accept(result); 100 } 101 }; 102 task.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); 103 return task; 104 } 105 106 /** 107 * Start a print job. Results will be notified to the listener. Do not start more than 108 * one job at a time. 109 */ print(Uri uri, PrintJob printJob, LocalPrinterCapabilities capabilities, Consumer<JobStatus> listener)110 public void print(Uri uri, PrintJob printJob, LocalPrinterCapabilities capabilities, 111 Consumer<JobStatus> listener) { 112 if (DEBUG) Log.d(TAG, "print()"); 113 114 mJobStatusListener = listener; 115 mCurrentJobStatus = new JobStatus(); 116 117 mStartTask = new StartJobTask(mContext, this, uri, printJob, capabilities) { 118 @Override 119 public void onCancelled(Integer result) { 120 if (DEBUG) Log.d(TAG, "StartJobTask onCancelled " + result); 121 onPostExecute(ERROR_CANCEL); 122 } 123 124 @Override 125 protected void onPostExecute(Integer result) { 126 if (DEBUG) Log.d(TAG, "StartJobTask onPostExecute " + result); 127 mStartTask = null; 128 if (result > 0) { 129 mCurrentJobStatus = new JobStatus.Builder(mCurrentJobStatus).setId(result) 130 .build(); 131 } else if (mJobStatusListener != null) { 132 String jobResult = BackendConstants.JOB_DONE_ERROR; 133 if (result == ERROR_CANCEL) { 134 jobResult = BackendConstants.JOB_DONE_CANCELLED; 135 } else if (result == ERROR_FILE) { 136 jobResult = BackendConstants.JOB_DONE_CORRUPT; 137 } 138 139 // If the start attempt failed and we are still listening, notify and be done 140 mCurrentJobStatus = new JobStatus.Builder() 141 .setJobState(BackendConstants.JOB_STATE_DONE) 142 .setJobResult(jobResult).build(); 143 mJobStatusListener.accept(mCurrentJobStatus); 144 mJobStatusListener = null; 145 } 146 } 147 }; 148 mStartTask.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); 149 } 150 151 /** Attempt to cancel the current job */ cancel()152 public void cancel() { 153 if (DEBUG) Log.d(TAG, "cancel()"); 154 155 if (mStartTask != null) { 156 if (DEBUG) Log.d(TAG, "cancelling start task"); 157 mStartTask.cancel(true); 158 } else if (mCurrentJobStatus != null && mCurrentJobStatus.getId() != JobStatus.ID_UNKNOWN) { 159 if (DEBUG) Log.d(TAG, "cancelling job via new task"); 160 new CancelJobTask(this, mCurrentJobStatus.getId()) 161 .executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); 162 } else { 163 if (DEBUG) Log.d(TAG, "Nothing to cancel in backend, ignoring"); 164 } 165 } 166 167 /** 168 * Call when it is safe to release document-centric resources related to a print job 169 */ closeDocument()170 public void closeDocument() { 171 // Tell the renderer it may release resources for the document 172 PdfRender.getInstance(mContext).closeDocument(); 173 } 174 175 /** 176 * Call when service is shutting down, nothing else is happening, and this object 177 * is no longer required. After closing this object it should be discarded. 178 */ close()179 public void close() { 180 new Thread(this::nativeExit).start(); 181 PdfRender.getInstance(mContext).close(); 182 } 183 184 /** Called by JNI */ 185 @Override jobCallback(final int jobId, final JobCallbackParams params)186 public void jobCallback(final int jobId, final JobCallbackParams params) { 187 mMainHandler.post(() -> { 188 if (DEBUG) Log.d(TAG, "jobCallback() jobId=" + jobId + ", params=" + params); 189 190 JobStatus.Builder builder = new JobStatus.Builder(mCurrentJobStatus); 191 192 builder.setId(params.jobId); 193 194 if (params.certificate != null) { 195 builder.setCertificate(params.certificate); 196 } 197 198 if (!TextUtils.isEmpty(params.printerState)) { 199 updateBlockedReasons(builder, params); 200 } else if (!TextUtils.isEmpty(params.jobState)) { 201 builder.setJobState(params.jobState); 202 if (!TextUtils.isEmpty(params.jobDoneResult)) { 203 builder.setJobResult(params.jobDoneResult); 204 } 205 updateBlockedReasons(builder, params); 206 } 207 mCurrentJobStatus = builder.build(); 208 209 if (mJobStatusListener != null) { 210 mJobStatusListener.accept(mCurrentJobStatus); 211 } 212 213 if (mCurrentJobStatus.isJobDone()) { 214 nativeEndJob(jobId); 215 // Reset status for next job. 216 mCurrentJobStatus = new JobStatus(); 217 mJobStatusListener = null; 218 219 FileUtils.deleteAll(new File(mContext.getFilesDir(), Backend.TEMP_JOB_FOLDER)); 220 } 221 }); 222 } 223 224 /** Update the blocked reason list with non-empty strings */ updateBlockedReasons(JobStatus.Builder builder, JobCallbackParams params)225 private void updateBlockedReasons(JobStatus.Builder builder, JobCallbackParams params) { 226 if ((params.blockedReasons != null) && (params.blockedReasons.length > 0)) { 227 builder.clearBlockedReasons(); 228 for (String reason : params.blockedReasons) { 229 if (!TextUtils.isEmpty(reason)) { 230 builder.addBlockedReason(reason); 231 } 232 } 233 } 234 } 235 236 /** 237 * Extracts the ip portion of x.x.x.x/y/z 238 * 239 * @param address any string in the format xxx/yyy/zzz 240 * @return the part before the "/" or "xxx" in this case 241 */ getIp(String address)242 static String getIp(String address) { 243 int i = address.indexOf('/'); 244 return i == -1 ? address : address.substring(0, i); 245 } 246 247 /** 248 * Initialize the lower layer. 249 * 250 * @param jobCallback job callback to use whenever job updates arrive 251 * @param dataDir directory to use for temporary files 252 * @param apiVersion local system API version to be supplied to printers 253 * @return {@link BackendConstants#STATUS_OK} or an error code. 254 */ nativeInit(JobCallback jobCallback, String dataDir, int apiVersion)255 native int nativeInit(JobCallback jobCallback, String dataDir, int apiVersion); 256 257 /** 258 * Supply additional information about the source of jobs. 259 * 260 * @param appName human-readable name of application providing data to the printer 261 * @param version version of delivering application 262 * @param appId identifier for the delivering application 263 */ nativeSetSourceInfo(String appName, String version, String appId)264 native void nativeSetSourceInfo(String appName, String version, String appId); 265 266 /** 267 * Request capabilities from a printer. 268 * 269 * @param address IP address or hostname (e.g. "192.168.1.2") 270 * @param port port to use (e.g. 631) 271 * @param httpResource path of print resource on host (e.g. "/ipp/print") 272 * @param uriScheme scheme (e.g. "ipp") 273 * @param timeout milliseconds to wait before giving up on request 274 * @param capabilities target object to be filled with printer capabilities, if successful 275 * @return {@link BackendConstants#STATUS_OK} or an error code. 276 */ nativeGetCapabilities(String address, int port, String httpResource, String uriScheme, long timeout, LocalPrinterCapabilities capabilities)277 native int nativeGetCapabilities(String address, int port, String httpResource, 278 String uriScheme, long timeout, LocalPrinterCapabilities capabilities); 279 280 /** 281 * Determine initial parameters to be used for jobs 282 * 283 * @param jobParams object to be filled with default parameters 284 * @return {@link BackendConstants#STATUS_OK} or an error code. 285 */ nativeGetDefaultJobParameters(LocalJobParams jobParams)286 native int nativeGetDefaultJobParameters(LocalJobParams jobParams); 287 288 /** 289 * Update job parameters to align with known printer capabilities 290 * 291 * @param jobParams on input, contains requested job parameters; on output contains final 292 * job parameter selections. 293 * @param capabilities printer capabilities to be used when finalizing job parameters 294 * @return {@link BackendConstants#STATUS_OK} or an error code. 295 */ nativeGetFinalJobParameters(LocalJobParams jobParams, LocalPrinterCapabilities capabilities)296 native int nativeGetFinalJobParameters(LocalJobParams jobParams, 297 LocalPrinterCapabilities capabilities); 298 299 /** 300 * Begin job delivery to a target printer. Updates on the job will be sent to the registered 301 * {@link JobCallback}. 302 * 303 * @param address IP address or hostname (e.g. "192.168.1.2") 304 * @param port port to use (e.g. 631) 305 * @param mimeType MIME type of data being sent 306 * @param jobParams job parameters to use when providing the job to the printer 307 * @param capabilities printer capabilities for the printer being used 308 * @param fileList list of files to be provided of the given MIME type 309 * @param debugDir directory to receive debugging information, if any 310 * @param scheme URI scheme (e.g. ipp/ipps) 311 * @return {@link BackendConstants#STATUS_OK} or an error code. 312 */ nativeStartJob(String address, int port, String mimeType, LocalJobParams jobParams, LocalPrinterCapabilities capabilities, String[] fileList, String debugDir, String scheme)313 native int nativeStartJob(String address, int port, String mimeType, LocalJobParams jobParams, 314 LocalPrinterCapabilities capabilities, String[] fileList, String debugDir, 315 String scheme); 316 317 /** 318 * Request cancellation of the identified job. 319 * 320 * @param jobId identifier of the job to cancel 321 * @return {@link BackendConstants#STATUS_OK} or an error code. 322 */ nativeCancelJob(int jobId)323 native int nativeCancelJob(int jobId); 324 325 /** 326 * Finalizes a job after it is ends for any reason 327 * 328 * @param jobId identifier of the job to end 329 * @return {@link BackendConstants#STATUS_OK} or an error code. 330 */ nativeEndJob(int jobId)331 native int nativeEndJob(int jobId); 332 333 /** 334 * Shut down and clean up resources in the JNI layer on system exit 335 * 336 * @return {@link BackendConstants#STATUS_OK} or an error code. 337 */ nativeExit()338 native int nativeExit(); 339 } 340