diff --git a/CHANGELOG.rst b/CHANGELOG.rst
index cd9171fa595e950a6406d1d4f7b6f57df159192d..61c17cb7b98766253bcc7108a04d10c4fedea174 100644
--- a/CHANGELOG.rst
+++ b/CHANGELOG.rst
@@ -1,5 +1,15 @@
 Changelog
 =========
+0.24.2
+------
+* New UnknownTransaction exception that is raised when using get_transaction with an unkown trx_id
+* New function is_transaction_existing which returns false, when a trx_id does not exists
+* beempy info does not show information for a trx_id
+* broadcast from TransactionBuilder can now return a trx_id, when set trx_id to True (default)
+* sign and finalizeOp from Hive and Steem return now the trx_id in a field
+* add export parameter to all broadcast commands in beempy
+* When setting unsigned in beempy, the default value of expires is changed to 3600
+
 0.24.1
 ------
 * fixed missing module in setup.py
diff --git a/beem/blockchain.py b/beem/blockchain.py
index 9781fb2b30748c4cd6a1ebf69a0b5a423c03fa01..22fe1776ab11ecd6663cc1ecf45934e3f6f29316 100644
--- a/beem/blockchain.py
+++ b/beem/blockchain.py
@@ -20,7 +20,7 @@ from .utils import formatTimeString, addTzInfo
 from .block import Block, BlockHeader
 from beemapi.node import Nodes
 from .exceptions import BatchedCallsNotSupported, BlockDoesNotExistsException, BlockWaitTimeExceeded, OfflineHasNoRPCException
-from beemapi.exceptions import NumRetriesReached
+from beemapi.exceptions import NumRetriesReached, UnknownTransaction
 from beemgraphenebase.py23 import py23_bytes
 from beem.instance import shared_blockchain_instance
 from .amount import Amount
@@ -176,7 +176,7 @@ class Blockchain(object):
     """ This class allows to access the blockchain and read data
         from it
 
-        :param Steem blockchain_instance: Steem instance
+        :param Steem/Hive blockchain_instance: Steem or Hive instance
         :param str mode: (default) Irreversible block (``irreversible``) or
             actual head block (``head``)
         :param int max_block_wait_repetition: maximum wait repetition for next block
@@ -251,6 +251,14 @@ class Blockchain(object):
     def is_irreversible_mode(self):
         return self.mode == 'last_irreversible_block_num'
 
+    def is_transaction_existing(self, transaction_id):
+        """ Returns true, if the transaction_id is valid"""
+        try:
+            self.get_transaction(transaction_id)
+            return True
+        except UnknownTransaction:
+            return False
+
     def get_transaction(self, transaction_id):
         """ Returns a transaction from the blockchain
 
diff --git a/beem/blockchaininstance.py b/beem/blockchaininstance.py
index 25ec1e7438a69ba028317bee65701cd00e143efa..3f0889d94e2aafa5ba49976ee6591f23890bb4f1 100644
--- a/beem/blockchaininstance.py
+++ b/beem/blockchaininstance.py
@@ -840,11 +840,11 @@ class BlockChainInstance(object):
             :param string permission: The required permission for
                 signing (active, owner, posting)
             :param TransactionBuilder append_to: This allows to provide an instance of
-                TransactionBuilder (see :func:`Steem.new_tx()`) to specify
+                TransactionBuilder (see :func:`BlockChainInstance.new_tx()`) to specify
                 where to put a specific operation.
 
             .. note:: ``append_to`` is exposed to every method used in the
-                Steem class
+                BlockChainInstance class
 
             .. note::   If ``ops`` is a list of operation, they all need to be
                         signable by the same key! Thus, you cannot combine ops
@@ -852,9 +852,12 @@ class BlockChainInstance(object):
                         posting permission. Neither can you use different
                         accounts for different operations!
 
-            .. note:: This uses :func:`Steem.txbuffer` as instance of
+            .. note:: This uses :func:`BlockChainInstance.txbuffer` as instance of
                 :class:`beem.transactionbuilder.TransactionBuilder`.
                 You may want to use your own txbuffer
+
+            .. note:: when doing sign + broadcast, the trx_id is added to the returned dict
+
         """
         if self.offline:
                 return {}
@@ -887,8 +890,10 @@ class BlockChainInstance(object):
         else:
             # default behavior: sign + broadcast
             self.txbuffer.appendSigner(account, permission)
-            self.txbuffer.sign()
-            return self.txbuffer.broadcast()
+            ret_sign = self.txbuffer.sign()
+            ret = self.txbuffer.broadcast()
+            ret["trx_id"] = ret_sign.id
+            return ret
 
     def sign(self, tx=None, wifs=[], reconstruct_tx=True):
         """ Sign a provided transaction with the provided key(s)
