Git Product home page Git Product logo

fm-xml2json's Introduction


Logo

fm-xml2json

An XML Parser that converts XML into JSON on the FileMaker Pro Platform.
Explore the docs »

Report Bug · Request Feature

FileMaker Version FileMaker Platform Commit Shield Contributors MIT License Facebook LinkedId

Table of Contents

About The Project

Logo

As we delved into the world of REST APIs using FileMaker, we stumbled across an issue.

How do we handle XML responses?

When the REST response is a JSON object, we don't have any issue and can make use of the JSON functions introduced in FileMaker 16 to manipulate that response as we need.

However, we found out during our development of fmapi-aws-s3 that REST APIs can respond in XML.

This was an issue and with any issue, solutions are born.

We present fm-xml2json.

A FileMaker script which when passed a valid XML string, will covert this string into a JSON object.

Built With

Features

  • Maintain Order of Elements: Most parsers will convert <a/><b/><a/> to {a:[{},{}],b:{}} which merges any node of same name into an array. This script can create the following to preserve the order of elements: {"elements":[{"type":"element","name":"a"},{"type":"element","name":"b"},{"type":"element","name":"a"}]}.

Read also Compact vs Non-Compact for more info.

  • Fully XML Compliant: Can parse: elements, attributes, texts, comments, CDATA, DOCTYPE, XML declarations, and Processing Instructions.

  • Minimal Dependencies: This script depends only on 2 custom functions and no plugins.

  • Change Property Key Name: Usually output of XML attributes are stored in @attr, _atrr, $attr or $ in order to avoid conflicting with name of sub-elements. This library store them in attributes, but most importantly, you can change this to whatever you like.

Compact vs Non-Compact

Most XML to JSON converters (including online converters) convert <a/> to some compact output like {"a":{}} instead of non-compact output like {"elements":[{"type":"element","name":"a"}]}.

While compact output might work in most situations, there are cases when elements of different names are mixed inside a parent element. Lets use <a x="1"/><b x="2"/><a x="3"/> as an example. Most converters will produce compact output like this {a:[{_:{x:"1"}},{_:{x:"3"}}], b:{_:{x:"2"}}}, which has merged both <a> elements into an array. If you try to convert this back to xml, you will get <a x="1"/><a x="3"/><b x="2"/> which has not preserved the order of elements!

