API считывателя карт UniMag не будет принимать XML-файл конфигурации во время установки на Xamarin

Я внедряю API слайдера кредитных карт UniMag на Android Xamarin, чтобы мое приложение могло читать кредитные карты. API предназначен только для нативного Android, поэтому я следовал приведенным здесь инструкциям, чтобы создать оболочку для файла API.jar. Это прекрасно работает и очень просто.

Я могу инициализировать API и получать обратные вызовы слушателя при подключении и отключении слайдера. Поскольку API-интерфейс поддерживает не все устройства Android, UniMagReader должен прочитать предоставленный XML-файл конфигурации.

У меня проблема в том, что я не могу получить API для чтения файла.

Вызов API для использования в этом;

var reader = new UniMagReader(new UniMagMessage(), Android.App.Application.Context);
reader.SetSaveLogEnable(false);

reader.SetVerboseLoggingEnable(true);
reader.RegisterListen();

string fileNameWithPath = GetConfigurationFileFromRaw();
reader.SetXMLFileNameWithPath(fileNameWithPath);
reader.LoadingConfigurationXMLFile(false);

Я попытался поместить файл в каталоги Resources.Raw, Resources.Assets и корневые каталоги и попробовал несколько способов предоставления URL-адреса API.

Это способ, которым демонстрационное приложение предлагает:

  private string GetXMLFileFromRaw(string fileName)
    {
        //the target filename in the application path
        string fileNameWithPath = null;
        fileNameWithPath = fileName;

        try
        {
            //using (var inStream = Android.App.Application.Context.Assets.Open("default_config.xml")) // assest file access
            using (var inStream = Android.App.Application.Context.Resources.OpenRawResource(Resource.Raw.default_config)) // raw file access
            {
                var length = GetStreamLength(inStream);
                byte[] buffer = new byte[length];
                inStream.Read(buffer, 0, (int)length);
                inStream.Close();
                Android.App.Application.Context.DeleteFile(fileNameWithPath);
                var fout = Android.App.Application.Context.OpenFileOutput(fileNameWithPath, Android.Content.FileCreationMode.Private);
                fout.Write(buffer, 0, (int)length);
                fout.Close();

                // to refer to the application path
                var fileDir = Android.App.Application.Context.FilesDir;
                fileNameWithPath = fileDir.Parent + Path.DirectorySeparatorChar + fileDir.Name;
                fileNameWithPath += Path.DirectorySeparatorChar + "default_config.xml";
            }
        }
        catch (System.Exception e)
        {
            fileNameWithPath = null;
        }
        return fileNameWithPath;
    }

    // Return the length of a stream that does not have a usable Length property
    public static long GetStreamLength(Stream stream)
    {
        long originalPosition = 0;
        long totalBytesRead = 0;

        if (stream.CanSeek)
        {
            originalPosition = stream.Position;
            stream.Position = 0;
        }

        try
        {
            byte[] readBuffer = new byte[4096];

            int bytesRead;

            while ((bytesRead = stream.Read(readBuffer, 0, 4096)) > 0)
            {
                totalBytesRead += bytesRead;
            }
        }
        finally
        {
            if (stream.CanSeek)
            {
                stream.Position = originalPosition;
            }
        }

        return totalBytesRead;
    }

И это еще один способ, которым я пытался:

var file = new Java.IO.File(Android.Net.Uri.Parse("file:///default_config.xml").ToString());
var uri = file.AbsolutePath;

Я получаю ошибки от API. При подключении swiper к устройству вижу:

 [UMSDK] SDK: headset attached
 [UMSDK] SDK: reader attached, but no config loaded

Из обратного вызова OnReceiveMsgFailureInfo я вижу "XML-файл не существует и автоматическое обновление отключено".

И в выводе приложения я вижу:

[UMSDK] UmXmlParser: parsing XML failed due to exception
[UMSDK] org.apache.harmony.xml.ExpatParser$ParseException: At line 1, column 0: not well-formed (invalid token)
[UMSDK]     at org.apache.harmony.xml.ExpatParser.parseFragment(ExpatParser.java:519)
[UMSDK]     at org.apache.harmony.xml.ExpatParser.parseDocument(ExpatParser.java:478)
[UMSDK]     at org.apache.harmony.xml.ExpatReader.parse(ExpatReader.java:316)
[UMSDK]     at org.apache.harmony.xml.ExpatReader.parse(ExpatReader.java:279)
[UMSDK]     at com.idtechproducts.acom.AcomXmlParser.parseFile(AcomXmlParser.java:91)
[UMSDK]     at com.idtechproducts.unimagsdk.UniMagConfigHelper.loadingXMLFile(UniMagConfigHelper.java:116)
[UMSDK]     at com.idtechproducts.unimagsdk.UniMagConfigHelper.loadingXMLFile(UniMagConfigHelper.java:46)
[UMSDK]     at IDTech.MSR.uniMag.uniMagReader.loadingConfigurationXMLFile(uniMagReader.java:496)

1 ответ

Решение

В этом примере файл находится в папке "Ресурсы" или "Сырье". Чтобы использовать файл, находящийся в папках "Активы" или "Сырье", необходимо получить поток, а затем записать этот поток в файл, который доступен за пределами самого приложения (любой файл, добавленный в проект приложения в среде IDE, в конечном итоге упаковывается в сам APK не доступен никому, кроме самого APK). Пример Java, который вы отправили, делает именно это (скопированный из примера в вашем посте, но сокращенный, чтобы показать только определенные строки):

private string GetXMLFileFromRaw(string fileName)
{
...
    try
    {
        // Gets the stream from the raw resource
        using (var inStream = Android.App.Application.Context.Resources.OpenRawResource(Resource.Raw.default_config)) // raw file access
        {
            // Reads the stream into a buffer
            ...
            inStream.Read(buffer, 0, (int)length);
            ...
            // Deletes any existing file
            Android.App.Application.Context.DeleteFile(fileNameWithPath);
            // Writes the stream to a file
            ...
            fout.Write(buffer, 0, (int)length);
            ...
        }
    }
    ...
}

Итак, сначала настройте имя файла и переменные пути:

string filename = "default_config.xml";
string directory = Android.App.Application.Context.FilesDir.AbsolutePath;
string fullPath = Path.Combine(directory, filename);
string xmlFileContents;

Тогда получите поток.

Если файл default_config.xml в ресурсах:

using (StreamReader sr = new StreamReader(Assets.Open(filename)))

Если default_config.xml в Необработанных ресурсах:

using (StreamReader sr = new StreamReader(Resources.OpenRawResource(Resource.Raw.default_config))

Затем сделайте следующее:

{
    xmlFileContents = sr.ReadToEnd();
    if (File.Exists(fullPath)) {
        File.Delete(fullPath);
    }
    File.WriteAllText(fullPath, xmlFileContents);
}

Тогда вы можете пройти fullPath к API:

reader.SetXMLFileNameWithPath(fullPath);
Другие вопросы по тегам