@@ -902,6 +907,8 @@ class BlockChainInstance(object):
                 is already contructed, it will not reconstructed
                 and already added signatures remain
 
+            .. note:: The trx_id is added to the returned dict
+
         """
         if tx:
             txbuffer = TransactionBuilder(tx, blockchain_instance=self)
@@ -909,11 +916,13 @@ class BlockChainInstance(object):
             txbuffer = self.txbuffer
         txbuffer.appendWif(wifs)
         txbuffer.appendMissingSignatures()
-        txbuffer.sign(reconstruct_tx=reconstruct_tx)
-        return txbuffer.json()
+        ret_sign = txbuffer.sign(reconstruct_tx=reconstruct_tx)
+        ret = txbuffer.json()
+        ret["trx_id"] = ret_sign.id
+        return ret
 
     def broadcast(self, tx=None):
-        """ Broadcast a transaction to the Steem network
+        """ Broadcast a transaction to the Hive/Steem network
 
             :param tx tx: Signed transaction to broadcast
 
@@ -1236,7 +1245,7 @@ class BlockChainInstance(object):
         json_meta=None,
         **kwargs
     ):
-        """ Create new account on Steem
+        """ Create new account on Hive/Steem
 
             The brainkey/password can be used to recover all generated keys
             (see :class:`beemgraphenebase.account` for more details.
diff --git a/beem/cli.py b/beem/cli.py
index ed2c9d2c7362e0fed962f8588121ee0d3f6dc265..73b969749991d7a1e953bed032ad39f1d872b476 100644
--- a/beem/cli.py
+++ b/beem/cli.py
@@ -190,6 +190,12 @@ def unlock_token_wallet(stm, sc2, password=None):
         return True
 
 
+def export_trx(tx, export):
+    if export is not None:
+        with open(export, "w", encoding="utf-8") as f:
+            json.dump(tx, f)    
+
+
 @shell(prompt='beempy> ', intro='Starting beempy... (use help to list all commands)', chain=True)
 # @click.group(chain=True)
 @click.option(
@@ -201,7 +207,7 @@ def unlock_token_wallet(stm, sc2, password=None):
 @click.option(
     '--no-wallet', '-p', is_flag=True, default=False, help="Do not load the wallet")
 @click.option(
-    '--unsigned', '-x', is_flag=True, default=False, help="Nothing will be signed")
+    '--unsigned', '-x', is_flag=True, default=False, help="Nothing will be signed, changes the default value of expires to 3600")
 @click.option(
     '--create-link', '-l', is_flag=True, default=False, help="Creates steemconnect/hivesigner links from all broadcast operations")
 @click.option(
@@ -218,7 +224,7 @@ def unlock_token_wallet(stm, sc2, password=None):
     '--token', '-t', is_flag=True, default=False, help="Uses a hivesigner/steemconnect token to broadcast (only broadcast operation with posting permission)")
 @click.option(
     '--expires', '-e', default=30,
-    help='Delay in seconds until transactions are supposed to expire(defaults to 60)')
+    help='Delay in seconds until transactions are supposed to expire(defaults to 30)')
 @click.option(
     '--verbose', '-v', default=3, help='Verbosity')
 @click.version_option(version=__version__)
@@ -236,6 +242,10 @@ def cli(node, offline, no_broadcast, no_wallet, unsigned, create_link, steem, hi
     ch.setFormatter(formatter)
     log.addHandler(ch)
 
+    if unsigned and expires == 30:
+        # Change expires to max duration when setting unsigned
+        expires = 3600
+
     keys_list = []
     autoconnect = False
     if keys and keys != "":
@@ -1056,7 +1066,8 @@ def listaccounts(role, max_account_index, max_sequence):
 @click.argument('post', nargs=1)
 @click.option('--weight', '-w', help='Vote weight (from 0.1 to 100.0)')
 @click.option('--account', '-a', help='Voter account name')
-def upvote(post, account, weight):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def upvote(post, account, weight, export):
     """Upvote a post/comment
 
         POST is @author/permlink
@@ -1087,13 +1098,16 @@ def upvote(post, account, weight):
     except exceptions.VotingInvalidOnArchivedPost:
         print("Post/Comment is older than 7 days! Did not upvote.")
         tx = {}
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
+
 @cli.command()
 @click.argument('post', nargs=1)