The reason behind this behavior is due to the inherent limitation in the compact representation. Because output like {a:{_:{x:"1"}}, b:{_:{x:"2"}}, a:{_:{x:"3"}}} is illegal (same property name a should not appear twice in an object). This leaves no option but to use array {a:[{_:{x:"1"}},{_:{x:"3"}}].

The non-compact output, which is supported by this script, will produce more information and always guarantees the order of the elements as they appeared in the XML file.

Another drawback of compact output is the resultant element can be an object or an array and therefore makes the client code a little awkward in terms of the extra check needed on object type before processing.

NOTE: Although non-compact output is more accurate representation of original XML than compact version, the non-compact version is verbose and consumes more space. This script provides both options. Use # ( "compact": False ) if you are not sure because it preserves everything; otherwise use # ( "compact": True ) if you want to save space and you don't care about mixing elements of same name and losing their order.

Tip: You can reduce the output size by using shorter key names.

Getting Started

Prerequisites

Version

The FileMaker script makes use of the JSON Functions introduced in FileMaker 16.

This means that this script will only work with FileMaker 16+ products.

Using the script with anything less than 16 will have unexpected behaviour.

Custom Functions

We make use of the #Name-Value custom function provided by filemakerstandards.

Please copy the following Custom Functions to your solution before copying the script:

Limitations

Large XML Data

FileMaker Pro is not the most efficient tool for text parsing. This script was designed primarily for small XML packets sent back from APIs. It wasn't designed, in mind, to convert large XML files into JSON.

However, if given enough time, this script will convert large XML files into JSON and has been optimised to the best of our ability.

Unfortunately, FileMaker can only go so far before it starts to fall over.

We've performed some benchmark tests to give you an idea of how long it takes the script to execute.

XML Size (KB) Execution Time (ms) Readable Time
4 958 0.9s
8 1644 1.6s
17 3430 3.4s
35 8767 8.7s
71 26435 26.4s
142 87973 1min 27.9s
283 314285 5mins 14.3s

KB was determined using FileMaker's Length ( field ) function.

Benchmark tests where performed using the #( "compact" ; True ) option and run as a server script with a 4Ghz Quad core CPU. Your experience may vary depending on if you run the script over WAN, LAN or server architecture

As stated above, this script wasn't developed with the purpose of taking large XML data and converting it to JSON.

If you do have this need, we strongly suggest using a plugin or performing the conversion outside of FileMaker and then importing the result back into FileMaker.

FileMaker Text Parsing Functions

We have built the rules for identifying XML nodes in FileMaker using all the Text Functions available to us and from our testing it has handled all the testing XML data thrown at it.

However, there may be some fringe cases where this breaks down. If you discover such a case, please create a Bug Report and we'll see if we can correct it. Unfortunately, there may be cases that can't be solved.

Usage

Installation

  1. Make sure that the Custom Functions have been added to your solution.
  2. Copy the fm-xml2json script to your solution.

Quick Start

There are 2 fmp12 files provided here

  1. fm-xml2json.fmp12
  2. fm-xml2json-tests.fmp12

fm-xml2json.fmp12 file contains the script and custom function that you need to copy over.

fm-xml2json-tests.fmp12 file contains our test suite to confirm that the script behaves as intended.

If you open the fm-xml2json file, you can paste in some xml and convert it to a json object by pressing the Convert button.

Sample Conversions

XML JS/JSON compact JS/JSON non-compact
<a/> {"a":{}} {"elements":[{"name":"a","type":"element"}]}
<a/><b/> {"a":{},"b":{}} {"elements":[{"name":"a","type":"element"},{"name":"b","type":"element"}]}
<a><b/></a> {"a":{"b":{}}} {"elements":[{"elements":[{"name":"b","type":"element"}],"name":"a","type":"element"}]}
<a> Hi </a> {"a":{"_text":" Hi "}} {"elements":[{"elements":[{"text":" Hi ","type":"text"}],"name":"a","type":"element"}]}
<a x="1.234" y="It's"/> {"a":{"_attributes":{"x":"1.234","y":"It's"}}} {"elements":[{"attributes":{"x":"1.234","y":"It's"},"name":"a","type":"element"}]}
<?xml?> {"_declaration":{}} {"declaration":{},"elements":[]}
<?go there?> {"_instruction":{"go":"there"}} {"elements":[{"instruction":"there","name":"go","type":"instruction"}]}
<?xml version="1.0" encoding="utf-8"?> {"_declaration":{"_attributes":{"version":"1.0","encoding":"utf-8"}}} {"declaration":{"attributes":{"encoding":"utf-8","version":"1.0"}},"elements":[]}
<!--Hello, World!--> {"_comment":"Hello, World!"} {"elements":[{"comment":"Hello, World!","type":"comment"}]}
<![CDATA[<foo></bar>]]> {"_cdata":"<foo></bar>"} {"elements":[{"cdata":"<foo></bar>","type":"cdata"}]}

Parameters

The below parameters can be used as # ( name ; value ) parameters for the script.

name Default value Description
compact false Whether to produce detailed object or compact object.
ignore_declaration false Whether to ignore parsing declaration property. That is, no declaration property will be generated.
ignore_instruction false Whether to ignore parsing processing instruction property. That is, no instruction property will be generated.
ignore_attributes false Whether to ignore parsing attributes of elements.That is, no attributes property will be generated.
ignore_comment false Whether to ignore parsing comments of the elements. That is, no comment will be generated.
ignore_cdata false Whether to ignore parsing CDATA of the elements. That is, no cdata will be generated.
ignore_doctype false Whether to ignore parsing Doctype of the elements. That is, no doctype will be generated.
ignore_text false Whether to ignore parsing texts of the elements. That is, no text will be generated.

The below parameters are under consideration but currently NOT in the script.

name Default value Description
nativeType false Whether to attempt converting text of numerals or of boolean values to native type. For example, "123" will be 123 and "true" will be true
nativeTypeAttributes false Whether to attempt converting attributes of numerals or of boolean values to native type. See also nativeType above.
addParent false Whether to add parent property in each element object that points to parent object.

Options for Changing Key Names

To change default key names in the output object, use the following parameters:

name Default value Description
declaration_key "declaration" or "_declaration" Name of the property key which will be used for the declaration. For example, if # ( "declaration_key" ; "$declaration" ) then output of <?xml?> will be {"$declaration":{}} (in compact form)
instruction_key "instruction" or "_instruction" Name of the property key which will be used for the processing instruction. For example, if # ( "instruction_key" ; "$instruction" ) then output of <?go there?> will be {"$instruction":{"go":"there"}} (in compact form)
attributes_key "attributes" or "_attributes" Name of the property key which will be used for the attributes. For example, if # ( "attributes_key" ; "$attributes" ) then output of <a x="hello"/> will be {"a":{$attributes:{"x":"hello"}}} (in compact form)
comment_key "comment" or "_comment" Name of the property key which will be used for the comment. For example, if # ( "comment_key" ; "$comment" ) then output of <!--note--> will be {"$comment":"note"} (in compact form)
cdata_key "cdata" or "_cdata" Name of the property key which will be used for the cdata. For example, if # ( "cdata_key" ; "$cdata" ) then output of <![CDATA[1 is < 2]]> will be {"$cdata":"1 is < 2"} (in compact form)
doctype_key "doctype" or "_doctype" Name of the property key which will be used for the doctype. For example, if # ( "doctype_key" ; "$doctype" ) then output of <!DOCTYPE foo> will be {"$doctype":" foo} (in compact form)
text_key "text" or "_text" Name of the property key which will be used for the text. For example, if # ( "text_key" ; "$text" ) then output of <a>hi</a> will be {"a":{"$text":"Hi"}} (in compact form)

Two default values mean the first is used for non-compact output and the second is for compact output.

TIP: In compact mode, you can further reduce output result by using fewer characters for key names # ( "text_key" ; "_" ) & # ( "attributes_key" ; "$" ) & # ( "comment_key" ; "value" ). This is also applicable to non-compact mode.

TIP: In non-compact mode, you probably want to set # ( "text_key" ; "value" ) & # ( "cdata_key" ; "value" ) & # ( "comment_key" ; "value" ) to make it more consistent and easier for your client code to go through the contents of text, cdata, and comment.

The below parameters are under consideration but currently NOT in the script.

name Default value Description
parent_key "parent" or "_parent" Name of the property key which will be used for the parent. For example, if # ( "parent_key" ; "$parent" ) then output of <a></b></a> will be {"a":{"b":{$parent:_points_to_a}}} (in compact form)
type_key "type" Name of the property key which will be used for the type. For example, if # ( "type_key" ; "$type" ) then output of <a></a> will be {"elements":[{"$type":"element","name":"a"}]} (in non-compact form)
name_key "name" Name of the property key which will be used for the name. For example, if # ( "name_key" ; "$name" ) then output of <a></a> will be {"elements":[{"type":"element","$name":"a"}]} (in non-compact form)
elements_key "elements" Name of the property key which will be used for the elements. For example, if # ( "elements_key" ; "$elements" ) then output of <a></a> will be {"$elements":[{"type":"element","name":"a"}]} (in non-compact form)

Contributing

Contributions are what make the open source community such an amazing place to be learn, inspire, and create. Any contributions you make are greatly appreciated.

  1. Fork the Project
  2. Create your Feature Branch (git checkout -b feature/AmazingFeature)
  3. Commit your Changes (git commit -m 'Add some AmazingFeature)
  4. Push to the Branch (git push origin feature/AmazingFeature)
  5. Open a Pull Request

License

Distributed under the MIT License. See LICENSE for more information.

Contact

Steven McGill - WhiteSpace Systems Ltd - [email protected]

Project Link: fm-xml2json

Acknowledgements

fmapi Product Suite

When FileMaker 16 introduced cURL with the Insert from URL script step. Use cases for FileMaker Pro increased dramatically in relation to REST APIs.

It allowed developers to finally communicate with other web services and APIs without the need for 3rd party plugins. Integration with Couriers, Payment Gateways & Social Media Sites all became within touching distance.

However, without prior knowledge of cURL, HTTP Request Methods, HTTP Headers, JSON, OAuth Authentication, API Keys, API documentation etc, it can be extremely difficult to get started for the novice user or even the most proficient FileMaker developer.

And with that comes our goal, to simplify communications between your FileMaker App and the vast amount of Web Services available, no matter your ability level.

Read more over at What is fmapi?

fmapi Apps

A collection of FileMaker apps that communicate directly with popular 3rd party REST APIs.

  • fmapi-vies-vat - Integrate the EU Commissions Vies VAT API directly into you FileMaker App allowing you to validate EU VAT Numbers.

fm-xml2json's People

Contributors

stevenwhitespacesystems avatar

Watchers

 avatar

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    🖖 Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. 📊📈🎉

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google ❤️ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.