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);