-@click.option('--account', '-a', help='Voter account name')
-def delete(post, account):
+@click.option('--account', '-a', help='Account name')
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def delete(post, account, export):
     """delete a post/comment
 
         POST is @author/permlink
@@ -1116,15 +1130,17 @@ def delete(post, account):
     except exceptions.VotingInvalidOnArchivedPost:
         print("Could not delete post.")
         tx = {}
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
 
 @cli.command()
 @click.argument('post', nargs=1)
-@click.option('--account', '-a', help='Voter account name')
+@click.option('--account', '-a', help='Downvoter account name')
 @click.option('--weight', '-w', default=100, help='Downvote weight (from 0.1 to 100.0)')
-def downvote(post, account, weight):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def downvote(post, account, weight, export):
     """Downvote a post/comment
 
         POST is @author/permlink
@@ -1153,6 +1169,7 @@ def downvote(post, account, weight):
     except exceptions.VotingInvalidOnArchivedPost:
         print("Post/Comment is older than 7 days! Did not downvote.")
         tx = {}
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -1163,8 +1180,9 @@ def downvote(post, account, weight):
 @click.argument('asset', nargs=1, callback=asset_callback)
 @click.argument('memo', nargs=1, required=False)
 @click.option('--account', '-a', help='Transfer from this account')
-def transfer(to, amount, asset, memo, account):
-    """Transfer SBD/HD STEEM/HIVE"""
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def transfer(to, amount, asset, memo, account, export):
+    """Transfer SBD/HBD or STEEM/HIVE"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
         stm.rpc.rpcconnect()
@@ -1180,6 +1198,7 @@ def transfer(to, amount, asset, memo, account):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -1187,8 +1206,9 @@ def transfer(to, amount, asset, memo, account):
 @cli.command()
 @click.argument('amount', nargs=1)
 @click.option('--account', '-a', help='Powerup from this account')
-@click.option('--to', help='Powerup this account', default=None)
-def powerup(amount, account, to):
+@click.option('--to', '-t', help='Powerup this account', default=None)
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def powerup(amount, account, to, export):
     """Power up (vest STEEM/HIVE as STEEM/HIVE POWER)"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -1207,6 +1227,7 @@ def powerup(amount, account, to):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -1214,7 +1235,8 @@ def powerup(amount, account, to):
 @cli.command()
 @click.argument('amount', nargs=1)
 @click.option('--account', '-a', help='Powerup from this account')
-def powerdown(amount, account):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def powerdown(amount, account, export):
     """Power down (start withdrawing VESTS from Steem POWER)
 
         amount is in VESTS
@@ -1236,6 +1258,7 @@ def powerdown(amount, account):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -1244,7 +1267,8 @@ def powerdown(amount, account):
 @click.argument('amount', nargs=1)
 @click.argument('to_account', nargs=1)
 @click.option('--account', '-a', help='Delegate from this account')
-def delegate(amount, to_account, account):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def delegate(amount, to_account, account, export):
     """Delegate (start delegating VESTS to another account)
 
         amount is in VESTS / Steem
@@ -1271,6 +1295,7 @@ def delegate(amount, to_account, account):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -1281,7 +1306,8 @@ def delegate(amount, to_account, account):
 @click.option('--account', '-a', help='Powerup from this account')
 @click.option('--auto_vest', help='Set to true if the from account should receive the VESTS as'
               'VESTS, or false if it should receive them as STEEM/HIVE.', is_flag=True)
-def powerdownroute(to, percentage, account, auto_vest):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def powerdownroute(to, percentage, account, auto_vest, export):
     """Setup a powerdown route"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -1296,13 +1322,15 @@ def powerdownroute(to, percentage, account, auto_vest):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
 @cli.command()
 @click.argument('new_recovery_account', nargs=1)
 @click.option('--account', '-a', help='Change the recovery account from this account')
-def changerecovery(new_recovery_account, account):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def changerecovery(new_recovery_account, account, export):
     """Changes the recovery account with the owner key (needs 30 days to be active)"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -1334,6 +1362,7 @@ def changerecovery(new_recovery_account, account):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -1341,7 +1370,8 @@ def changerecovery(new_recovery_account, account):
 @cli.command()
 @click.argument('amount', nargs=1)
 @click.option('--account', '-a', help='Powerup from this account')
-def convert(amount, account):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def convert(amount, account, export):
     """Convert SBD/HBD to Steem/Hive (takes a week to settle)"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -1360,6 +1390,7 @@ def convert(amount, account):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -1639,12 +1670,13 @@ def permissions(account):
 @click.argument('foreign_account', nargs=1, required=False)
 @click.option('--permission', default="posting", help='The permission to grant (defaults to "posting")')
 @click.option('--account', '-a', help='The account to allow action for')
-@click.option('--weight', help='The weight to use instead of the (full) threshold. '
+@click.option('--weight', '-w', help='The weight to use instead of the (full) threshold. '
               'If the weight is smaller than the threshold, '
               'additional signatures are required')
-@click.option('--threshold', help='The permission\'s threshold that needs to be reached '
+@click.option('--threshold', '-t', help='The permission\'s threshold that needs to be reached '
               'by signatures to be able to interact')
-def allow(foreign_account, permission, account, weight, threshold):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def allow(foreign_account, permission, account, weight, threshold, export):
     """Allow an account/key to interact with your account
 
         foreign_account: The account or key that will be allowed to interact with account.
@@ -1673,17 +1705,19 @@ def allow(foreign_account, permission, account, weight, threshold):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
 
 @cli.command()
 @click.argument('foreign_account', nargs=1, required=False)
-@click.option('--permission', default="posting", help='The permission to grant (defaults to "posting")')
+@click.option('--permission', '-p', default="posting", help='The permission to grant (defaults to "posting")')
 @click.option('--account', '-a', help='The account to disallow action for')
-@click.option('--threshold', help='The permission\'s threshold that needs to be reached '
+@click.option('--threshold', '-t', help='The permission\'s threshold that needs to be reached '
               'by signatures to be able to interact')
-def disallow(foreign_account, permission, account, threshold):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def disallow(foreign_account, permission, account, threshold, export):
     """Remove allowance an account/key to interact with your account"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -1707,6 +1741,7 @@ def disallow(foreign_account, permission, account, threshold):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -1715,7 +1750,8 @@ def disallow(foreign_account, permission, account, threshold):
 @click.argument('creator', nargs=1, required=True)
 @click.option('--fee', help='When fee is 0 (default) a subsidized account is claimed and can be created later with create_claimed_account', default=0.0)
 @click.option('--number', '-n', help='Number of subsidized accounts to be claimed (default = 1), when fee = 0 STEEM', default=1)
-def claimaccount(creator, fee, number):
+@click.option('--export', '-e', help='When set, transaction is stored in a file (should be used with number = 1)')
+def claimaccount(creator, fee, number, export):
     """Claim account for claimed account creation."""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -1758,6 +1794,7 @@ def claimaccount(creator, fee, number):
     else:
         tx = stm.claim_account(creator, fee=fee)
     if tx is not None:
+        export_trx(tx, export)
         tx = json.dumps(tx, indent=4)
         print(tx)
 
@@ -1769,7 +1806,8 @@ def claimaccount(creator, fee, number):
 @click.option('--posting', help='posting public key - when not given, a passphrase is used to create keys.')
 @click.option('--memo', help='Memo public key - when not given, a passphrase is used to create keys.')
 @click.option('--import-pub', '-i', help='Load public keys from file.')
-def changekeys(account, owner, active, posting, memo, import_pub):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def changekeys(account, owner, active, posting, memo, import_pub, export):
     """Changes all keys for the specified account 
     Keys are given in their public form.
     Asks for the owner key for broadcasting the op to the chain."""
@@ -1821,6 +1859,7 @@ def changekeys(account, owner, active, posting, memo, import_pub):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -1835,7 +1874,8 @@ def changekeys(account, owner, active, posting, memo, import_pub):
 @click.option('--wif', '-w', help='Defines how many times the password is replaced by its WIF representation for password based keys (default = 0).', default=0)
 @click.option('--create-claimed-account', '-c', help='Instead of paying the account creation fee a subsidized account is created.', is_flag=True, default=False)
 @click.option('--import-pub', '-i', help='Load public keys from file.')
-def newaccount(accountname, account, owner, active, memo, posting, wif, create_claimed_account, import_pub):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def newaccount(accountname, account, owner, active, memo, posting, wif, create_claimed_account, import_pub, export):
     """Create a new account
        Default setting is that a fee is payed for account creation
        Use --create-claimed-account for free account creation
@@ -1893,6 +1933,7 @@ def newaccount(accountname, account, owner, active, memo, posting, wif, create_c
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -1902,7 +1943,8 @@ def newaccount(accountname, account, owner, active, memo, posting, wif, create_c
 @click.argument('value', nargs=1, required=False)
 @click.option('--account', '-a', help='setprofile as this user')
 @click.option('--pair', '-p', help='"Key=Value" pairs', multiple=True)
-def setprofile(variable, value, account, pair):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def setprofile(variable, value, account, pair, export):
     """Set a variable in an account\'s profile"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -1933,6 +1975,7 @@ def setprofile(variable, value, account, pair):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -1940,7 +1983,8 @@ def setprofile(variable, value, account, pair):
 @cli.command()
 @click.argument('variable', nargs=-1, required=True)
 @click.option('--account', '-a', help='delprofile as this user')
-def delprofile(variable, account):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def delprofile(variable, account, export):
     """Delete a variable in an account\'s profile"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -1961,6 +2005,7 @@ def delprofile(variable, account):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -2027,7 +2072,8 @@ def importaccount(account, roles):
 @cli.command()
 @click.option('--account', '-a', help='The account to updatememokey action for')
 @click.option('--key', help='The new memo key')
-def updatememokey(account, key):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def updatememokey(account, key, export):
     """Update an account\'s memo key"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -2050,6 +2096,7 @@ def updatememokey(account, key):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -2057,7 +2104,8 @@ def updatememokey(account, key):
 @cli.command()
 @click.argument('authorperm', nargs=1)
 @click.argument('beneficiaries', nargs=-1)
-def beneficiaries(authorperm, beneficiaries):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def beneficiaries(authorperm, beneficiaries, export):
     """Set beneficaries"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -2087,6 +2135,7 @@ def beneficiaries(authorperm, beneficiaries):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -2438,7 +2487,8 @@ def createpost(markdown_file, account, title, tags, community, beneficiaries, pe
 @click.option('--max-accepted-payout', '-m', help='Default is 1000000.000 [SBD]')
 @click.option('--no-parse-body', '-n', help='Disable parsing of links, tags and images', is_flag=True, default=False)
 @click.option('--no-patch-on-edit', '-e', help='Disable patch posting on edits (when the permlink already exists)', is_flag=True, default=False)
-def post(markdown_file, account, title, permlink, tags, reply_identifier, community, canonical_url, beneficiaries, percent_steem_dollars, max_accepted_payout, no_parse_body, no_patch_on_edit):
+@click.option('--export', help='When set, transaction is stored in a file')
+def post(markdown_file, account, title, permlink, tags, reply_identifier, community, canonical_url, beneficiaries, percent_steem_dollars, max_accepted_payout, no_parse_body, no_patch_on_edit, export):
     """broadcasts a post/comment. All image links which links to a file will be uploaded.
     The yaml header can contain:
     
@@ -2610,6 +2660,7 @@ def post(markdown_file, account, title, permlink, tags, reply_identifier, commun
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -2619,7 +2670,8 @@ def post(markdown_file, account, title, permlink, tags, reply_identifier, commun
 @click.argument('body', nargs=1)
 @click.option('--account', '-a', help='Account are you posting from')
 @click.option('--title', '-t', help='Title of the post')
-def reply(authorperm, body, account, title):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def reply(authorperm, body, account, title, export):
     """replies to a comment"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -2638,6 +2690,7 @@ def reply(authorperm, body, account, title):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -2645,7 +2698,8 @@ def reply(authorperm, body, account, title):
 @cli.command()
 @click.argument('witness', nargs=1)
 @click.option('--account', '-a', help='Your account')
-def approvewitness(witness, account):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def approvewitness(witness, account, export):
     """Approve a witnesses"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -2660,6 +2714,7 @@ def approvewitness(witness, account):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -2667,7 +2722,8 @@ def approvewitness(witness, account):
 @cli.command()
 @click.argument('witness', nargs=1)
 @click.option('--account', '-a', help='Your account')
-def disapprovewitness(witness, account):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def disapprovewitness(witness, account, export):
     """Disapprove a witnesses"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -2682,6 +2738,7 @@ def disapprovewitness(witness, account):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -2689,7 +2746,8 @@ def disapprovewitness(witness, account):
 @cli.command()
 @click.argument('proxy', nargs=1)
 @click.option('--account', '-a', help='Your account')
-def setproxy(proxy, account):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def setproxy(proxy, account, export):
     """Set your witness/proposal system proxy"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -2704,13 +2762,15 @@ def setproxy(proxy, account):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
 
 @cli.command()
 @click.option('--account', '-a', help='Your account')
-def delproxy(account):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def delproxy(account, export):
     """Delete your witness/proposal system proxy"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -2725,6 +2785,7 @@ def delproxy(account):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -2760,7 +2821,7 @@ def sign(file, outfile):
 
 
 @cli.command()
-@click.option('--file', help='Load transaction from file. If "-", read from stdin (defaults to "-")')
+@click.option('--file', '-f', help='Load transaction from file. If "-", read from stdin (defaults to "-")')
 def broadcast(file):
     """broadcast a signed transaction"""
     stm = shared_blockchain_instance()
@@ -2993,7 +3054,8 @@ def orderbook(chart, limit, show_date, width, height, ascii):
 @click.argument('price', nargs=1, required=False)
 @click.option('--account', '-a', help='Buy with this account (defaults to "default_account")')
 @click.option('--orderid', help='Set an orderid')
-def buy(amount, asset, price, account, orderid):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def buy(amount, asset, price, account, orderid, export):
     """Buy STEEM/HIVE or SBD/HBD from the internal market
 
         Limit buy price denoted in (SBD per STEEM or HBD per HIVE)
@@ -3030,6 +3092,7 @@ def buy(amount, asset, price, account, orderid):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -3040,7 +3103,8 @@ def buy(amount, asset, price, account, orderid):
 @click.argument('price', nargs=1, required=False)
 @click.option('--account', '-a', help='Sell with this account (defaults to "default_account")')
 @click.option('--orderid', help='Set an orderid')
-def sell(amount, asset, price, account, orderid):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def sell(amount, asset, price, account, orderid, export):
     """Sell STEEM/HIVE or SBD/HBD from the internal market
 
         Limit sell price denoted in (SBD per STEEM) or (HBD per HIVE)
@@ -3076,6 +3140,7 @@ def sell(amount, asset, price, account, orderid):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -3083,7 +3148,8 @@ def sell(amount, asset, price, account, orderid):
 @cli.command()
 @click.argument('orderid', nargs=1)
 @click.option('--account', '-a', help='Sell with this account (defaults to "default_account")')
-def cancel(orderid, account):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def cancel(orderid, account, export):
     """Cancel order in the internal market"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -3099,6 +3165,7 @@ def cancel(orderid, account):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -3152,7 +3219,8 @@ def reblog(identifier, account):
 @click.argument('follow', nargs=1)
 @click.option('--account', '-a', help='Follow from this account')
 @click.option('--what', help='Follow these objects (defaults to ["blog"])', default=["blog"])
-def follow(follow, account, what):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def follow(follow, account, what, export):
     """Follow another account"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -3169,6 +3237,7 @@ def follow(follow, account, what):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -3177,7 +3246,8 @@ def follow(follow, account, what):
 @click.argument('mute', nargs=1)
 @click.option('--account', '-a', help='Mute from this account')
 @click.option('--what', help='Mute these objects (defaults to ["ignore"])', default=["ignore"])
-def mute(mute, account, what):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def mute(mute, account, what, export):
     """Mute another account"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -3194,6 +3264,7 @@ def mute(mute, account, what):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -3201,7 +3272,8 @@ def mute(mute, account, what):
 @cli.command()
 @click.argument('unfollow', nargs=1)
 @click.option('--account', '-a', help='UnFollow/UnMute from this account')
-def unfollow(unfollow, account):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def unfollow(unfollow, account, export):
     """Unfollow/Unmute another account"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -3216,6 +3288,7 @@ def unfollow(unfollow, account):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -3227,7 +3300,8 @@ def unfollow(unfollow, account):
 @click.option('--sbd_interest_rate', help='SBD interest rate in percent')
 @click.option('--url', help='Witness URL')
 @click.option('--signing_key', help='Signing Key')
-def witnessupdate(witness, maximum_block_size, account_creation_fee, sbd_interest_rate, url, signing_key):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def witnessupdate(witness, maximum_block_size, account_creation_fee, sbd_interest_rate, url, signing_key, export):
     """Change witness properties"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -3250,13 +3324,15 @@ def witnessupdate(witness, maximum_block_size, account_creation_fee, sbd_interes
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
 
 @cli.command()
 @click.argument('witness', nargs=1)
-def witnessdisable(witness):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def witnessdisable(witness, export):
     """Disable a witness"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -3275,6 +3351,7 @@ def witnessdisable(witness):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -3282,7 +3359,8 @@ def witnessdisable(witness):
 @cli.command()
 @click.argument('witness', nargs=1)
 @click.argument('signing_key', nargs=1)
-def witnessenable(witness, signing_key):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def witnessenable(witness, signing_key, export):
     """Enable a witness"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -3298,6 +3376,7 @@ def witnessenable(witness, signing_key):
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -3309,7 +3388,8 @@ def witnessenable(witness, signing_key):
 @click.option('--account_creation_fee', help='Account creation fee', default=0.1)
 @click.option('--sbd_interest_rate', help='SBD interest rate in percent', default=0.0)
 @click.option('--url', help='Witness URL', default="")
-def witnesscreate(witness, pub_signing_key, maximum_block_size, account_creation_fee, sbd_interest_rate, url):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def witnesscreate(witness, pub_signing_key, maximum_block_size, account_creation_fee, sbd_interest_rate, url, export):
     """Create a witness"""
     stm = shared_blockchain_instance()
     if stm.rpc is not None:
@@ -3330,6 +3410,7 @@ def witnesscreate(witness, pub_signing_key, maximum_block_size, account_creation
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -4204,7 +4285,8 @@ def pending(accounts, only_sum, post, comment, curation, length, author, permlin
 @click.option('--claim_all_steem', help='Claim all STEEM/HIVE, overwrites reward_steem', is_flag=True)
 @click.option('--claim_all_sbd', help='Claim all SBD/HBD, overwrites reward_sbd', is_flag=True)
 @click.option('--claim_all_vests', help='Claim all VESTS, overwrites reward_vests', is_flag=True)
-def claimreward(account, reward_steem, reward_sbd, reward_vests, claim_all_steem, claim_all_sbd, claim_all_vests):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def claimreward(account, reward_steem, reward_sbd, reward_vests, claim_all_steem, claim_all_sbd, claim_all_vests, export):
     """Claim reward balances
 
         By default, this will claim ``all`` outstanding balances.
@@ -4236,6 +4318,7 @@ def claimreward(account, reward_steem, reward_sbd, reward_vests, claim_all_steem
         tx = stm.steemconnect.url_from_tx(tx)
     elif stm.unsigned and stm.nobroadcast and stm.hivesigner is not None:
         tx = stm.hivesigner.url_from_tx(tx)
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -4245,7 +4328,8 @@ def claimreward(account, reward_steem, reward_sbd, reward_vests, claim_all_steem
 @click.argument('json_data', nargs=-1)
 @click.option('--account', '-a', help='The account which broadcasts the custom_json')
 @click.option('--active', '-t', help='When set, the active key is used for broadcasting', is_flag=True, default=False)
-def customjson(jsonid, json_data, account, active):
+@click.option('--export', '-e', help='When set, transaction is stored in a file')
+def customjson(jsonid, json_data, account, active, export):
     """Broadcasts a custom json
     
         First parameter is the cusom json id, the second field is a json file or a json key value combination
@@ -4296,6 +4380,7 @@ def customjson(jsonid, json_data, account, active):
         tx = stm.custom_json(jsonid, data, required_auths=[account])
     else:
         tx = stm.custom_json(jsonid, data, required_posting_auths=[account])
+    export_trx(tx, export)
     tx = json.dumps(tx, indent=4)
     print(tx)
 
@@ -4509,6 +4594,17 @@ def info(objects):
                 print(t)
             else:
                 print("Post now known" % obj)
+        elif re.match(r"^[a-zA-Z0-9\_]{40}$", obj):
+            b = Blockchain(blockchain_instance=stm)
+            trx = b.get_transaction(obj)
+            t = PrettyTable(["Key", "Value"])
+            t.align = "l"
+            for key in trx:
+                value = trx[key]
+                if key in ["operations", "signatures"]:
+                    value = json.dumps(value, indent=4)
+                t.add_row([key, value])
+            print(t)            
         else:
             print("Couldn't identify object to read")
 
@@ -4542,6 +4638,67 @@ def userdata(account, signing_account):
     print(t)
 
 
+@cli.command()
+@click.argument('account', nargs=1, required=False)
+@click.option('--number', '-n', help='Defines how many ops should be printed (default=10)', default=10)
+@click.option('--order', help='Defines how many ops should be printed (default=-1)', default=-1)
+@click.option('--virtual-ops', '-v', help='When set, virtual ops are also shown', is_flag=True, default=False)
+@click.option('--only-ops', '-o', help='Included komma seperated list of op types, which limits the shown operations. When set, virtual-ops is always set to true')
+@click.option('--exclude-ops', '-e', help='Excluded komma seperated list of op types, which limits the shown operations.')
+@click.option('--json-file', '-j', help='When set, the results are written into a json file')
+def history(account, number, order, virtual_ops, only_ops, exclude_ops, json_file):
+    """ Returns account history operations as table
+
+    """
+    stm = shared_blockchain_instance()
+    if stm.rpc is not None:
+        stm.rpc.rpcconnect()
+    if not account:
+        if "default_account" in stm.config:
+            account = stm.config["default_account"]
+    account = Account(account, blockchain_instance=stm)
+    t = PrettyTable(["Nr", "Hist op"])
+    t.align = "l"
+    cnt = 0
+    batch_size = 1000
+    if batch_size > int(number) + 1 and int(number) > 0:
+        batch_size = int(number) + 1
+    if only_ops is None:
+        only_ops = []
+    else:
+        only_ops = only_ops.split(",")
+    if exclude_ops is None:
+        exclude_ops = []
+    else:
+        exclude_ops = exclude_ops.split(",")
+    if len(only_ops) > 0:
+        virtual_ops = True
+    data = []
+    if int(order) == -1:
+        hist = account.history_reverse(batch_size=batch_size, only_ops=only_ops, exclude_ops=exclude_ops)
+    else:
+        hist = account.history(batch_size=batch_size, only_ops=only_ops, exclude_ops=exclude_ops)
+    for h in hist:
+        if h["virtual_op"] == 1 and not virtual_ops:
+            continue
+        
+        cnt += 1
+        if cnt > int(number) and int(number) > 0:
+            break
+        if json_file is not None:
+            data.append(h)
+        else:
+            # if key in ["operations", "signatures"]:
+            value = json.dumps(h, indent=4)
+            t.add_row([str(cnt), value])
+
+    if json_file is not None:
+        with open(json_file, "w", encoding="utf-8") as f:
+            json.dump(data, f)
+    else:
+        print(t)
+
+
 @cli.command()
 @click.argument('account', nargs=1, required=False)
 @click.option('--signing-account', '-s', help='Signing account, when empty account is used.')
diff --git a/beem/transactionbuilder.py b/beem/transactionbuilder.py
index 7fa915c333cb6fb02eb9d7cf2f164d225aa16c9f..45bf7203c06340699a20e03448642b0a8a9c2b01 100644
--- a/beem/transactionbuilder.py
+++ b/beem/transactionbuilder.py
@@ -11,7 +11,6 @@ from binascii import unhexlify
 from beemgraphenebase.py23 import bytes_types, integer_types, string_types, text_type
 from .account import Account
 from .utils import formatTimeFromNow
-from .steemconnect import SteemConnect
 from beembase.objects import Operation
 from beemgraphenebase.account import PrivateKey, PublicKey
 from beembase.signedtransactions import Signed_Transaction
@@ -473,7 +472,7 @@ class TransactionBuilder(dict):
 
         return ret
 
-    def broadcast(self, max_block_age=-1):
+    def broadcast(self, max_block_age=-1, trx_id=True):
         """ Broadcast a transaction to the steem network
             Returns the signed transaction and clears itself
             after broadast
@@ -482,11 +481,14 @@ class TransactionBuilder(dict):
 
             :param int max_block_age: parameter only used
                 for appbase ready nodes
+            :param bool trx_id: When True, trx_id is return
 
         """
         # Cannot broadcast an empty transaction
         if not self._is_signed():
-            self.sign()
+            sign_ret = self.sign()
+        else:
+            sign_ret = None
 
         if "operations" not in self or not self["operations"]:
             return
@@ -524,7 +526,8 @@ class TransactionBuilder(dict):
             # log.error("Could Not broadcasting anything!")
             self.clear()
             raise e
-
+        if sign_ret is not None and "trx_id" not in ret and trx_id:
+            ret["trx_id"] = sign_ret.id
         self.clear()
         return ret
 
diff --git a/beemapi/exceptions.py b/beemapi/exceptions.py
index d72962fdc35b9fe71225ac777508d9f05acd6ac3..5c758ae68c6de4dcd04de30ac9b15f37992a5068 100644
--- a/beemapi/exceptions.py
+++ b/beemapi/exceptions.py
@@ -110,3 +110,7 @@ class TimeoutException(Exception):
 
 class VotedBeforeWaitTimeReached(Exception):
     pass
+
+
+class UnknownTransaction(Exception):
+    pass
diff --git a/beemapi/noderpc.py b/beemapi/noderpc.py
index e78db37752ae95001f85723c7ec44248d24d27e3..ba8e671fa2b1b19fa4e7a7304ac75096ae52bd54 100644
--- a/beemapi/noderpc.py
+++ b/beemapi/noderpc.py
@@ -115,6 +115,8 @@ class NodeRPC(GrapheneRPC):
             raise exceptions.NoMethodWithName(msg)
         elif re.search("Could not find method", msg):
             raise exceptions.NoMethodWithName(msg)
+        elif re.search("Unknown Transaction", msg):
+            raise exceptions.UnknownTransaction(msg)
         elif re.search("Could not find API", msg):
             if self._check_api_name(msg):
                 if self.nodes.working_nodes_count > 1 and self.nodes.num_retries > -1: